diff --git a/demos/reanalysis-forced.ipynb b/demos/reanalysis-forced.ipynb index 72f56527..f66e8333 100644 --- a/demos/reanalysis-forced.ipynb +++ b/demos/reanalysis-forced.ipynb @@ -117,22 +117,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "## Step 2: Prepare ocean forcing data\n", - "\n", - "We need to cut out our ocean forcing. The package expects an initial condition and one time-dependent segment per non-land boundary. Naming convention is `\"east_unprocessed\"` for segments and `\"ic_unprocessed\"` for the initial condition.\n", - "\n", - "Data can be downloaded directly from the [Copernicus Marine data store](https://data.marine.copernicus.eu/product/GLOBAL_MULTIYEAR_PHY_001_030/download) via their GUI (once logged in).\n", - "\n", - "1. Initial condition: Using the GUI, select an area matching your `longitude_extent` and `latitude_extent` that corresponds to the first day in your date range. Download the initial condition and save it with filename `ic_unprocessed.nc` inside the `glorys_path` directory.\n", - "2. Boundary forcing: Using the GUI, select the Eastern boundary of your domain (if you have one that contains ocean). Allow for a buffer of ~0.5 degrees in all directions, and download for the prescribed `date_range`. Download and name `east_unprocessed.nc`.\n", - "3. Repeat step 2 for the remaining sections of the domain." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "## Step 3: Make experiment object\n", + "## Step 2: Make experiment object\n", "The `regional_mom6.experiment` contains the regional domain basics, and also generates the horizontal and vertical grids, `hgrid` and `vgrid` respectively, and sets up the directory structures. " ] }, @@ -185,6 +170,31 @@ "```" ] }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Step 3: Prepare ocean forcing data\n", + "\n", + "We need to cut out our ocean forcing. The package expects an initial condition and one time-dependent segment per non-land boundary. Naming convention is `\"east_unprocessed\"` for segments and `\"ic_unprocessed\"` for the initial condition.\n", + "\n", + "In this notebook, we are forcing with the Copernicus Marine \"Glorys\" reanalysis dataset. There's a function in the `mom6-regional` package that generates a bash script to download the correct boundary forcing files for your experiment. First, you will need to create an account with Copernicus, and you'll be prompted for your username and password when you try to run the bash script.\n", + "\n", + "The function is called `get_glorys_rectangular` because the fully automated setup is only supported for domains with boundaries parallel to lines of longitude and latitude. To download more complex domain shapes you can call `rmom6.get_glorys_data` directly." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "expt.get_glorys_rectangular(\n", + " raw_boundaries_path=glorys_path,\n", + " boundaries=[\"north\", \"south\", \"east\", \"west\"],\n", + ")" + ] + }, { "cell_type": "markdown", "metadata": {}, diff --git a/regional_mom6/regional_mom6.py b/regional_mom6/regional_mom6.py index 26fa71be..7568a27a 100644 --- a/regional_mom6/regional_mom6.py +++ b/regional_mom6/regional_mom6.py @@ -13,7 +13,7 @@ import shutil import os import importlib.resources - +import datetime from .utils import quadrilateral_areas @@ -144,6 +144,54 @@ def longitude_slicer(data, longitude_extent, longitude_coords): return data +from pathlib import Path + +def get_glorys_data( + longitude_extent, + latitude_extent, + timerange, + segment_name, + download_path, + modify_existing = True, + buffer = 1 +): + """ + Generates a bash script to download all of the required ocean forcing data. + + Args: + longitude_extent (tuple of floats): Westward and Eastward extents of the segment + latitude_extent (tuple of floats): Southward and Northward extents of the segment + timerange (tule of datetime strings): Start and end of the segment in format %Y-%m-%d %H:%M:%S + segment_range (str): name of the segment (minus .nc extension, eg east_unprocessed) + download_path (str): Location of where this script is saved + modify_existing (bool): Whether to add to an existing script or start a new one + buffer (int): number of degrees to add to pad the file with to ensure that interpolation onto desired domain doesn't fail. + """ + path = Path(download_path) + file = open( + path / "get_glorysdata.sh", + "r" + ) + + if modify_existing: + lines = file.readlines() + else: + lines = ["#!/bin/bash\ncopernicusmarine login"] + file.close() + file = open( + path / "get_glorysdata.sh", + "w" + ) + + lines.append( + f""" +copernicusmarine subset --dataset-id cmems_mod_glo_phy_my_0.083deg_P1D-m --variable so --variable thetao --variable uo --variable vo --variable zos --start-datetime {timerange[0].replace(" ","T")} --end-datetime {timerange[1].replace(" ","T")} --minimum-longitude {longitude_extent[0]} --maximum-longitude {longitude_extent[1]} --minimum-latitude {latitude_extent[0]} --maximum-latitude {latitude_extent[1]} --minimum-depth 0 --maximum-depth 6000 -o {str(path)} -f {segment_name}.nc --force-download\n +""" + ) + file.writelines(lines) + file.close() + return + def hyperbolictan_thickness_profile(nlayers, ratio, total_depth): """Generate a hyperbolic tangent thickness profile with ``nlayers`` vertical @@ -899,6 +947,65 @@ def initial_condition( return + def get_glorys_rectangular( + self, + raw_boundaries_path, + boundaries=["south", "north", "west", "east"] + ): + """ + This function is a wrapper for `get_glorys_data`, calling this function once for each of the rectangular boundary segments and the initial condition. For more complex boundary shapes, call `get_glorys_data` directly for each of your boundaries that aren't parallel to lines of constant latitude or longitude. + + args: + raw_boundaries_path (str): Path to the directory containing the raw boundary forcing files. + boundaries (List[str]): List of cardinal directions for which to create boundary forcing files. + Default is `["south", "north", "west", "east"]`. + """ + + # Initial Condition + get_glorys_data( + self.longitude_extent, + self.latitude_extent, + [self.date_range[0],datetime.datetime.strptime(self.date_range[0], "%Y-%m-%d %H:%M:%S") + datetime.timedelta(days=1)], + "ic_unprocessed", + raw_boundaries_path, + modify_existing=False + ) + if "east" in boundaries: + get_glorys_data( + [self.longitude_extent[1] - 1,self.longitude_extent[1] + 1], + [self.latitude_extent[0] -1, self.latitude_extent[1] + 1], + self.date_range, + "east_unprocessed", + raw_boundaries_path, + ) + if "west" in boundaries: + get_glorys_data( + [self.longitude_extent[0] - 1,self.longitude_extent[0] + 1], + [self.latitude_extent[0] -1, self.latitude_extent[1] + 1], + self.date_range, + "west_unprocessed", + raw_boundaries_path, + ) + if "north" in boundaries: + get_glorys_data( + [self.longitude_extent[0] - 1,self.longitude_extent[1] + 1], + [self.latitude_extent[1] -1, self.latitude_extent[1] + 1], + self.date_range, + "north_unprocessed", + raw_boundaries_path, + ) + if "south" in boundaries: + get_glorys_data( + [self.longitude_extent[0] - 1,self.longitude_extent[1] + 1], + [self.latitude_extent[0] -1, self.latitude_extent[0] + 1], + self.date_range, + "south_unprocessed", + raw_boundaries_path, + ) + + print(f"script `get_glorys_data.sh` has been greated at {raw_boundaries_path}. Run this script via bash to download the data from a terminal with internet access. You will need to enter your Copernicus Marine username and password. If you don't have an account, make one here:\nhttps://data.marine.copernicus.eu/register") + return + def rectangular_boundaries( self, raw_boundaries_path,