#!/usr/bin/env python # coding: utf-8 # # Building ERDDAP Datasets # # This notebook documents the process of creating XML fragments # for nowcast system run results files # for inclusion in `/opt/tomcat/content/erddap/datasets.xml` # on the `skookum` ERDDAP server instance. # # The contents are a combination of: # # * instructions for using the # `GenerateDatasetsXml.sh` and `DasDds.sh` tools found in the # `/opt/tomcat/webapps/erddap/WEB-INF/` directory # * instructions for forcing the server to update the datasets collection # via the `/results/erddap/flags/` directory # * code and metadata to transform the output of `GenerateDatasetsXml.sh` # into XML fragments that are ready for inclusion in `/opt/tomcat/content/erddap/datasets.xml` # In[1]: from collections import OrderedDict from lxml import etree # **NOTE** # # The next cell mounts the `/results` filesystem on `skookum` locally. # It is intended for use if when this notebook is run on a laptop # or other non-Waterhole machine that has `sshfs` installed # and a mount point for `/results` available in its root filesystem. # # Don't execute the cell if that doesn't describe your situation. # In[2]: get_ipython().system('sshfs skookum:/results /results') # The `metadata` dictionary below contains information for dataset # attribute tags whose values need to be changed, # or that need to be added for all datasets. # # The keys are the dataset attribute names. # # The values are dicts containing a required `text` item # and perhaps an optional `after` item. # # The value associated with the `text` key is the text content # for the attribute tag. # # When present, # the value associated with the `after` key is the name # of the dataset attribute after which a new attribute tag # containing the `text` value is to be inserted. # In[3]: metadata = OrderedDict([ ('coverage_content_type', { 'text': 'modelResult', 'after': 'cdm_data_type', }), ('infoUrl', { 'text': 'https://salishsea-meopar-tools.readthedocs.org/en/latest/results_server/index.html#salish-sea-model-results', }), ('institution', {'text': 'UBC EOAS'}), ('institution_fullname', { 'text': 'Earth, Ocean & Atmospheric Sciences, University of British Columbia', 'after': 'institution', }), ('license', { 'text': '''The SalishSeaCast NEMO model results are copyright 2013 – present by the SalishSeaCast Project Contributors and The University of British Columbia. They are licensed under the Apache License, Version 2.0. http://www.apache.org/licenses/LICENSE-2.0''', }), ('project', { 'text':'SalishSeaCast NEMO Model', 'after': 'title', }), ('creator_name', { 'text': 'SalishSeaCast Project Contributors', 'after': 'project', }), ('creator_email', { 'text': 'sallen@eoas.ubc.ca', 'after': 'creator_name', }), ('creator_url', { 'text': 'https://salishsea-meopar-docs.readthedocs.org/', 'after': 'creator_email', }), ('acknowledgement', { 'text': 'MEOPAR, ONC, Compute Canada', 'after': 'creator_url', }), ('drawLandMask', { 'text': 'over', 'after': 'acknowledgement', }), ]) # The `datasets` dictionary below provides the content # for the dataset `title` and `summary` attributes. # # The `title` attribute content appears in the the datasets list table # (among other places). # It should be `<`80 characters long, # and note that only the 1st 40 characters will appear in the table. # # The `summary` atribute content appears # (among other places) # when a user hovers the cursor over the `?` icon beside the `title` # content in the datasets list table. # The text that is inserted into the `summary` attribute tag # by code later in this notebook is the # `title` content followed by the `summary` content, # separated by a blank line. # # The keys of the `datasets` dict are the `datasetID` strings that # are used in many places by the ERDDAP server. # They are structured as follows: # # * `ubc` to indicate that the dataset was produced at UBC # * `SS` to indicate that the dataset is a product of the SalishSeaCast NEMO model # * `n` to indicate that the dataset is from a nowcast run; # other possibilities include `f` for forecast, # `f2` for forecast2 (aka preliminary forecast), # `ng` for nowcat-green, # `a` for atmospheric forcing. # * a description of the dataset variables; e.g. `PointAtkinsonSSH` or `3DuVelocity` # * the time interval of values in the dataset; e.g. `15m`, `1h`, `1d` # * the dataset version; e.g. `V1` # # So: # # * `ubcSSnPointAtkinsonSSH15mV1` is the version 1 dataset of 15 minute averaged sea surface height values at Point Atkinson from `PointAtkinson.nc` output files # * `ubcSSn3DwVelocity1hV2` is the version 2 dataset of 1 hr averaged vertical (w) velocity values over the entire domain from `SalishSea_1h_*_grid_W.nc` output files # * `ubcSSnSurfaceTracers1dV1` is the version 1 dataset of daily averaged surface tracer values over the entire domain from `SalishSea_1d_*_grid_T.nc` output files # # The dataset version part of the `datasetID` is used to indicate changes in the variables # contained in the dataset. # For example, # the transition from the `ubcSSn3DwVelocity1hV1` to the `ubcSSn3DwVelocity1hV2` dataset # occurred on 24-Jan-2016 when we started to output vertical eddy viscosity and diffusivity # values at the `w` grid points. # # All datasets start at `V1` and their `summary` ends with a notation about the variables # that they contain; e.g. # ``` # v1: wVelocity variable # ``` # When the a dataset version is incremented a line describing the change is added # to the end of its `summary`; e.g. # ``` # v1: wVelocity variable # v2: Added eddy viscosity & diffusivity variables ve_eddy_visc & ve_eddy_diff # ``` # In[21]: datasets = { 'ubcSSnBathymetry2V1' :{ 'type': 'geolocation bathymetry', 'title': 'SalishSeaCast NEMO model Grid, Geo-location and Bathymetry, v1', 'summary':'''Longitude, latitude, and bathymetry of the SalishSeaCast NEMO model grid. The bathymetry values are those calculated by NEMO from the input bathymetry file. NEMO modifies the input bathymetry to remove isolated holes, and too-small partial steps. The model grid includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. v1: longitude, latitude and bathymetry variables''', 'fileNameRegex': '.*SalishSea2_NEMO_bathy\.nc$' }, 'ubcSSn2DMeshMask2V1': { 'type': 'geolocation bathymetry', 'title': 'SalishSeaCast NEMO model Grid, 2D Mesh Mask, v1', 'summary':'''NEMO grid variable value for the u-v plane of the SalishSeaCast NEMO model Arakawa-C grid. The values are those calculated by NEMO from the input coordinates and bathymetry files. The variable names are those used by NEMO-3.4, see the NEMO-3.4 book (http://www.nemo-ocean.eu/Media/Files/NEMO_book_V3_4.pdf) for details, or the long_name attributes of the variables for succinct descriptions of the variables. The model grid includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. v1: e1t, e2t, e1u, e2u, e1v, e2v, e1f, e2f, glamt, gphit, glamu, gphiu, glamv, gphiv, tmaskutil, umaskutil, vmaskutil, fmaskutil, ff, mbathy variables''', 'fileNameRegex': '.*mesh_mask_SalishSea2\.nc$', }, 'ubcSSn3DMeshMask2V1': { 'type': 'geolocation bathymetry', 'title': 'SalishSeaCast NEMO model Grid, 3D Mesh Mask, v1', 'summary':'''NEMO grid variable value for the SalishSeaCast NEMO model Arakawa-C grid. The values are those calculated by NEMO from the input coordinates and bathymetry files. The variable names are those used by NEMO-3.4, see the NEMO-3.4 book (http://www.nemo-ocean.eu/Media/Files/NEMO_book_V3_4.pdf) for details, or the long_name attributes of the variables for succinct descriptions of the variables. The model grid includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. v1: e3t, e3u, e3v, e3w, gdept, gdepu, gdepv, gdepw, tmask, umask, vmask, fmask variables''', 'fileNameRegex': '.*mesh_mask_SalishSea2\.nc$' }, 'ubcSSnPointAtkinsonSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Point Atkinson, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minute intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Point Atkinson tide gauge station on the north side of English Bay, near Vancouver, British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*PointAtkinson\.nc$', }, 'ubcSSnCampbellRiverSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Campbell River, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Campbell River tide gauge station at the north end of the Strait of Georgia, near Campbell River, British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*CampbellRiver\.nc$', }, 'ubcSSnCherryPointSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Cherry Point, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Cherry Point tide gauge station in the southern Strait of Georgia, near Birch Bay, Washington. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*CherryPoint\.nc$', }, 'ubcSSnFridayHarborSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Friday Harbor, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Friday Harbor tide gauge station at San Juan Island in Haro Strait, near Friday Harbor, Washington. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*FridayHarbor\.nc$', }, 'ubcSSnNanaimoSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Nanaimo, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Nanaimo tide gauge station on the west side of the central Strait of Georgia, near Nanaimo, British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*Nanaimo\.nc$', }, 'ubcSSnNeahBaySSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Neah Bay, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Neah Bay tide gauge station on the south side of the west end of the Juan de Fuca Strait, near Neah Bay, Washington. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*NeahBay\.nc$', }, 'ubcSSnVictoriaSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Victoria, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Victoria tide gauge station on the north side of the east end of the Juan de Fuca Strait, in the Victoria Inner Harbour, near Victoria, British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*Victoria\.nc$', }, 'ubcSSnSandHeadsSSH15mV1': { 'type': 'tide gauge', 'title': 'Nowcast, Sand Heads, Sea Surface Height, 15min, v1', 'summary': '''Sea surface height values averaged over 15 minutes intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated at the model grid point closest to the Sand Heads light station on the east side of the central Strait of Georgia, near Steveston, British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: ssh variable''', 'fileNameRegex': '.*Sandheads\.nc$', }, 'ubcSSn3DTracerFields1hV1': { 'type': '3d fields', 'title': 'Nowcast, Salish Sea, 3d Tracer Fields, Hourly, v1', 'summary': '''3d salinity and water temperature field values averaged over 1 hour intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated for the entire model grid that includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: salinity (practical) and temperature variables''', 'fileNameRegex': '.*SalishSea_1h_\d{8}_\d{8}_grid_T\.nc$', }, 'ubcSSnSurfaceTracerFields1hV1': { 'type': 'surface fields', 'title': 'Nowcast, Salish Sea, Surface Tracer Fields, Hourly, v1', 'summary': '''2d sea surface height and rainfall rate field values averaged over 1 hour intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated for the surface of the model grid that includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: sea surface height and rainfall rate variables''', 'fileNameRegex': '.*SalishSea_1h_\d{8}_\d{8}_grid_T\.nc$', }, 'ubcSSn3DuVelocity1hV1': { 'type': '3d fields', 'title': 'Nowcast, Salish Sea, 3d u Velocity Field, Hourly, v1', 'summary': '''3d zonal (u) component velocity field values averaged over 1 hour intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated for the entire model grid that includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: uVelocity variable''', 'fileNameRegex': '.*SalishSea_1h_\d{8}_\d{8}_grid_U\.nc$', }, 'ubcSSn3DvVelocity1hV1': { 'type': '3d fields', 'title': 'Nowcast, Salish Sea, 3d v Velocity Field, Hourly, v1', 'summary': '''3d meridional (v) component velocity field values averaged over 1 hour intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated for the entire model grid that includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: vVelocity variable''', 'fileNameRegex': '.*SalishSea_1h_\d{8}_\d{8}_grid_V\.nc$', }, 'ubcSSn3DwVelocity1hV1': { 'type': '3d fields', 'title': 'Nowcast, Salish Sea, 3d w Velocity Field, Hourly, v1', 'summary': '''3d vertical (w) component velocity field values averaged over 1 hour intervals from SalishSeaCast NEMO model nowcast runs. The values are calculated for the entire model grid that includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location and depth data for the SalishSeaCast NEMO model grid are available in the ubcSSnBathymetry2V1 dataset. v1: wVelocity variable''', 'fileNameRegex': '.*SalishSea_1h_\d{8}_\d{8}_grid_W\.nc$', }, 'ubcSSaSurfaceAtmosphereFieldsV1': { 'type': 'surface fields', 'title': 'HRDPS, Salish Sea, Atmospheric Forcing Fields, Hourly, v1', 'summary': '''2d hourly atmospheric field values from the Environment Canada HRDPS atmospheric forcing model that are used to force the SalishSeaCast NEMO model. The model grid includes the Juan de Fuca Strait, the Strait of Georgia, Puget Sound, and Johnstone Strait on the coasts of Washington State and British Columbia. Geo-location data for the atmospheric forcing grid are available in the ubcSSaAtmosphereGridV1 dataset. Atmospheric field values are interpolated on to the SalishSeaCast NEMO model grid (ubcSSnBathymetry2V1 dataset) on-the-fly by NEMO. v1: atmospheric pressure, precipitation rate, 2m specific humidity, 2m air temperature, short-wave radiation flux, long-wave radiation flux, 10m u wind component, 10m v wind component variables''', 'fileNameRegex': '.*ops_y\d{4}m\d{2}d\d{2}\.nc$', }, } datasets['ubcSSn3DwVelocity1hV2'] = datasets['ubcSSn3DwVelocity1hV1'] datasets['ubcSSn3DwVelocity1hV2'].update({ 'title': datasets['ubcSSn3DwVelocity1hV1']['title'].replace(', v1', ', v2'), 'summary': datasets['ubcSSn3DwVelocity1hV1']['summary'] + ''' v2: Added eddy viscosity & diffusivity variables ve_eddy_visc & ve_eddy_diff''', }) # The `dataset_vars` dictionary below is used to rename # variables from the often cryptic NEMO names to the names # that appear in the ERDDAP generated files and web content. # # The keys are the NEMO variable names to replace. # # The values are dicts that map the variable names to use in ERDDAP # to the `destinationName` attribute name. # In[5]: dataset_vars = { 'sossheig': {'destinationName': 'ssh'}, 'vosaline': {'destinationName': 'salinity'}, 'votemper': {'destinationName': 'temperature'}, 'vozocrtx': {'destinationName': 'uVelocity'}, 'vomecrty': {'destinationName': 'vVelocity'}, 'vovecrtz': {'destinationName': 'wVelocity'}, } # A few convenient functions to reduce code repetition: # In[6]: def print_tree(root): """Display an XML tree fragment with indentation. """ print(etree.tostring(root, pretty_print=True).decode('ascii')) # In[7]: def find_att(root, att): """Return the dataset attribute element named att or raise a ValueError exception if it cannot be found. """ e = root.find('.//att[@name="{}"]'.format(att)) if e is None: raise ValueError('{} attribute element not found'.format(att)) return e # In[8]: def replace_yx_with_lonlat(root): new_axes = { 'y': {'sourceName': 'nav_lon', 'destinationName': 'longitude'}, 'x': {'sourceName': 'nav_lat', 'destinationName': 'latitude'}, } for axis in root.findall('.//axisVariable'): if axis.find('.//sourceName').text in new_axes: key = axis.find('.//sourceName').text new_axis = etree.Element('axisVariable') etree.SubElement(new_axis, 'sourceName').text = new_axes[key]['sourceName'] etree.SubElement(new_axis, 'destinationName').text = new_axes[key]['destinationName'] axis.getparent().replace(axis, new_axis) # Now we're ready to produce a dataset!!! # # Use the `/opt/tomcat/webapps/erddap/WEB-INF/GenerateDatasetsXml.sh` script # generate the initial version of an XML fragment for a dataset: # ``` # $ cd /opt/tomcat/webapps/erddap/WEB-INF/ # $ bash GenerateDatasetsXml.sh EDDGridFromNcFiles /results/SalishSea/nowcast/ # ``` # The `EDDGridFromNcFiles` and `/results/SalishSea/nowcast/` arguments # tell the script which `EDDType` and what parent directory to use, # avoiding having to type those in answer to prompts. # Answer the remaining prompts, # for example: # ``` # File name regex (e.g., ".*\.nc") (default="") # ? .*SalishSea_1h_\d{8}_\d{8}_grid_W\.nc$ # # Full file name of one file (default="") # ? /results/SalishSea/nowcast/28jan16/SalishSea_1h_20160128_20160128_grid_W.nc # # ReloadEveryNMinutes (e.g., 10080) (default="") # ? 10080 # ``` # Other examples of file name regex are: # # * `.*PointAtkinson.nc$` # * `.*SalishSea_1d_\d{8}_\d{8}_grid_W\.nc$` # # The output is written to `/results/erddap/logs/GenerateDatasetsXml.out` # # Now, we: # # * set the `datasetID` we want to use # * parse the output of `GenerateDatasetsXml.sh` into an XML tree data structure # * set the `datasetID` dataset attribute value # * re-set the `fileNameRegex` dataset attribute value because it looses its `\` characters during parsing(?) # * edit and add dataset attributes from the `metadata` dict # * set the `title` and `summary` dataset attributes from the `datasets` dict # * set the names of the grid `x` and `y` axis variables # * rename data variables as specified in the `dataset_vars` dict # In[10]: def update_xml(root, datasetID, metadata, datasets, dataset_vars): root.attrib['datasetID'] = datasetID root.find('.//fileNameRegex').text = datasets[datasetID]['fileNameRegex'] for att, info in metadata.items(): e = etree.Element('att', name=att) e.text = info['text'] try: root.find('.//att[@name="{}"]'.format(info['after'])).addnext(e) except KeyError: find_att(root, att).text = info['text'] title = datasets[datasetID]['title'] find_att(root, 'title').text = title find_att(root, 'summary').text = '{0}\n\n{1}'.format(title, datasets[datasetID]['summary']) for axis_name in root.findall('.//axisVariable/destinationName'): if axis_name.text in ('x', 'y'): axis_name.text = 'grid{}'.format(axis_name.text.upper()) if datasets[datasetID]['type'] == 'tide gauge': replace_yx_with_lonlat(root) for var_name in root.findall('.//dataVariable/destinationName'): if var_name.text in dataset_vars: var_name.text = dataset_vars[var_name.text]['destinationName'] # In[17]: parser = etree.XMLParser(remove_blank_text=True) tree = etree.parse('/results/erddap/logs/GenerateDatasetsXml.out', parser) root = tree.getroot() datasetID = 'ubcSSn3dTracerFields1hV1' update_xml(root, datasetID, metadata, datasets, dataset_vars) # Inspect the resulting dataset XML fragment below and edit the dicts and # code cell above until it is what is required for the dataset: # In[24]: print_tree(root) # Extra processing step are required for some types of datasets. # See: # # * [Surface Field Datasets](#Surface-Field-Datasets) # * [Model Grid Geo-location and Bathymetry Datasets](#Model-Grid-Geo-location-and-Bathymetry-Datasets) # * [EC HDRPS Atmospheric Forcing Datasets](#EC-HDRPS-Atmospheric-Forcing-Datasets) # Store the XML fragment for the dataset: # In[19]: with open('/results/erddap_datasets_xml/{}.xml'.format(datasetID), 'wb') as f: f.write(etree.tostring(root, pretty_print=True)) # Edit `/opt/tomcat/content/erddap/datasets.xml` to include the # XML fragment for the dataset that was stored by the abave cell. # # Run `/opt/tomcast/webapps/erddap/WEB-INF/DasDds.sh` to: # # * confirm that the XML fragment is correct # * create the `.das` and `.ddl` files for the dataset # # `DasDds.hs` takes a `datasetID` as input from the command-line # and generates a ridiculous amount of output on the screen # and in `/results/erddap/logs/DasDds.log` # and `/results/erddap/logs/DasDds.out`. # # If `DasDds.sh` finishes without errors and produces # metadata that looks sensible for the dataset, # create a flag file to signal the ERDDAP server process to load the dataset: # ``` # $ cd /results/erddap/flag/ # $ touch # ``` # ## Surface Field Datasets # # The `/opt/tomcat/webapps/erddap/WEB-INF/GenerateDatasetsXml.sh` script produces and XML # fragment that uses all of the dimensions that it finds in the sample file it parses, # and includes only the variables that have all of those dimensions. # To produce an XML fragment for surface fields we need to do some additional work: # # * Delete the depth axis # * Delete all of the `dataVariable` elements # * Add `dataVariable` elements for the surface variables # In[22]: for axis in root.findall('.//axisVariable'): if axis.find('.//destinationName').text == 'depth': axis.getparent().remove(axis) break for var in root.findall('.//dataVariable'): var.getparent().remove(var) var = etree.SubElement(root, 'dataVariable') etree.SubElement(var, 'sourceName').text = 'sossheig' etree.SubElement(var, 'destinationName').text = 'ssh' etree.SubElement(var, 'dataType').text = 'float' attrs = etree.SubElement(var, 'addAttributes') etree.SubElement(attrs, 'att', name='_ChunkSize').text = 'null' etree.SubElement(attrs, 'att', name='coordinates').text = 'null' var = etree.SubElement(root, 'dataVariable') etree.SubElement(var, 'sourceName').text = 'rain_rate' etree.SubElement(var, 'destinationName').text = 'rain_rate' etree.SubElement(var, 'dataType').text = 'float' attrs = etree.SubElement(var, 'addAttributes') etree.SubElement(attrs, 'att', name='_ChunkSize').text = 'null' etree.SubElement(attrs, 'att', name='coordinates').text = 'null' find_att(root, 'keywords').text = ( 'model results, height, local, sea, sea surface height, sossheig, source, surface, time_counter') # In[23]: print_tree(root) # In[22]: with open('/results/erddap_datasets_xml/{}.xml'.format(datasetID), 'wb') as f: f.write(etree.tostring(root, pretty_print=True)) # ## Model Grid Geo-location and Bathymetry Datasets # # Model grid geo-location and bathymetry datasets require a lot of hand editing # because they are not model generated. # Here is an example of a finished one: # In[26]: parser = etree.XMLParser(remove_blank_text=True) tree = etree.parse('/results/erddap_datasets_xml/ubcSSnBathymetry2V1.xml', parser) root = tree.getroot() print_tree(root) # ## EC HDRPS Atmospheric Forcing Datasets # # ### Atmospheric Forcing Grid Geo-location Dataset # # Use the `/opt/tomcat/webapps/erddap/WEB-INF/GenerateDatasetsXml.sh` script # generate the initial version of an XML fragment for the dataset: # ``` # $ cd /opt/tomcat/webapps/erddap/WEB-INF/ # $ bash GenerateDatasetsXml.sh EDDGridFromNcFiles /results/forcing/atmospheric/GEM2.5/operational/ ops_y\d{4}m\d{2}d\d{2}.nc$ /results/forcing/atmospheric/GEM2.5/operational/ops_y2016m03d07.nc 10080 # ``` # # Like the model grid geo-location and bathymetry dataset, # the atmospheric forcing grid dataset requires a lot of hand editing. # Here is the finished dataset: # In[12]: parser = etree.XMLParser(remove_blank_text=True) tree = etree.parse('/results/erddap_datasets_xml/ubcSSaAtmosphereGridV1.xml', parser) root = tree.getroot() print_tree(root) # ### Atmospheric Forcing Model Fields # # * Change the value of the `recursive` element to `false` so that the `/results/forcing/atmospheric/GEM2.5/operational/fcst/` directory is excluded # * Add Environment Canada acknowledgement and terms & conditions of use to `license` element # * Add Environment Canada to `acknowledgement` element # In[45]: root.find('.//recursive').text = 'false' find_att(root, 'license').text += ''' This dataset is derived from a product of the Environment Canada HRDPS (High Resolution Deterministic Prediction System) model. The Terms and conditions of use of Meteorological Data from Environment Canada are available at http://dd.weather.gc.ca/doc/LICENCE_GENERAL.txt.''' find_att(root, 'acknowledgement').text += ', Environment Canada' # In[47]: for axis in root.findall('.//axisVariable'): axis_name = axis.find('.//sourceName').text if 'time' not in axis_name: attrs = axis.find('.//addAttributes') etree.SubElement(attrs, 'att', name='grid_spacing').text = 'null' etree.SubElement(attrs, 'att', name='units').text = 'null' etree.SubElement(attrs, 'att', name='long_name').text = axis_name.upper() etree.SubElement(attrs, 'att', name='standard_name').text = axis_name # In[48]: print_tree(root) # In[49]: with open('/results/erddap_datasets_xml/{}.xml'.format(datasetID), 'wb') as f: f.write(etree.tostring(root, pretty_print=True))