Source code for plate_model_manager.present_day_rasters

import glob
import json
import logging
import os
from hashlib import sha256
from typing import Dict

import requests

from .network_requests import fetch_file
from .utils import download, misc

DEFAULT_PRESENT_DAY_RASTERS_MANIFEST = (
    "https://repo.gplates.org/webdav/pmm/present_day_rasters.json"
)
logger = logging.getLogger("pmm")


class RasterNameNotFound(Exception):
    pass


[docs] class PresentDayRasterManager: """Manage the present-day rasters."""
[docs] def __init__(self, data_dir="present-day-rasters", raster_manifest=None): """Constructor. Create a :class:`PresentDayRasterManager` instance. :param raster_manifest: The URL to a ``present_day_rasters.json`` metadata file. Normally you don't need to provide this parameter unless you would like to setup your own present-day raster server. :param data_dir: The path to a folder to save the present-day raster files. """ if not raster_manifest: self.raster_manifest = DEFAULT_PRESENT_DAY_RASTERS_MANIFEST else: self.raster_manifest = raster_manifest self._rasters = None self.data_dir = data_dir # check if the model manifest file is a local file if os.path.isfile(self.raster_manifest): with open(self.raster_manifest) as f: self._rasters = json.load(f) elif self.raster_manifest.startswith( "http://" ) or self.raster_manifest.startswith("https://"): # try the http(s) url try: r = requests.get(self.raster_manifest) self._rasters = r.json() except requests.exceptions.ConnectionError: raise Exception( f"Unable to fetch {self.raster_manifest}. " + "No network connection or invalid URL!" ) else: raise Exception( f"The model_manifest '{self.raster_manifest}' should be either a local file path or a http(s) URL." )
@property def rasters(self) -> Dict: """The metadata of rasters.""" if self._rasters is not None: return self._rasters else: raise Exception( "The self._rasters is None. This should not happen. Something Extraordinary must have happened." ) @rasters.setter def rasters(self, var) -> None: self._rasters = var
[docs] def set_data_dir(self, data_dir): """Set a new data folder to save the present-day rasters.""" self.data_dir = data_dir
[docs] def list_present_day_rasters(self): """Return a list of available present-day rasters.""" return [name for name in self.rasters]
def _check_raster_avail(self, _name: str): """Check if the raster name is in raster configuration.""" name = _name.lower() if not name in self.rasters: raise RasterNameNotFound(f"Raster {name} is not found in {self.rasters}.") return name
[docs] def is_wms(self, _name: str, check_raster_avail_flag=True): """Return ``True`` if the raster is served by ``Web Map Service``, otherwise ``False`` :param _name: The raster name of interest. :type _name: str :param check_raster_avail_flag: If the flag is ``True``, validate the raster name against the raster configuration. :type check_raster_avail_flag: bool """ if check_raster_avail_flag: name = self._check_raster_avail(_name) else: name = _name.lower() if ( isinstance(self.rasters[name], dict) and "service" in self.rasters[name] and self.rasters[name]["service"] == "WMS" ): return True else: return False
[docs] def get_raster( self, _name: str, width=1800, height=800, bbox=[-180, -80, 180, 80], large_file_hint=True, ): """Download a raster file by name, save the raster file in ``self.data_dir`` and return the local path to the raster file. Call :meth:`list_present_day_rasters()` to see a list of available present-day raster names. :param _name: The raster name of interest. :type _name: str :return: The local path to the downloaded raster file. :rtype: str """ name = self._check_raster_avail(_name) is_wms_flag = self.is_wms(name, check_raster_avail_flag=False) if not is_wms_flag: downloader = download.FileDownloader( self.rasters[name], f"{self.data_dir}/{name}/.metadata.json", f"{self.data_dir}/{name}/", large_file_hint=large_file_hint, ) # only re-download when necessary if downloader.check_if_file_need_update(): downloader.download_file_and_update_metadata() else: if downloader.check_if_expire_date_need_update(): # update the expiry date downloader.update_metadata() logger.debug( f"The local raster file {self.data_dir}/{name} is still good. Will not download again at this moment." ) files = glob.glob(f"{self.data_dir}/{name}/*") if len(files) == 0: raise Exception(f"Failed to get raster {name}") if len(files) > 1: misc.print_warning( f"Multiple raster files have been detected.{files}. Return the first one found {files[0]}." ) return files[0] else: server_url = self.rasters[name]["server_url"] version = self.rasters[name]["version"] layers = self.rasters[name]["layers"] if self.rasters[name]["hillshade_layer"]: layers.append(self.rasters[name]["hillshade_layer"]) styles = self.rasters[name]["styles"] if self.rasters[name]["hillshade_style"]: styles.append(self.rasters[name]["hillshade_style"]) format = "image/geotiff" url = ( f"{server_url}/wms?service=WMS&version={version}&request=GetMap&layers={','.join(layers)}" + f"&bbox={bbox[0]},{bbox[1]},{bbox[2]},{bbox[3]}&width={width}&height={height}&srs=EPSG:4326" + f"&styles={','.join(styles)}&format={format}" ) filepath = ( f"{self.data_dir}/{name}/{sha256(url.encode('utf-8')).hexdigest()}" ) if not os.path.isfile(f"{filepath}/{name}.tiff"): fetch_file( url, f"{self.data_dir}/{name}/{sha256(url.encode('utf-8')).hexdigest()}", filename=f"{name}.tiff", ) return f"{filepath}/{name}.tiff"