Source code for compass.landice.tests.eismint2.run_experiment

import numpy
from netCDF4 import Dataset as NetCDFFile
import shutil

from mpas_tools.logging import check_call

from compass.model import run_model
from compass.step import Step


[docs]class RunExperiment(Step): """ A step for performing forward MALI runs as part of eismint2 test cases. Attributes ---------- experiment : {'a', 'b', 'c', 'd', 'f', 'g'} The EISMINT2 experiment (a-d or f-g) to perform suffixes : list of str, optional a list of suffixes for namelist and streams files produced for this step. Most steps most runs will just have a ``namelist.landice`` and a ``streams.landice`` (the default) but the ``restart_run`` step of the ``restart_test`` runs the model twice, the second time with ``namelist.landice.rst`` and ``streams.landice.rst`` """
[docs] def __init__(self, test_case, experiment, name='run_model', subdir=None, cores=1, min_cores=None, threads=1, suffixes=None): """ Create a new test case Parameters ---------- test_case : compass.TestCase The test case this step belongs to experiment : {'a', 'b', 'c', 'd', 'f', 'g'} The EISMINT2 experiment (a-d or f-g) to perform name : str, optional the name of the test case subdir : str, optional the subdirectory for the step. The default is ``name`` cores : int, optional the number of cores the step would ideally use. If fewer cores are available on the system, the step will run on all available cores as long as this is not below ``min_cores`` min_cores : int, optional the number of cores the step requires. If the system has fewer than this number of cores, the step will fail threads : int, optional the number of threads the step will use suffixes : list of str, optional a list of suffixes for namelist and streams files produced for this step. Most steps most runs will just have a ``namelist.landice`` and a ``streams.landice`` (the default) but the ``restart_run`` step of the ``restart_test`` runs the model twice, the second time with ``namelist.landice.rst`` and ``streams.landice.rst`` """ self.experiment = experiment if suffixes is None: suffixes = ['landice'] self.suffixes = suffixes if min_cores is None: min_cores = cores super().__init__(test_case=test_case, name=name, subdir=subdir, cores=cores, min_cores=min_cores, threads=threads) for suffix in suffixes: self.add_namelist_file( 'compass.landice.tests.eismint2', 'namelist.landice', out_name='namelist.{}'.format(suffix)) self.add_streams_file( 'compass.landice.tests.eismint2', 'streams.landice', out_name='streams.{}'.format(suffix)) if experiment in ('a', 'f', 'g'): self.add_input_file(filename='landice_grid.nc', target='../setup_mesh/landice_grid.nc') else: self.add_input_file(filename='experiment_a_output.nc', target='../experiment_a/output.nc') self.add_input_file(filename='graph.info', target='../setup_mesh/graph.info') self.add_model_as_input() self.add_output_file(filename='output.nc')
# no setup() is needed
[docs] def run(self): """ Run this step of the test case """ _setup_eismint2_initial_conditions(self.logger, self.experiment, filename='initial_condition.nc') for suffix in self.suffixes: run_model(self, namelist='namelist.{}'.format(suffix), streams='streams.{}'.format(suffix))
def _setup_eismint2_initial_conditions(logger, experiment, filename): """ Add the initial condition for the given EISMINT2 experiment to the given MPAS mesh file Parameters ---------- logger : logging.Logger A logger for output from the step experiment : {'a', 'b', 'c', 'd', 'f', 'g'} The name of the experiment filename : str file to add the initial condition to """ if experiment in ('a', 'b', 'c', 'd', 'f', 'g'): logger.info('Setting up EISMINT2 Experiment {}'.format(experiment)) else: raise ValueError("Invalid experiment specified: {}. Please specify " "an experiment between 'a' and 'g', excluding " "'e'".format(experiment)) # Setup dictionaries of parameter values for each experiment # Mmax: Maximum SMB at center of domain (m a-1) # Sb: gradient of SMB with horizontal distance (m a-1 km-1) # Rel: radial distance from summit where SMB = 0 (km) # Tmin: surface temperature at summit (K) # ST: gradient of air temperature with horizontal distance (K km-1) # beta: basal traction coefficient (Pa m-1 a) # Note: beta is the inverse of parameter B in Payne et al. (2000) exp_params = {'a': {'Mmax': 0.5, 'Sb': 10.0**-2, 'Rel': 450.0, 'Tmin': 238.15, 'ST': 1.67e-2, 'beta': 1.0e8}, 'b': {'Mmax': 0.5, 'Sb': 10.0**-2, 'Rel': 450.0, 'Tmin': 243.15, 'ST': 1.67e-2, 'beta': 1.0e8}, 'c': {'Mmax': 0.25, 'Sb': 10.0**-2, 'Rel': 425.0, 'Tmin': 238.15, 'ST': 1.67e-2, 'beta': 1.0e8}, 'd': {'Mmax': 0.5, 'Sb': 10.0**-2, 'Rel': 425.0, 'Tmin': 238.15, 'ST': 1.67e-2, 'beta': 1.0e8}, 'f': {'Mmax': 0.5, 'Sb': 10.0**-2, 'Rel': 450.0, 'Tmin': 223.15, 'ST': 1.67e-2, 'beta': 1.0e8}, 'g': {'Mmax': 0.5, 'Sb': 10.0**-2, 'Rel': 450.0, 'Tmin': 238.15, 'ST': 1.67e-2, 'beta': 1.0e3}} xsummit = 750000.0 ysummit = 750000.0 rhoi = 910.0 scyr = 3600.0 * 24.0 * 365.0 # Some experiments start from scratch, others start from the SS of a previous experiment if experiment in ('a', 'f', 'g'): # we will build the mesh from scratch shutil.copyfile('landice_grid.nc', filename) else: # use the final state of experiment A args = ['ncks', '-O', '-d', 'Time,-1', 'experiment_a_output.nc', filename] check_call(args, logger) # Open the new input file, get needed dimensions & variables gridfile = NetCDFFile(filename, 'r+') nVertLevels = len(gridfile.dimensions['nVertLevels']) # Get variables xCell = gridfile.variables['xCell'][:] yCell = gridfile.variables['yCell'][:] xEdge = gridfile.variables['xEdge'][:] yEdge = gridfile.variables['yEdge'][:] xVertex = gridfile.variables['xVertex'][:] yVertex = gridfile.variables['yVertex'][:] # =================== # initial conditions # =================== # If starting from scratch, setup dimension variables and initial condition # variables if experiment in ('a', 'f', 'g'): # Find center of domain x0 = xCell[:].min() + 0.5 * (xCell[:].max() - xCell[:].min()) y0 = yCell[:].min() + 0.5 * (yCell[:].max() - yCell[:].min()) # Calculate distance of each cell center from dome center r = ((xCell[:] - x0)**2 + (yCell[:] - y0)**2)**0.5 # Center the dome in the center of the cell that is closest to the # center of the domain. centerCellIndex = numpy.abs(r[:]).argmin() # EISMINT-2 puts the center of the domain at 750,750 km instead of 0,0. # Adjust to use that origin. xShift = -1.0 * xCell[centerCellIndex] + xsummit yShift = -1.0 * yCell[centerCellIndex] + ysummit xCell[:] = xCell[:] + xShift yCell[:] = yCell[:] + yShift xEdge[:] = xEdge[:] + xShift yEdge[:] = yEdge[:] + yShift xVertex[:] = xVertex[:] + xShift yVertex[:] = yVertex[:] + yShift gridfile.variables['xCell'][:] = xCell[:] gridfile.variables['yCell'][:] = yCell[:] gridfile.variables['xEdge'][:] = xEdge[:] gridfile.variables['yEdge'][:] = yEdge[:] gridfile.variables['xVertex'][:] = xVertex[:] gridfile.variables['yVertex'][:] = yVertex[:] # Assign initial condition variable values for EISMINT-2 experiment # Start with no ice gridfile.variables['thickness'][:] = 0.0 # flat bed at sea level gridfile.variables['bedTopography'][:] = 0.0 # constant, arbitrary temperature, degrees K (doesn't matter since # there is no ice initially) gridfile.variables['temperature'][:] = 273.15 # Setup layerThicknessFractions gridfile.variables['layerThicknessFractions'][:] = 1.0 / nVertLevels else: StrLen = len(gridfile.dimensions['StrLen']) gridfile.variables['xtime'][0, :] = list( '000000-01-01_00:00:00'.ljust(StrLen, ' ')) # Now update/set origin location and distance array r = ((xCell[:] - xsummit)**2 + (yCell[:] - ysummit)**2)**0.5 # =================== # boundary conditions # =================== # Define values prescribed by Payne et al. 2000 paper. params = exp_params[experiment] logger.info("Parameters for this experiment: {}".format(params)) # SMB field specified by EISMINT, constant in time for EISMINT2 # It is a function of geographical position (not elevation) # maximum accumulation rate [m/yr] converted to [m/s] Mmax = params['Mmax'] / scyr # gradient of accumulation rate change with horizontal distance [m/a/km] # converted to [m/s/m] Sb = params['Sb'] / scyr / 1000.0 # accumulation rate at 0 position [km] converted to [m] Rel = params['Rel'] * 1000.0 SMB = numpy.minimum(Mmax, Sb * (Rel - r)) # [m ice/s] SMB = SMB * rhoi # in kg/m2/s if 'sfcMassBal' in gridfile.variables: sfcMassBalVar = gridfile.variables['sfcMassBal'] else: datatype = gridfile.variables[ 'xCell'].dtype # Get the datatype for double precision float sfcMassBalVar = gridfile.createVariable('sfcMassBal', datatype, ('Time', 'nCells')) sfcMassBalVar[0, :] = SMB # Surface temperature # minimum surface air temperature [K] Tmin = params['Tmin'] # gradient of air temperature change with horizontal distance [K/km] # converted to [K/m] ST = params['ST'] / 1000.0 if 'surfaceAirTemperature' in gridfile.variables: surfaceAirTemperatureVar = gridfile.variables['surfaceAirTemperature'] else: datatype = gridfile.variables[ 'xCell'].dtype # Get the datatype for double precision float surfaceAirTemperatureVar = gridfile.createVariable( 'surfaceAirTemperature', datatype, ('Time', 'nCells')) surfaceAirTemperatureVar[0, :] = Tmin + ST * r # beta beta = params['beta'] if 'beta' in gridfile.variables: betaVar = gridfile.variables['beta'] else: datatype = gridfile.variables[ 'xCell'].dtype # Get the datatype for double precision float betaVar = gridfile.createVariable('beta', datatype, ('Time', 'nCells')) betaVar[0, :] = beta gridfile.close() logger.info('Successfully added initial conditions for EISMINT2, ' 'experiment {} to the file: {}'.format(experiment, filename))