From b73b3d5a9c8d54a532500c58984e2d40de20120e Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 23 Aug 2023 17:22:29 -0500 Subject: [PATCH 01/16] Add convergence test to drying_slope --- compass/ocean/tests/drying_slope/__init__.py | 5 + compass/ocean/tests/drying_slope/analysis.py | 97 ++++++++++++++++++ .../drying_slope/convergence/__init__.py | 98 +++++++++++++++++++ .../ocean/tests/drying_slope/viz/__init__.py | 4 + 4 files changed, 204 insertions(+) create mode 100644 compass/ocean/tests/drying_slope/analysis.py create mode 100644 compass/ocean/tests/drying_slope/convergence/__init__.py diff --git a/compass/ocean/tests/drying_slope/__init__.py b/compass/ocean/tests/drying_slope/__init__.py index 75b0cd8c3e..ab6c3de9e2 100644 --- a/compass/ocean/tests/drying_slope/__init__.py +++ b/compass/ocean/tests/drying_slope/__init__.py @@ -1,4 +1,5 @@ from compass.ocean.tests.drying_slope.decomp import Decomp +from compass.ocean.tests.drying_slope.convergence import Convergence from compass.ocean.tests.drying_slope.default import Default from compass.ocean.tests.drying_slope.loglaw import LogLaw from compass.ocean.tests.drying_slope.ramp import Ramp @@ -31,3 +32,7 @@ def __init__(self, mpas_core): self.add_test_case( LogLaw(test_group=self, resolution=resolution, coord_type=coord_type)) + for coord_type in ['sigma', 'single_layer']: + self.add_test_case( + Convergence(test_group=self, + coord_type=coord_type)) diff --git a/compass/ocean/tests/drying_slope/analysis.py b/compass/ocean/tests/drying_slope/analysis.py new file mode 100644 index 0000000000..f57b6087a9 --- /dev/null +++ b/compass/ocean/tests/drying_slope/analysis.py @@ -0,0 +1,97 @@ +import matplotlib.pyplot as plt +import numpy as np +import pandas as pd +import xarray as xr +from scipy.interpolate import interp1d + +from compass.step import Step + + +class Analysis(Step): + """ + A step for visualizing drying slope results, as well as comparison with + analytical solution and ROMS results. + + Attributes + ---------- + """ + def __init__(self, test_case, resolutions, damping_coeff): + super().__init__(test_case=test_case, name='analysis') + self.damping_coeff = damping_coeff + self.resolutions = resolutions + for resolution in resolutions: + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' + self.add_input_file(filename=f'output_{res_name}.nc', + target=f'../forward_{res_name}/output.nc') + self.add_output_file(filename='convergence.png') + + def run(self): + times = ['0.05', '0.15', '0.25', '0.30', '0.40', '0.50'] + for time in times: + self._plot_convergence(time) + + def _compute_rmse(self, ds, t): + x_exact, ssh_exact = self._exact_solution(t, self.damping_coeff) + ds = ds.isel(Time=t).groupby('yCell').mean( + dim=xr.ALL_DIMS) + x_mpas = ds.yCell.values / 1000.0 + ssh_mpas = ds.ssh.values + # Interpolate mpas solution to the points at which we have an exact + # solution + f = interp1d(x_mpas, ssh_mpas) + ssh_mpas_interp = f(x_exact) + rmse = np.sqrt(np.mean(np.square(ssh_mpas_interp - ssh_exact))) + return rmse + + def _plot_convergence(self, time): + """ + Plot convergence curves + time : float + simulation time at which to evaluate rmse + """ + + fig, ax = plt.subplots(nrows=1, ncols=1) + + max_rmse = 0 + rmse = np.zeros(len(self.resolutions)) + for i, resolution in enumerate(self.resolutions): + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' + ds = xr.open_dataset(f'output_{res_name}.nc') + rmse[i] = self._compute_rmse(ds, time) + + if rmse[i] > max_rmse: + max_rmse = rmse[i] + + ax.loglog(self.resolutions, rmse, + linestyle='-', marker='o', label='comp') + + rmse_1st_order = np.zeros(len(self.resolutions)) + rmse_1st_order[0] = max_rmse + for i in range(len(self.resolutions) - 1): + rmse_1st_order[i + 1] = rmse_1st_order[i] / 2.0 + + ax.loglog(self.resolutions, np.flip(rmse_1st_order), + linestyle='-', color='k', alpha=.25, label='1st order') + + ax.set_xlabel('Cell size (km)') + ax.set_ylabel('RMSE (m)') + + ax.legend(loc='lower right') + ax.set_title('Layer thickness convergence') + fig.tight_layout() + fig.savefig('convergence.png') + + def _exact_solution(self, time): + """ + Returns distance, ssh + """ + datafile = f'./r{self.damping_coeff}d{time}-'\ + f'analytical.csv' + data = pd.read_csv(datafile, header=None) + return data[0], data[1] diff --git a/compass/ocean/tests/drying_slope/convergence/__init__.py b/compass/ocean/tests/drying_slope/convergence/__init__.py new file mode 100644 index 0000000000..b5509dc973 --- /dev/null +++ b/compass/ocean/tests/drying_slope/convergence/__init__.py @@ -0,0 +1,98 @@ +from compass.config import CompassConfigParser +from compass.ocean.tests.drying_slope.analysis import Analysis +from compass.ocean.tests.drying_slope.forward import Forward +from compass.ocean.tests.drying_slope.initial_state import InitialState +from compass.ocean.tests.drying_slope.viz import Viz +from compass.testcase import TestCase +from compass.validate import compare_variables + + +class Convergence(TestCase): + """ + The default drying_slope test case + + Attributes + ---------- + resolution : float + The resolution of the test case in km + + coord_type : str + The type of vertical coordinate (``sigma``, ``single_layer``, etc.) + """ + + def __init__(self, test_group, coord_type): + """ + Create the test case + + Parameters + ---------- + test_group : compass.ocean.tests.drying_slope.DryingSlope + The test group that this test case belongs to + + coord_type : str + The type of vertical coordinate (``sigma``, ``single_layer``) + """ + name = 'convergence' + + self.coord_type = coord_type + damping_coeffs = [0.01] + self.damping_coeffs = damping_coeffs + subdir = f'{coord_type}/{name}' + super().__init__(test_group=test_group, name=name, + subdir=subdir) + self.resolutions = None + # add the steps with default resolutions so they can be listed + config = CompassConfigParser() + config.add_from_package('compass.ocean.tests.parabolic_bowl', + 'parabolic_bowl.cfg') + self._setup_steps(config) + + def _setup_steps(self, config): + """ setup steps given resolutions """ + + default_resolutions = '5, 10, 20' + + # set the default values that a user may change before setup + config.set('drying_slope_convergence', 'resolutions', + default_resolutions, + comment='a list of resolutions (km) to test') + + # get the resolutions back, perhaps with values set in the user's + # config file + resolutions = config.getlist('drying_slope_convergence', + 'resolutions', dtype=int) + + if self.resolutions is not None and self.resolutions == resolutions: + return + + # start fresh with no steps + self.steps = dict() + self.steps_to_run = list() + + self.resolutions = resolutions + + for resolution in self.resolutions: + + res_name = f'{resolution}km' + self.add_step(InitialState(test_case=self, + name=f'initial_state_{res_name}', + resolution=resolution, + coord_type=self.coord_type)) + self.add_step(Forward(test_case=self, resolution=resolution, + name=f'forward_{res_name}', + ntasks=4, openmp_threads=1, + damping_coeff=self.damping_coeffs, + coord_type=self.coord_type)) + self.add_step(Analysis(test_case=self, + resolutions=resolutions, + damping_coeff=self.damping_coeffs[0])) + + def validate(self): + """ + Validate variables against a baseline + """ + super().validate() + variables = ['layerThickness', 'normalVelocity'] + for res in self.resolutions: + compare_variables(test_case=self, variables=variables, + filename1=f'forward_{res}km/output.nc') diff --git a/compass/ocean/tests/drying_slope/viz/__init__.py b/compass/ocean/tests/drying_slope/viz/__init__.py index 46a96ba7a8..4eb1ed7b51 100644 --- a/compass/ocean/tests/drying_slope/viz/__init__.py +++ b/compass/ocean/tests/drying_slope/viz/__init__.py @@ -83,6 +83,10 @@ def run(self): self._images_to_movies(framesPerSecond=frames_per_second, outFolder=outFolder, extension=movie_format) + def _forcing(self, t): + ssh = 10. * numpy.sin(t * numpy.pi / 12.) - 10. + return ssh + def _plot_ssh_time_series(self, outFolder='.'): """ Plot ssh forcing on the right x boundary as a function of time against From 7e26596ecffece87a0cdc520c2b1a7f0ca59fd2a Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 20 Sep 2023 16:25:55 -0500 Subject: [PATCH 02/16] Add resolution arguments to init and forward --- .../tests/drying_slope/default/__init__.py | 24 +++----------- compass/ocean/tests/drying_slope/forward.py | 32 ++++++++++++++----- .../ocean/tests/drying_slope/initial_state.py | 24 +++++++------- .../ocean/tests/drying_slope/ramp/__init__.py | 20 ++---------- 4 files changed, 43 insertions(+), 57 deletions(-) diff --git a/compass/ocean/tests/drying_slope/default/__init__.py b/compass/ocean/tests/drying_slope/default/__init__.py index 977af6920b..784dc2c884 100644 --- a/compass/ocean/tests/drying_slope/default/__init__.py +++ b/compass/ocean/tests/drying_slope/default/__init__.py @@ -1,7 +1,7 @@ -from compass.testcase import TestCase -from compass.ocean.tests.drying_slope.initial_state import InitialState from compass.ocean.tests.drying_slope.forward import Forward +from compass.ocean.tests.drying_slope.initial_state import InitialState from compass.ocean.tests.drying_slope.viz import Viz +from compass.testcase import TestCase from compass.validate import compare_variables @@ -44,7 +44,8 @@ def __init__(self, test_group, resolution, coord_type): subdir = f'{res_name}/{coord_type}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) - self.add_step(InitialState(test_case=self, coord_type=coord_type)) + self.add_step(InitialState(test_case=self, resolution=resolution, + coord_type=coord_type)) if coord_type == 'single_layer': self.add_step(Forward(test_case=self, resolution=resolution, ntasks=4, openmp_threads=1, @@ -60,23 +61,6 @@ def __init__(self, test_group, resolution, coord_type): self.damping_coeffs = damping_coeffs self.add_step(Viz(test_case=self, damping_coeffs=damping_coeffs)) - def configure(self): - """ - Modify the configuration options for this test case. - """ - - resolution = self.resolution - config = self.config - ny = round(28 / resolution) - if resolution < 1.: - ny += 2 - dc = 1e3 * resolution - - config.set('drying_slope', 'ny', f'{ny}', comment='the number of ' - 'mesh cells in the y direction') - config.set('drying_slope', 'dc', f'{dc}', comment='the distance ' - 'between adjacent cell centers') - def validate(self): """ Validate variables against a baseline diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 64bd2e20ec..15592ea265 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -1,3 +1,5 @@ +import time + from compass.model import run_model from compass.step import Step @@ -57,14 +59,6 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, self.add_namelist_file('compass.ocean.tests.drying_slope', 'namelist.forward') - if resolution < 1.: - res_name = f'{int(resolution*1e3)}m' - else: - res_name = f'{int(resolution)}km' - self.add_namelist_file('compass.ocean.tests.drying_slope', - f'namelist.{res_name}.forward') - self.add_namelist_file('compass.ocean.tests.drying_slope', - f'namelist.{coord_type}.forward') if damping_coeff is not None: # update the Rayleigh damping coeff to the requested value options = {'config_Rayleigh_damping_coeff': f'{damping_coeff}'} @@ -96,5 +90,27 @@ def run(self): """ Run this step of the test case """ + dt = self.get_dt() + self.update_namelist_at_runtime(options={'config_dt': dt}, + out_name='namelist.ocean') run_model(self) + + def get_dt(self): + """ + Get the time step + + Returns + ------- + dt : str + the time step in HH:MM:SS + """ + config = self.config + # dt is proportional to resolution + dt_per_km = config.getfloat('drying_slope', 'dt_per_km') + + dt = dt_per_km * self.resolution + # https://stackoverflow.com/a/1384565/7728169 + dt = time.strftime('%H:%M:%S', time.gmtime(dt)) + + return dt diff --git a/compass/ocean/tests/drying_slope/initial_state.py b/compass/ocean/tests/drying_slope/initial_state.py index ea2be441fe..be4711c2eb 100644 --- a/compass/ocean/tests/drying_slope/initial_state.py +++ b/compass/ocean/tests/drying_slope/initial_state.py @@ -1,12 +1,9 @@ -import xarray -import numpy - -from mpas_tools.planar_hex import make_planar_hex_mesh from mpas_tools.io import write_netcdf from mpas_tools.mesh.conversion import convert, cull +from mpas_tools.planar_hex import make_planar_hex_mesh -from compass.step import Step from compass.model import run_model +from compass.step import Step class InitialState(Step): @@ -14,7 +11,8 @@ class InitialState(Step): A step for creating a mesh and initial condition for drying slope test cases """ - def __init__(self, test_case, coord_type='sigma'): + def __init__(self, test_case, resolution, name='initial_state', + coord_type='sigma'): """ Create the step @@ -23,11 +21,13 @@ def __init__(self, test_case, coord_type='sigma'): test_case : compass.ocean.tests.drying_slope.default.Default The test case this step belongs to """ - super().__init__(test_case=test_case, name='initial_state', ntasks=1, + super().__init__(test_case=test_case, name=name, ntasks=1, min_tasks=1, openmp_threads=1) self.coord_type = coord_type + self.resolution = resolution + self.add_namelist_file('compass.ocean.tests.drying_slope', 'namelist.init', mode='init') @@ -59,10 +59,12 @@ def run(self): 'config_tidal_boundary_layer_type': f"'{coord_type}'"} self.update_namelist_at_runtime(options) - section = config['drying_slope'] - nx = section.getint('nx') - ny = section.getint('ny') - dc = section.getfloat('dc') + # Determine mesh parameters + nx = config.getint('drying_slope', 'nx') + ny = round(28 / self.resolution) + if self.resolution < 1.: + ny += 2 + dc = 1e3 * self.resolution logger.info(' * Make planar hex mesh') dsMesh = make_planar_hex_mesh(nx=nx, ny=ny, dc=dc, nonperiodic_x=False, diff --git a/compass/ocean/tests/drying_slope/ramp/__init__.py b/compass/ocean/tests/drying_slope/ramp/__init__.py index 39f1380311..793d83cfeb 100644 --- a/compass/ocean/tests/drying_slope/ramp/__init__.py +++ b/compass/ocean/tests/drying_slope/ramp/__init__.py @@ -44,7 +44,8 @@ def __init__(self, test_group, resolution, coord_type): subdir = f'{res_name}/{coord_type}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) - self.add_step(InitialState(test_case=self, coord_type=coord_type)) + self.add_step(InitialState(test_case=self, resolution=resolution, + coord_type=coord_type)) if coord_type == 'single_layer': forward_step = Forward(test_case=self, resolution=resolution, ntasks=4, openmp_threads=1, @@ -66,23 +67,6 @@ def __init__(self, test_group, resolution, coord_type): self.damping_coeffs = damping_coeffs self.add_step(Viz(test_case=self, damping_coeffs=damping_coeffs)) - def configure(self): - """ - Modify the configuration options for this test case. - """ - - resolution = self.resolution - config = self.config - ny = round(28 / resolution) - if resolution < 1.: - ny += 2 - dc = 1e3 * resolution - - config.set('drying_slope', 'ny', f'{ny}', comment='the number of ' - 'mesh cells in the y direction') - config.set('drying_slope', 'dc', f'{dc}', comment='the distance ' - 'between adjacent cell centers') - def validate(self): """ Validate variables against a baseline From 03b27c8c4c60c1b48fd3aaea42ff3d86a6af4f4d Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 20 Sep 2023 16:27:42 -0500 Subject: [PATCH 03/16] Move time step to config option --- compass/ocean/tests/drying_slope/drying_slope.cfg | 8 ++++++++ compass/ocean/tests/drying_slope/forward.py | 2 +- compass/ocean/tests/drying_slope/namelist.1km.forward | 1 - compass/ocean/tests/drying_slope/namelist.250m.forward | 1 - 4 files changed, 9 insertions(+), 3 deletions(-) delete mode 100644 compass/ocean/tests/drying_slope/namelist.1km.forward delete mode 100644 compass/ocean/tests/drying_slope/namelist.250m.forward diff --git a/compass/ocean/tests/drying_slope/drying_slope.cfg b/compass/ocean/tests/drying_slope/drying_slope.cfg index 31c514c747..b203416de6 100644 --- a/compass/ocean/tests/drying_slope/drying_slope.cfg +++ b/compass/ocean/tests/drying_slope/drying_slope.cfg @@ -10,6 +10,14 @@ vert_levels = 10 # the number of grid cells in x nx = 6 +# time step in s per km of horizontal resolution +dt_per_km = 30 + +# config options for visualizing drying slope ouptut +[drying_slope_convergence] + +resolutions = 1., 2., 4., 16. + # config options for visualizing drying slope ouptut [drying_slope_viz] diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 15592ea265..35ecf237c8 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -91,7 +91,7 @@ def run(self): Run this step of the test case """ dt = self.get_dt() - self.update_namelist_at_runtime(options={'config_dt': dt}, + self.update_namelist_at_runtime(options={'config_dt': f"'{dt}'"}, out_name='namelist.ocean') run_model(self) diff --git a/compass/ocean/tests/drying_slope/namelist.1km.forward b/compass/ocean/tests/drying_slope/namelist.1km.forward deleted file mode 100644 index ef4481cc0a..0000000000 --- a/compass/ocean/tests/drying_slope/namelist.1km.forward +++ /dev/null @@ -1 +0,0 @@ -config_dt='0000_00:00:30' diff --git a/compass/ocean/tests/drying_slope/namelist.250m.forward b/compass/ocean/tests/drying_slope/namelist.250m.forward deleted file mode 100644 index 6198aa4969..0000000000 --- a/compass/ocean/tests/drying_slope/namelist.250m.forward +++ /dev/null @@ -1 +0,0 @@ -config_dt='0000_00:00:08' From afab39f1420a1db370b344673b938ee7fa012710 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 20 Sep 2023 17:18:06 -0500 Subject: [PATCH 04/16] Evaluate convergence with all times for which analytic solution available --- compass/ocean/tests/drying_slope/analysis.py | 80 +++++++++++++------- 1 file changed, 53 insertions(+), 27 deletions(-) diff --git a/compass/ocean/tests/drying_slope/analysis.py b/compass/ocean/tests/drying_slope/analysis.py index f57b6087a9..8101d59f52 100644 --- a/compass/ocean/tests/drying_slope/analysis.py +++ b/compass/ocean/tests/drying_slope/analysis.py @@ -1,3 +1,5 @@ +import os + import matplotlib.pyplot as plt import numpy as np import pandas as pd @@ -19,6 +21,7 @@ def __init__(self, test_case, resolutions, damping_coeff): super().__init__(test_case=test_case, name='analysis') self.damping_coeff = damping_coeff self.resolutions = resolutions + self.times = ['0.05', '0.15', '0.25', '0.30', '0.40', '0.50'] for resolution in resolutions: if resolution < 1.: res_name = f'{int(resolution*1e3)}m' @@ -26,50 +29,73 @@ def __init__(self, test_case, resolutions, damping_coeff): res_name = f'{int(resolution)}km' self.add_input_file(filename=f'output_{res_name}.nc', target=f'../forward_{res_name}/output.nc') + for time in self.times: + filename = f'r{damping_coeff}d{time}-analytical'\ + '.csv' + self.add_input_file(filename=filename, target=filename, + database='drying_slope') self.add_output_file(filename='convergence.png') def run(self): - times = ['0.05', '0.15', '0.25', '0.30', '0.40', '0.50'] - for time in times: - self._plot_convergence(time) + self._plot_convergence() def _compute_rmse(self, ds, t): - x_exact, ssh_exact = self._exact_solution(t, self.damping_coeff) - ds = ds.isel(Time=t).groupby('yCell').mean( - dim=xr.ALL_DIMS) + x_exact, ssh_exact = self._exact_solution(t) + tidx = int((float(t) / 0.2 + 1e-16) * 24.0) + ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], + ['yCell', 'ssh'])) + ds = ds.isel(Time=tidx) x_mpas = ds.yCell.values / 1000.0 ssh_mpas = ds.ssh.values # Interpolate mpas solution to the points at which we have an exact # solution + idx_min = np.argwhere(x_exact - x_mpas[0] >= 0.).item(0) + idx_max = np.argwhere(x_exact - x_mpas[-1] <= 0.).item(-1) f = interp1d(x_mpas, ssh_mpas) - ssh_mpas_interp = f(x_exact) - rmse = np.sqrt(np.mean(np.square(ssh_mpas_interp - ssh_exact))) + ssh_mpas_interp = f(x_exact[idx_min:idx_max]) + rmse = np.sqrt(np.mean(np.square(ssh_mpas_interp - + ssh_exact[idx_min:idx_max]))) return rmse - def _plot_convergence(self, time): + def _plot_convergence(self): """ Plot convergence curves - time : float - simulation time at which to evaluate rmse """ + comparisons = [] + cases = {'standard': '../../../standard/convergence/analysis', + 'ramp': '../../../ramp/convergence/analysis'} + for case in cases: + include = True + for resolution in self.resolutions: + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' + if not os.path.exists(f'{cases[case]}/output_{res_name}.nc'): + include = False + if include: + comparisons.append(case) + fig, ax = plt.subplots(nrows=1, ncols=1) max_rmse = 0 - rmse = np.zeros(len(self.resolutions)) - for i, resolution in enumerate(self.resolutions): - if resolution < 1.: - res_name = f'{int(resolution*1e3)}m' - else: - res_name = f'{int(resolution)}km' - ds = xr.open_dataset(f'output_{res_name}.nc') - rmse[i] = self._compute_rmse(ds, time) - - if rmse[i] > max_rmse: - max_rmse = rmse[i] - - ax.loglog(self.resolutions, rmse, - linestyle='-', marker='o', label='comp') + for k, comp in enumerate(comparisons): + rmse = np.zeros((len(self.resolutions), len(self.times))) + for i, resolution in enumerate(self.resolutions): + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' + ds = xr.open_dataset( + f'{cases[comp]}/output_{res_name}.nc') + for j, time in enumerate(self.times): + rmse[i, j] = self._compute_rmse(ds, time) + rmse_tav = np.mean(rmse, axis=1) + if np.max(rmse_tav) > max_rmse: + max_rmse = np.max(rmse_tav) + ax.loglog(self.resolutions, rmse_tav, + linestyle='-', marker='o', label=comp) rmse_1st_order = np.zeros(len(self.resolutions)) rmse_1st_order[0] = max_rmse @@ -83,7 +109,7 @@ def _plot_convergence(self, time): ax.set_ylabel('RMSE (m)') ax.legend(loc='lower right') - ax.set_title('Layer thickness convergence') + ax.set_title('SSH convergence') fig.tight_layout() fig.savefig('convergence.png') @@ -94,4 +120,4 @@ def _exact_solution(self, time): datafile = f'./r{self.damping_coeff}d{time}-'\ f'analytical.csv' data = pd.read_csv(datafile, header=None) - return data[0], data[1] + return data[0], -data[1] From 3dff6975d3d1699217b7395dd3cbccdce17d03d5 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 20 Sep 2023 17:19:52 -0500 Subject: [PATCH 05/16] Fixup default resolution and resolution arguments --- .../drying_slope/convergence/__init__.py | 21 +++++++++++++------ .../ocean/tests/drying_slope/drying_slope.cfg | 2 +- compass/ocean/tests/drying_slope/forward.py | 4 +++- 3 files changed, 19 insertions(+), 8 deletions(-) diff --git a/compass/ocean/tests/drying_slope/convergence/__init__.py b/compass/ocean/tests/drying_slope/convergence/__init__.py index b5509dc973..c80f8d78f2 100644 --- a/compass/ocean/tests/drying_slope/convergence/__init__.py +++ b/compass/ocean/tests/drying_slope/convergence/__init__.py @@ -50,7 +50,7 @@ def __init__(self, test_group, coord_type): def _setup_steps(self, config): """ setup steps given resolutions """ - default_resolutions = '5, 10, 20' + default_resolutions = '0.5, 1, 2' # set the default values that a user may change before setup config.set('drying_slope_convergence', 'resolutions', @@ -60,7 +60,7 @@ def _setup_steps(self, config): # get the resolutions back, perhaps with values set in the user's # config file resolutions = config.getlist('drying_slope_convergence', - 'resolutions', dtype=int) + 'resolutions', dtype=float) if self.resolutions is not None and self.resolutions == resolutions: return @@ -73,7 +73,10 @@ def _setup_steps(self, config): for resolution in self.resolutions: - res_name = f'{resolution}km' + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' self.add_step(InitialState(test_case=self, name=f'initial_state_{res_name}', resolution=resolution, @@ -81,7 +84,7 @@ def _setup_steps(self, config): self.add_step(Forward(test_case=self, resolution=resolution, name=f'forward_{res_name}', ntasks=4, openmp_threads=1, - damping_coeff=self.damping_coeffs, + damping_coeff=self.damping_coeffs[0], coord_type=self.coord_type)) self.add_step(Analysis(test_case=self, resolutions=resolutions, @@ -93,6 +96,12 @@ def validate(self): """ super().validate() variables = ['layerThickness', 'normalVelocity'] - for res in self.resolutions: + damping_coeff = self.damping_coeffs[0] + for resolution in self.resolutions: + if resolution < 1.: + res_name = f'{int(resolution*1e3)}m' + else: + res_name = f'{int(resolution)}km' + name = f'{res_name}_{damping_coeff}' compare_variables(test_case=self, variables=variables, - filename1=f'forward_{res}km/output.nc') + filename1=f'forward_{name}/output.nc') diff --git a/compass/ocean/tests/drying_slope/drying_slope.cfg b/compass/ocean/tests/drying_slope/drying_slope.cfg index b203416de6..3fe040171c 100644 --- a/compass/ocean/tests/drying_slope/drying_slope.cfg +++ b/compass/ocean/tests/drying_slope/drying_slope.cfg @@ -16,7 +16,7 @@ dt_per_km = 30 # config options for visualizing drying slope ouptut [drying_slope_convergence] -resolutions = 1., 2., 4., 16. +resolutions = 0.5, 1, 2 # config options for visualizing drying slope ouptut [drying_slope_viz] diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 35ecf237c8..9dfdbebdc8 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -50,6 +50,8 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, """ if min_tasks is None: min_tasks = ntasks + input_path = f'../{name}' + input_path = input_path.replace('forward', 'initial_state') if damping_coeff is not None: name = f'{name}_{damping_coeff}' @@ -57,6 +59,7 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, ntasks=ntasks, min_tasks=min_tasks, openmp_threads=openmp_threads) + self.resolution = resolution self.add_namelist_file('compass.ocean.tests.drying_slope', 'namelist.forward') if damping_coeff is not None: @@ -67,7 +70,6 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, self.add_streams_file('compass.ocean.tests.drying_slope', 'streams.forward') - input_path = '../initial_state' self.add_input_file(filename='mesh.nc', target=f'{input_path}/culled_mesh.nc') From 11841478070d8d6c7bfc11286ea065ca5d82bdfd Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Thu, 21 Sep 2023 13:28:12 -0500 Subject: [PATCH 06/16] Change directory structure --- compass/ocean/suites/wetdry.txt | 26 ++++++++----- compass/ocean/tests/drying_slope/__init__.py | 30 +++++++-------- .../drying_slope/convergence/__init__.py | 36 ++++++++++-------- .../tests/drying_slope/decomp/__init__.py | 37 ++++++------------- .../tests/drying_slope/default/__init__.py | 29 ++++++++++----- .../ocean/tests/drying_slope/drying_slope.cfg | 2 +- compass/ocean/tests/drying_slope/forward.py | 8 ++-- .../tests/drying_slope/loglaw/__init__.py | 24 ++---------- 8 files changed, 91 insertions(+), 101 deletions(-) diff --git a/compass/ocean/suites/wetdry.txt b/compass/ocean/suites/wetdry.txt index 42169c7f29..c5c15e221e 100644 --- a/compass/ocean/suites/wetdry.txt +++ b/compass/ocean/suites/wetdry.txt @@ -1,13 +1,19 @@ -ocean/drying_slope/250m/sigma/default -ocean/drying_slope/250m/sigma/ramp -ocean/drying_slope/250m/single_layer/default -ocean/drying_slope/250m/single_layer/ramp -ocean/drying_slope/1km/sigma/default -ocean/drying_slope/1km/sigma/ramp -ocean/drying_slope/1km/sigma/decomp -ocean/drying_slope/1km/single_layer/default -ocean/drying_slope/1km/single_layer/ramp -ocean/drying_slope/1km/single_layer/decomp +ocean/drying_slope/sigma/standard/250m/default +ocean/drying_slope/sigma/standard/1km/default +ocean/drying_slope/sigma/standard/250m/decomp +ocean/drying_slope/sigma/standard/1km/decomp +ocean/drying_slope/single_layer/standard/250m/default +ocean/drying_slope/single_layer/standard/1km/default +ocean/drying_slope/sigma/ramp/250m/default +ocean/drying_slope/sigma/ramp/1km/default +ocean/drying_slope/sigma/ramp/250m/decomp +ocean/drying_slope/sigma/ramp/1km/decomp +ocean/drying_slope/single_layer/ramp/250m/default +ocean/drying_slope/single_layer/ramp/1km/default +ocean/drying_slope/sigma/standard/250m/loglaw +ocean/drying_slope/sigma/standard/1km/loglaw +ocean/drying_slope/single_layer/standard/250m/loglaw +ocean/drying_slope/single_layer/standard/1km/loglaw ocean/dam_break/40cm/default ocean/dam_break/40cm/ramp ocean/dam_break/120cm/default diff --git a/compass/ocean/tests/drying_slope/__init__.py b/compass/ocean/tests/drying_slope/__init__.py index ab6c3de9e2..45553fa801 100644 --- a/compass/ocean/tests/drying_slope/__init__.py +++ b/compass/ocean/tests/drying_slope/__init__.py @@ -2,7 +2,6 @@ from compass.ocean.tests.drying_slope.convergence import Convergence from compass.ocean.tests.drying_slope.default import Default from compass.ocean.tests.drying_slope.loglaw import LogLaw -from compass.ocean.tests.drying_slope.ramp import Ramp from compass.testgroup import TestGroup @@ -18,21 +17,22 @@ def __init__(self, mpas_core): """ super().__init__(mpas_core=mpas_core, name='drying_slope') - for resolution in [0.25, 1.]: + for method in ['standard', 'ramp']: for coord_type in ['sigma', 'single_layer']: + for resolution in [0.25, 1.]: + self.add_test_case( + Default(test_group=self, resolution=resolution, + coord_type=coord_type, method=method)) + self.add_test_case( + Decomp(test_group=self, resolution=resolution, + coord_type=coord_type, method=method)) + for coord_type in ['sigma']: self.add_test_case( - Default(test_group=self, resolution=resolution, - coord_type=coord_type)) - self.add_test_case( - Decomp(test_group=self, resolution=resolution, - coord_type=coord_type)) - self.add_test_case( - Ramp(test_group=self, resolution=resolution, - coord_type=coord_type)) + Convergence(test_group=self, + coord_type=coord_type, + method=method)) + for coord_type in ['sigma', 'single_layer']: + for resolution in [0.25, 1.]: self.add_test_case( LogLaw(test_group=self, resolution=resolution, - coord_type=coord_type)) - for coord_type in ['sigma', 'single_layer']: - self.add_test_case( - Convergence(test_group=self, - coord_type=coord_type)) + coord_type=coord_type, method='standard')) diff --git a/compass/ocean/tests/drying_slope/convergence/__init__.py b/compass/ocean/tests/drying_slope/convergence/__init__.py index c80f8d78f2..7962ea9e8e 100644 --- a/compass/ocean/tests/drying_slope/convergence/__init__.py +++ b/compass/ocean/tests/drying_slope/convergence/__init__.py @@ -20,7 +20,7 @@ class Convergence(TestCase): The type of vertical coordinate (``sigma``, ``single_layer``, etc.) """ - def __init__(self, test_group, coord_type): + def __init__(self, test_group, coord_type, method): """ Create the test case @@ -37,20 +37,20 @@ def __init__(self, test_group, coord_type): self.coord_type = coord_type damping_coeffs = [0.01] self.damping_coeffs = damping_coeffs - subdir = f'{coord_type}/{name}' + subdir = f'{coord_type}/{method}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) self.resolutions = None # add the steps with default resolutions so they can be listed config = CompassConfigParser() - config.add_from_package('compass.ocean.tests.parabolic_bowl', - 'parabolic_bowl.cfg') - self._setup_steps(config) + config.add_from_package('compass.ocean.tests.drying_slope', + 'drying_slope.cfg') + self._setup_steps(config, subdir=subdir, method=method) - def _setup_steps(self, config): + def _setup_steps(self, config, subdir, method): """ setup steps given resolutions """ - default_resolutions = '0.5, 1, 2' + default_resolutions = '0.25, 0.5, 1, 2' # set the default values that a user may change before setup config.set('drying_slope_convergence', 'resolutions', @@ -77,15 +77,21 @@ def _setup_steps(self, config): res_name = f'{int(resolution*1e3)}m' else: res_name = f'{int(resolution)}km' + init_name = f'initial_state_{res_name}' self.add_step(InitialState(test_case=self, - name=f'initial_state_{res_name}', + name=init_name, resolution=resolution, coord_type=self.coord_type)) - self.add_step(Forward(test_case=self, resolution=resolution, - name=f'forward_{res_name}', - ntasks=4, openmp_threads=1, - damping_coeff=self.damping_coeffs[0], - coord_type=self.coord_type)) + forward_step = Forward(test_case=self, resolution=resolution, + name=f'forward_{res_name}', + input_path=f'../{init_name}', + ntasks=4, openmp_threads=1, + damping_coeff=self.damping_coeffs[0], + coord_type=self.coord_type) + if method == 'ramp': + forward_step.add_namelist_options( + {'config_zero_drying_velocity_ramp': ".true."}) + self.add_step(forward_step) self.add_step(Analysis(test_case=self, resolutions=resolutions, damping_coeff=self.damping_coeffs[0])) @@ -96,12 +102,10 @@ def validate(self): """ super().validate() variables = ['layerThickness', 'normalVelocity'] - damping_coeff = self.damping_coeffs[0] for resolution in self.resolutions: if resolution < 1.: res_name = f'{int(resolution*1e3)}m' else: res_name = f'{int(resolution)}km' - name = f'{res_name}_{damping_coeff}' compare_variables(test_case=self, variables=variables, - filename1=f'forward_{name}/output.nc') + filename1=f'forward_{res_name}/output.nc') diff --git a/compass/ocean/tests/drying_slope/decomp/__init__.py b/compass/ocean/tests/drying_slope/decomp/__init__.py index aa969f4cd5..766d933c85 100644 --- a/compass/ocean/tests/drying_slope/decomp/__init__.py +++ b/compass/ocean/tests/drying_slope/decomp/__init__.py @@ -1,4 +1,3 @@ -from compass.ocean.tests import drying_slope from compass.ocean.tests.drying_slope.forward import Forward from compass.ocean.tests.drying_slope.initial_state import InitialState from compass.testcase import TestCase @@ -22,11 +21,14 @@ def __init__(self, test_group, resolution, coord_type): Parameters ---------- - test_group : compass.ocean.tests.baroclinic_channel.BaroclinicChannel + test_group : compass.ocean.tests.drying_slope.DryingSlope The test group that this test case belongs to - resolution : str - The resolution of the test case + resolution : float + The resolution of the test case in km + + coord_type : str + The type of vertical coordinate (``sigma``, ``single_layer``) """ name = 'decomp' self.resolution = resolution @@ -35,11 +37,12 @@ def __init__(self, test_group, resolution, coord_type): res_name = f'{int(resolution*1e3)}m' else: res_name = f'{int(resolution)}km' - subdir = f'{res_name}/{coord_type}/{name}' + subdir = f'{coord_type}/{method}/{res_name}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) + self.add_step(InitialState(test_case=self, resolution=resolution, + coord_type=coord_type)) - self.add_step(InitialState(test_case=self, coord_type=coord_type)) if coord_type == 'single_layer': damping_coeff = None else: @@ -51,27 +54,11 @@ def __init__(self, test_group, resolution, coord_type): ntasks=procs, openmp_threads=1, damping_coeff=damping_coeff, coord_type=coord_type) + if method == 'ramp': + forward_step.add_namelist_options( + {'config_zero_drying_velocity_ramp': ".true."}) self.add_step(forward_step) - def configure(self): - """ - Modify the configuration options for this test case. - """ - - resolution = self.resolution - config = self.config - ny = round(28 / resolution) - if resolution < 1.: - ny += 2 - dc = 1e3 * resolution - - config.set('drying_slope', 'ny', f'{ny}', comment='the number of ' - 'mesh cells in the y direction') - config.set('drying_slope', 'dc', f'{dc}', comment='the distance ' - 'between adjacent cell centers') - - # no run() method is needed - def validate(self): """ Test cases can override this method to perform validation of variables diff --git a/compass/ocean/tests/drying_slope/default/__init__.py b/compass/ocean/tests/drying_slope/default/__init__.py index 784dc2c884..2344360c14 100644 --- a/compass/ocean/tests/drying_slope/default/__init__.py +++ b/compass/ocean/tests/drying_slope/default/__init__.py @@ -18,7 +18,7 @@ class Default(TestCase): The type of vertical coordinate (``sigma``, ``single_layer``, etc.) """ - def __init__(self, test_group, resolution, coord_type): + def __init__(self, test_group, resolution, coord_type, method): """ Create the test case @@ -41,23 +41,32 @@ def __init__(self, test_group, resolution, coord_type): res_name = f'{int(resolution*1e3)}m' else: res_name = f'{int(resolution)}km' - subdir = f'{res_name}/{coord_type}/{name}' + subdir = f'{coord_type}/{method}/{res_name}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) self.add_step(InitialState(test_case=self, resolution=resolution, coord_type=coord_type)) + damping_coeffs = None if coord_type == 'single_layer': - self.add_step(Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, - coord_type=coord_type)) - damping_coeffs = None + forward_step = Forward(test_case=self, resolution=resolution, + ntasks=4, openmp_threads=1, + coord_type=coord_type) + if method == 'ramp': + forward_step.add_namelist_options( + {'config_zero_drying_velocity_ramp': ".true."}) + self.add_step(forward_step) else: damping_coeffs = [0.0025, 0.01] for damping_coeff in damping_coeffs: - self.add_step(Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, - damping_coeff=damping_coeff, - coord_type=coord_type)) + forward_step = Forward(test_case=self, resolution=resolution, + name=f'forward_{damping_coeff}', + ntasks=4, openmp_threads=1, + damping_coeff=damping_coeff, + coord_type=coord_type) + if method == 'ramp': + forward_step.add_namelist_options( + {'config_zero_drying_velocity_ramp': ".true."}) + self.add_step(forward_step) self.damping_coeffs = damping_coeffs self.add_step(Viz(test_case=self, damping_coeffs=damping_coeffs)) diff --git a/compass/ocean/tests/drying_slope/drying_slope.cfg b/compass/ocean/tests/drying_slope/drying_slope.cfg index 3fe040171c..10e89570de 100644 --- a/compass/ocean/tests/drying_slope/drying_slope.cfg +++ b/compass/ocean/tests/drying_slope/drying_slope.cfg @@ -16,7 +16,7 @@ dt_per_km = 30 # config options for visualizing drying slope ouptut [drying_slope_convergence] -resolutions = 0.5, 1, 2 +resolutions = 0.25, 0.5, 1, 2 # config options for visualizing drying slope ouptut [drying_slope_viz] diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 9dfdbebdc8..dddeaa1bf7 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -10,6 +10,7 @@ class Forward(Step): test cases. """ def __init__(self, test_case, resolution, name='forward', subdir=None, + input_path='../initial_state', ntasks=1, min_tasks=None, openmp_threads=1, damping_coeff=None, coord_type='sigma'): """ @@ -50,10 +51,6 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, """ if min_tasks is None: min_tasks = ntasks - input_path = f'../{name}' - input_path = input_path.replace('forward', 'initial_state') - if damping_coeff is not None: - name = f'{name}_{damping_coeff}' super().__init__(test_case=test_case, name=name, subdir=subdir, ntasks=ntasks, min_tasks=min_tasks, @@ -62,6 +59,9 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, self.resolution = resolution self.add_namelist_file('compass.ocean.tests.drying_slope', 'namelist.forward') + if coord_type == 'single_layer': + self.add_namelist_file('compass.ocean.tests.drying_slope', + 'namelist.single_layer.forward') if damping_coeff is not None: # update the Rayleigh damping coeff to the requested value options = {'config_Rayleigh_damping_coeff': f'{damping_coeff}'} diff --git a/compass/ocean/tests/drying_slope/loglaw/__init__.py b/compass/ocean/tests/drying_slope/loglaw/__init__.py index 1e941d41d9..5024f0ce5d 100644 --- a/compass/ocean/tests/drying_slope/loglaw/__init__.py +++ b/compass/ocean/tests/drying_slope/loglaw/__init__.py @@ -18,7 +18,7 @@ class LogLaw(TestCase): The type of vertical coordinate (``sigma``, ``single_layer``, etc.) """ - def __init__(self, test_group, resolution, coord_type): + def __init__(self, test_group, resolution, coord_type, method): """ Create the test case @@ -41,10 +41,11 @@ def __init__(self, test_group, resolution, coord_type): res_name = f'{int(resolution*1e3)}m' else: res_name = f'{int(resolution)}km' - subdir = f'{res_name}/{coord_type}/{name}' + subdir = f'{coord_type}/{method}/{res_name}/{name}' super().__init__(test_group=test_group, name=name, subdir=subdir) - self.add_step(InitialState(test_case=self, coord_type=coord_type)) + self.add_step(InitialState(test_case=self, coord_type=coord_type, + resolution=resolution)) forward_step = Forward(test_case=self, resolution=resolution, ntasks=4, openmp_threads=1, coord_type=coord_type) @@ -53,23 +54,6 @@ def __init__(self, test_group, resolution, coord_type): self.add_step(forward_step) self.add_step(Viz(test_case=self, damping_coeffs=None)) - def configure(self): - """ - Modify the configuration options for this test case. - """ - - resolution = self.resolution - config = self.config - ny = round(28 / resolution) - if resolution < 1.: - ny += 2 - dc = 1e3 * resolution - - config.set('drying_slope', 'ny', f'{ny}', comment='the number of ' - 'mesh cells in the y direction') - config.set('drying_slope', 'dc', f'{dc}', comment='the distance ' - 'between adjacent cell centers') - def validate(self): """ Validate variables against a baseline From f4f5b5c290e7c2d96f57c691af20fbd01f629465 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Thu, 21 Sep 2023 14:27:04 -0500 Subject: [PATCH 07/16] fixup namelist options --- compass/ocean/tests/drying_slope/forward.py | 4 ++-- compass/ocean/tests/drying_slope/namelist.forward | 1 + .../ocean/tests/drying_slope/namelist.single_layer.forward | 2 +- 3 files changed, 4 insertions(+), 3 deletions(-) diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index dddeaa1bf7..2177cd65c6 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -59,9 +59,9 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, self.resolution = resolution self.add_namelist_file('compass.ocean.tests.drying_slope', 'namelist.forward') - if coord_type == 'single_layer': + if coord_type == 'single_layer' or coord_type == 'sigma': self.add_namelist_file('compass.ocean.tests.drying_slope', - 'namelist.single_layer.forward') + f'namelist.{coord_type}.forward') if damping_coeff is not None: # update the Rayleigh damping coeff to the requested value options = {'config_Rayleigh_damping_coeff': f'{damping_coeff}'} diff --git a/compass/ocean/tests/drying_slope/namelist.forward b/compass/ocean/tests/drying_slope/namelist.forward index ecde58b0d0..01df338e9c 100644 --- a/compass/ocean/tests/drying_slope/namelist.forward +++ b/compass/ocean/tests/drying_slope/namelist.forward @@ -14,6 +14,7 @@ config_ALE_thickness_proportionality='weights_only' config_use_wetting_drying=.true. config_prevent_drying=.true. config_drying_min_cell_height=1.0e-3 +config_zero_drying_velocity_ramp_hmax = 1e-1 config_zero_drying_velocity=.true. config_verify_not_dry=.true. config_thickness_flux_type='upwind' diff --git a/compass/ocean/tests/drying_slope/namelist.single_layer.forward b/compass/ocean/tests/drying_slope/namelist.single_layer.forward index 7226997da9..96506c1159 100644 --- a/compass/ocean/tests/drying_slope/namelist.single_layer.forward +++ b/compass/ocean/tests/drying_slope/namelist.single_layer.forward @@ -1,4 +1,4 @@ -config_bottom_drag_mode = 'implicit' +config_implicit_bottom_drag_type = 'constant' config_implicit_constant_bottom_drag_coeff = 3.0e-3 config_disable_thick_vadv = .true. config_disable_thick_sflux = .true. From a6cb5987af2b55f266439fea2938e280889b4274 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Thu, 21 Sep 2023 15:12:01 -0500 Subject: [PATCH 08/16] Update docs --- compass/ocean/tests/drying_slope/analysis.py | 53 +++++++++++++++++-- .../drying_slope/convergence/__init__.py | 9 +++- compass/ocean/tests/drying_slope/forward.py | 2 +- docs/developers_guide/ocean/api.rst | 9 ++-- .../ocean/test_groups/drying_slope.rst | 21 ++++++-- .../ocean/test_groups/drying_slope.rst | 30 +++++++++-- 6 files changed, 106 insertions(+), 18 deletions(-) diff --git a/compass/ocean/tests/drying_slope/analysis.py b/compass/ocean/tests/drying_slope/analysis.py index 8101d59f52..e852c11f30 100644 --- a/compass/ocean/tests/drying_slope/analysis.py +++ b/compass/ocean/tests/drying_slope/analysis.py @@ -11,17 +11,42 @@ class Analysis(Step): """ - A step for visualizing drying slope results, as well as comparison with - analytical solution and ROMS results. + A step for analyzing the convergence of drying slope results and producing + a convergence plot. Attributes ---------- + damping_coeff : float + The Rayleigh damping coefficient used for the forward runs + + resolutions : float + The resolution of the test case + + times : list of float + The times at which to compare to the analytical solution """ def __init__(self, test_case, resolutions, damping_coeff): + """ + Create a forward step + + Parameters + ---------- + test_case : compass.TestCase + The test case this step belongs to + + resolutions : list of floats + The resolution of the test case + + damping_coeff: float + the value of the rayleigh damping coefficient + """ + super().__init__(test_case=test_case, name='analysis') + self.damping_coeff = damping_coeff self.resolutions = resolutions self.times = ['0.05', '0.15', '0.25', '0.30', '0.40', '0.50'] + for resolution in resolutions: if resolution < 1.: res_name = f'{int(resolution*1e3)}m' @@ -34,12 +59,33 @@ def __init__(self, test_case, resolutions, damping_coeff): '.csv' self.add_input_file(filename=filename, target=filename, database='drying_slope') + self.add_output_file(filename='convergence.png') def run(self): + """ + Run this step of the test case + """ self._plot_convergence() def _compute_rmse(self, ds, t): + """ + Get the time step + + Parameters + ---------- + ds : xarray.Dataset + the MPAS dataset containing output from a forward step + t : float + the time to evaluate the RMSE at + + Returns + ------- + rmse : float + the root-mean-squared-error of the MPAS output relative to the + analytical solution at time t + """ + x_exact, ssh_exact = self._exact_solution(t) tidx = int((float(t) / 0.2 + 1e-16) * 24.0) ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], @@ -61,7 +107,6 @@ def _plot_convergence(self): """ Plot convergence curves """ - comparisons = [] cases = {'standard': '../../../standard/convergence/analysis', 'ramp': '../../../ramp/convergence/analysis'} @@ -115,7 +160,7 @@ def _plot_convergence(self): def _exact_solution(self, time): """ - Returns distance, ssh + Returns distance, ssh of the analytic solution """ datafile = f'./r{self.damping_coeff}d{time}-'\ f'analytical.csv' diff --git a/compass/ocean/tests/drying_slope/convergence/__init__.py b/compass/ocean/tests/drying_slope/convergence/__init__.py index 7962ea9e8e..1567e801bc 100644 --- a/compass/ocean/tests/drying_slope/convergence/__init__.py +++ b/compass/ocean/tests/drying_slope/convergence/__init__.py @@ -9,7 +9,7 @@ class Convergence(TestCase): """ - The default drying_slope test case + The convergence drying_slope test case Attributes ---------- @@ -18,6 +18,10 @@ class Convergence(TestCase): coord_type : str The type of vertical coordinate (``sigma``, ``single_layer``, etc.) + + damping_coeffs: list of float + The damping coefficients at which to evaluate convergence. Must be of + length 1. """ def __init__(self, test_group, coord_type, method): @@ -31,6 +35,9 @@ def __init__(self, test_group, coord_type, method): coord_type : str The type of vertical coordinate (``sigma``, ``single_layer``) + + method: str + The wetting-and-drying method (``standard``, ``ramp``) """ name = 'convergence' diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 2177cd65c6..3a51020d29 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -14,7 +14,7 @@ def __init__(self, test_case, resolution, name='forward', subdir=None, ntasks=1, min_tasks=None, openmp_threads=1, damping_coeff=None, coord_type='sigma'): """ - Create a new test case + Create a forward step Parameters ---------- diff --git a/docs/developers_guide/ocean/api.rst b/docs/developers_guide/ocean/api.rst index 653ea00bd2..0bc05dd0b8 100644 --- a/docs/developers_guide/ocean/api.rst +++ b/docs/developers_guide/ocean/api.rst @@ -87,12 +87,16 @@ drying_slope DryingSlope + analysis.Analysis + analysis.Analysis.run + + convergence.Convergence + convergence.Convergence.validate + decomp.Decomp - decomp.Decomp.configure decomp.Decomp.validate default.Default - default.Default.configure default.Default.validate forward.Forward @@ -112,7 +116,6 @@ drying_slope viz.Viz viz.Viz.run - global_convergence ~~~~~~~~~~~~~~~~~~ diff --git a/docs/developers_guide/ocean/test_groups/drying_slope.rst b/docs/developers_guide/ocean/test_groups/drying_slope.rst index 5b58fbdbba..aef6c4cd42 100644 --- a/docs/developers_guide/ocean/test_groups/drying_slope.rst +++ b/docs/developers_guide/ocean/test_groups/drying_slope.rst @@ -22,8 +22,8 @@ duration, bottom drag, and tidal forcing options, as well as shared ``streams.init`` and ``streams.forward`` files that defines ``mesh``, ``input``, ``restart``, ``forcing`` and ``output`` streams. -Namelist options specific to resolutions and vertical coordinates are given in -``namelist.${RES}.*`` and ``namelist.${COORD}*`` files. +Namelist options specific vertical coordinates are given in +``namelist.${COORD}*`` files. initial_state ~~~~~~~~~~~~~ @@ -66,6 +66,13 @@ between the analytical solution, MPAS-Ocean and ROMS. Similar plots are used to create a movie showing the solution from MPAS-Ocean at more fine-grained time intervals. +analysis +~~~~~~~~ + +The class :py:class:`compass.ocean.tests.drying_slope.analysis.Analysis` +produces a convergence plot for a series of forward steps at different +resolutions. It uses the analytical solution available at 5 discrete times t +compute the RMSE. .. _dev_ocean_drying_slope_default: @@ -81,6 +88,14 @@ two cases at different values of ``config_Rayleigh_damping_coeff``, 0.0025 and 0.01, for which there is comparison data. The ``single_layer`` case runs at one value of the implicit bottom drag coefficient. +.. _dev_ocean_drying_slope_convergence: + +convergence +----------- + +The :py:class:`compass.ocean.tests.drying_slope.convergence.Convergence` expands +on the default class to include initial and forward steps for multiple +resolutions and an analysis step to generate a convergence plot. .. _dev_ocean_drying_slope_decomp: @@ -105,8 +120,6 @@ ramp The :py:class:`compass.ocean.tests.drying_slope.ramp.Ramp` is identical to the default class except it sets ``ramp`` to ``True`` for the forward step to enable the ramp feature for wetting and drying. - - .. _dev_ocean_drying_slope_log_law: loglaw diff --git a/docs/users_guide/ocean/test_groups/drying_slope.rst b/docs/users_guide/ocean/test_groups/drying_slope.rst index 37c40b12f5..b3723053f2 100644 --- a/docs/users_guide/ocean/test_groups/drying_slope.rst +++ b/docs/users_guide/ocean/test_groups/drying_slope.rst @@ -64,6 +64,14 @@ The config options for this test case are: # the number of grid cells in x and y nx = 6 + # time step in s per km of horizontal resolution + dt_per_km = 30 + + # config options for visualizing drying slope ouptut + [drying_slope_convergence] + + resolutions = 0.25, 0.5, 1, 2 + # config options for visualizing drying slope ouptut [drying_slope_viz] @@ -81,12 +89,24 @@ All units are mks. default ------- -``ocean/drying_slope/${RES}/${COORD}/default`` is the default version of the -drying slope test case for two short (12h) test runs with two different drag -coefficients and validation of sea surface height through visual inspection +``ocean/drying_slope/${COORD}/${METHOD}/${RES}/default`` is the default version +of the drying slope test case for two short (12h) test runs with two different +drag coefficients and validation of sea surface height through visual inspection against analytic and ROMS solutions. ``RES`` is either 250m or 1km. ``COORD`` -is either ``single_layer`` or ``sigma``. Rayleigh drag is not compatible with -``single_layer`` so implicit drag with a constant coefficient is used. +is either ``single_layer`` or ``sigma``. The wetting and drying ``METHOD`` is +either ``standard`` or ``ramp``. The ``ramp`` method ramps velocities and +velocity tendencies over a given layer thickness range rather than a binary +switch at the minimum thickness. + +convergence +----------- + +``ocean/drying_slope/${COORD}/${METHOD}/convergence`` is a convergence test in +horizontal resolution and time. It produces a convergence plot for the +resolutions specified in the config file. ``COORD`` is either ``single_layer`` +or ``sigma``. The wetting and drying ``METHOD`` is either ``standard`` or +``ramp``. If the other of the two methods has already been run, its convergence +curve will be included in the plot. decomp ------ From 88e4ebc4d55296600f62cd105ae810f7a477e1c4 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Tue, 26 Sep 2023 10:44:03 -0500 Subject: [PATCH 09/16] fixup viz --- .../ocean/tests/drying_slope/viz/__init__.py | 63 +++++++++---------- 1 file changed, 30 insertions(+), 33 deletions(-) diff --git a/compass/ocean/tests/drying_slope/viz/__init__.py b/compass/ocean/tests/drying_slope/viz/__init__.py index 4eb1ed7b51..1c643a75b7 100644 --- a/compass/ocean/tests/drying_slope/viz/__init__.py +++ b/compass/ocean/tests/drying_slope/viz/__init__.py @@ -1,9 +1,10 @@ import os -import xarray -import numpy +import subprocess + import matplotlib.pyplot as plt +import numpy as np import pandas as pd -import subprocess +import xarray from compass.step import Step @@ -62,9 +63,7 @@ def run(self): Run this step of the test case """ section = self.config['paths'] - datapath = section.get('ocean_database_root') section = self.config['vertical_grid'] - vert_levels = section.get('vert_levels') section = self.config['drying_slope_viz'] generate_movie = section.getboolean('generate_movie') @@ -84,7 +83,7 @@ def run(self): outFolder=outFolder, extension=movie_format) def _forcing(self, t): - ssh = 10. * numpy.sin(t * numpy.pi / 12.) - 10. + ssh = 10. * np.sin(t * np.pi / 12.) - 10. return ssh def _plot_ssh_time_series(self, outFolder='.'): @@ -95,11 +94,10 @@ def _plot_ssh_time_series(self, outFolder='.'): (2013) test case. """ colors = {'MPAS-O': 'k', 'analytical': 'b', 'ROMS': 'g'} - xSsh = numpy.linspace(0, 12.0, 100) - ySsh = 10.0*numpy.sin(xSsh*numpy.pi/12.0) - 10.0 + xSsh = np.linspace(0, 12.0, 100) + ySsh = 10.0 * np.sin(xSsh * np.pi / 12.0) - 10.0 figsize = [6.4, 4.8] - markersize = 20 damping_coeffs = self.damping_coeffs if damping_coeffs is None: @@ -112,11 +110,10 @@ def _plot_ssh_time_series(self, outFolder='.'): fig, _ = plt.subplots(nrows=naxes, ncols=1, figsize=figsize, dpi=100) for i in range(naxes): - ax = plt.subplot(naxes, 1, i+1) + ax = plt.subplot(naxes, 1, i + 1) ds = xarray.open_dataset(ncFilename[i]) - ssh = ds.ssh ympas = ds.ssh.where(ds.tidalInputMask).mean('nCells').values - xmpas = numpy.linspace(0, 1.0, len(ds.xtime))*12.0 + xmpas = np.linspace(0, 1.0, len(ds.xtime)) * 12.0 ax.plot(xmpas, ympas, marker='o', label='MPAS-O forward', color=colors['MPAS-O']) ax.plot(xSsh, ySsh, lw=3, label='analytical', @@ -140,7 +137,7 @@ def _plot_ssh_validation(self, outFolder='.'): colors = {'MPAS-O': 'k', 'analytical': 'b', 'ROMS': 'g'} locs = [7.2, 2.2, 0.2, 1.2, 4.2, 9.3] - locs = 0.92 - numpy.divide(locs, 11.) + locs = 0.92 - np.divide(locs, 11.) damping_coeffs = self.damping_coeffs times = self.times @@ -156,16 +153,16 @@ def _plot_ssh_validation(self, outFolder='.'): ncFilename = [f'output_{damping_coeff}.nc' for damping_coeff in damping_coeffs] - xBed = numpy.linspace(0, 25, 100) - yBed = 10.0/25.0*xBed + xBed = np.linspace(0, 25, 100) + yBed = 10.0 / 25.0 * xBed fig, _ = plt.subplots(nrows=naxes, ncols=1, sharex=True) for i in range(naxes): - ax = plt.subplot(naxes, 1, i+1) + ax = plt.subplot(naxes, 1, i + 1) ds = xarray.open_dataset(ncFilename[i]) - ds = ds.drop_vars(numpy.setdiff1d([j for j in ds.variables], - ['yCell', 'ssh'])) + ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], + ['yCell', 'ssh'])) ax.plot(xBed, yBed, '-k', lw=3) ax.set_xlim(0, 25) @@ -180,13 +177,13 @@ def _plot_ssh_validation(self, outFolder='.'): # Plot MPAS-O data # factor of 1e- needed to account for annoying round-off issue # to get right time slices - plottime = int((float(atime)/0.2 + 1e-16)*24.0) + plottime = int((float(atime) / 0.2 + 1e-16) * 24.0) ymean = ds.isel(Time=plottime).groupby('yCell').mean( - dim=xarray.ALL_DIMS) - x = ymean.yCell.values/1000.0 + dim=xarray.ALL_DIMS) + x = ymean.yCell.values / 1000.0 y = ymean.ssh.values - mpas = ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) + ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) ax.text(1, ay, atime + ' days', size=8, transform=ax.transAxes) if damping_coeffs is not None: @@ -225,7 +222,7 @@ def _plot_ssh_validation_for_movie(self, outFolder='.'): colors = {'MPAS-O': 'k', 'analytical': 'b', 'ROMS': 'g'} locs = [7.2, 2.2, 0.2, 1.2, 4.2, 9.3] - locs = 0.92 - numpy.divide(locs, 11.) + locs = 0.92 - np.divide(locs, 11.) damping_coeffs = self.damping_coeffs if damping_coeffs is None: @@ -241,32 +238,32 @@ def _plot_ssh_validation_for_movie(self, outFolder='.'): times = self.times datatypes = self.datatypes - xBed = numpy.linspace(0, 25, 100) - yBed = 10.0/25.0*xBed + xBed = np.linspace(0, 25, 100) + yBed = 10.0 / 25.0 * xBed ii = 0 # Plot profiles over the 12h simulation duration - for itime in numpy.linspace(0, 0.5, 5*12+1): + for itime in np.linspace(0, 0.5, 5 * 12 + 1): - plottime = int((float(itime)/0.2 + 1e-16)*24.0) + plottime = int((float(itime) / 0.2 + 1e-16) * 24.0) fig, _ = plt.subplots(nrows=naxes, ncols=1, sharex=True) for i in range(naxes): - ax = plt.subplot(naxes, 1, i+1) + ax = plt.subplot(naxes, 1, i + 1) ds = xarray.open_dataset(ncFilename[i]) - ds = ds.drop_vars(numpy.setdiff1d([j for j in ds.variables], - ['yCell', 'ssh'])) + ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], + ['yCell', 'ssh'])) # Plot MPAS-O snapshots # factor of 1e- needed to account for annoying round-off issue # to get right time slices ymean = ds.isel(Time=plottime).groupby('yCell').mean( - dim=xarray.ALL_DIMS) - x = ymean.yCell.values/1000.0 + dim=xarray.ALL_DIMS) + x = ymean.yCell.values / 1000.0 y = ymean.ssh.values ax.plot(xBed, yBed, '-k', lw=3) - mpas = ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) + ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) ax.set_ylim(-1, 11) ax.set_xlim(0, 25) From ed9ea1cd16c202751e9b25f6ef1119a2a73a020e Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Wed, 27 Sep 2023 12:30:09 -0500 Subject: [PATCH 10/16] Move thin film thickness to cfg option --- compass/ocean/tests/drying_slope/drying_slope.cfg | 3 +++ compass/ocean/tests/drying_slope/forward.py | 6 +++++- compass/ocean/tests/drying_slope/initial_state.py | 12 ++++++++---- compass/ocean/tests/drying_slope/namelist.forward | 1 - compass/ocean/tests/drying_slope/namelist.init | 1 - 5 files changed, 16 insertions(+), 7 deletions(-) diff --git a/compass/ocean/tests/drying_slope/drying_slope.cfg b/compass/ocean/tests/drying_slope/drying_slope.cfg index 10e89570de..c858234237 100644 --- a/compass/ocean/tests/drying_slope/drying_slope.cfg +++ b/compass/ocean/tests/drying_slope/drying_slope.cfg @@ -4,6 +4,9 @@ # Number of vertical levels vert_levels = 10 +# Thickness of each layer in the thin film region +thin_film_thickness = 1.0e-3 + # config options for drying slope test cases [drying_slope] diff --git a/compass/ocean/tests/drying_slope/forward.py b/compass/ocean/tests/drying_slope/forward.py index 3a51020d29..36c9c23527 100644 --- a/compass/ocean/tests/drying_slope/forward.py +++ b/compass/ocean/tests/drying_slope/forward.py @@ -93,7 +93,11 @@ def run(self): Run this step of the test case """ dt = self.get_dt() - self.update_namelist_at_runtime(options={'config_dt': f"'{dt}'"}, + section = self.config['vertical_grid'] + thin_film_thickness = section.getfloat('thin_film_thickness') + options = {'config_dt': f"'{dt}'", + 'config_drying_min_cell_height': f'{thin_film_thickness}'} + self.update_namelist_at_runtime(options=options, out_name='namelist.ocean') run_model(self) diff --git a/compass/ocean/tests/drying_slope/initial_state.py b/compass/ocean/tests/drying_slope/initial_state.py index be4711c2eb..97d312af42 100644 --- a/compass/ocean/tests/drying_slope/initial_state.py +++ b/compass/ocean/tests/drying_slope/initial_state.py @@ -47,16 +47,20 @@ def run(self): config = self.config logger = self.logger - config = self.config section = config['vertical_grid'] coord_type = self.coord_type + thin_film_thickness = section.getfloat('thin_film_thickness') + 1.0e-9 if coord_type == 'single_layer': - options = {'config_tidal_boundary_vert_levels': '1'} + options = {'config_tidal_boundary_vert_levels': '1', + 'config_drying_min_cell_height': + f'{thin_film_thickness}'} self.update_namelist_at_runtime(options) else: - vert_levels = section.get('vert_levels') + vert_levels = section.getint('vert_levels') options = {'config_tidal_boundary_vert_levels': f'{vert_levels}', - 'config_tidal_boundary_layer_type': f"'{coord_type}'"} + 'config_tidal_boundary_layer_type': f"'{coord_type}'", + 'config_drying_min_cell_height': + f'{thin_film_thickness}'} self.update_namelist_at_runtime(options) # Determine mesh parameters diff --git a/compass/ocean/tests/drying_slope/namelist.forward b/compass/ocean/tests/drying_slope/namelist.forward index 01df338e9c..3fcac6b181 100644 --- a/compass/ocean/tests/drying_slope/namelist.forward +++ b/compass/ocean/tests/drying_slope/namelist.forward @@ -13,7 +13,6 @@ config_vert_coord_movement='impermeable_interfaces' config_ALE_thickness_proportionality='weights_only' config_use_wetting_drying=.true. config_prevent_drying=.true. -config_drying_min_cell_height=1.0e-3 config_zero_drying_velocity_ramp_hmax = 1e-1 config_zero_drying_velocity=.true. config_verify_not_dry=.true. diff --git a/compass/ocean/tests/drying_slope/namelist.init b/compass/ocean/tests/drying_slope/namelist.init index f1ff98bfc2..a15464bd7c 100644 --- a/compass/ocean/tests/drying_slope/namelist.init +++ b/compass/ocean/tests/drying_slope/namelist.init @@ -6,6 +6,5 @@ config_tidal_boundary_left_bottom_depth=0.0 config_tidal_boundary_use_distances=.true. config_tidal_forcing_monochromatic_baseline=10.0 config_use_wetting_drying=.true. -config_drying_min_cell_height=1.000001e-3 config_use_tidal_forcing=.true. config_write_cull_cell_mask=.false. From 95213932285833da99f991400c8867734418b64b Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Tue, 3 Oct 2023 16:56:01 -0500 Subject: [PATCH 11/16] Remove Ramp class --- .../ocean/tests/drying_slope/ramp/__init__.py | 83 ------------------- docs/developers_guide/ocean/api.rst | 4 - 2 files changed, 87 deletions(-) delete mode 100644 compass/ocean/tests/drying_slope/ramp/__init__.py diff --git a/compass/ocean/tests/drying_slope/ramp/__init__.py b/compass/ocean/tests/drying_slope/ramp/__init__.py deleted file mode 100644 index 793d83cfeb..0000000000 --- a/compass/ocean/tests/drying_slope/ramp/__init__.py +++ /dev/null @@ -1,83 +0,0 @@ -from compass.ocean.tests.drying_slope.forward import Forward -from compass.ocean.tests.drying_slope.initial_state import InitialState -from compass.ocean.tests.drying_slope.viz import Viz -from compass.testcase import TestCase -from compass.validate import compare_variables - - -class Ramp(TestCase): - """ - The drying_slope test case with ramped wetting-velocity factor - - Attributes - ---------- - resolution : float - The resolution of the test case in km - - coord_type : str - The type of vertical coordinate (``sigma``, ``single_layer``, etc.) - """ - - def __init__(self, test_group, resolution, coord_type): - """ - Create the test case - - Parameters - ---------- - test_group : compass.ocean.tests.drying_slope.DryingSlope - The test group that this test case belongs to - - resolution : float - The resolution of the test case in km - - coord_type : str - The type of vertical coordinate (``sigma``, ``single_layer``) - """ - name = 'ramp' - - self.resolution = resolution - self.coord_type = coord_type - if resolution < 1.: - res_name = f'{int(resolution*1e3)}m' - else: - res_name = f'{int(resolution)}km' - subdir = f'{res_name}/{coord_type}/{name}' - super().__init__(test_group=test_group, name=name, - subdir=subdir) - self.add_step(InitialState(test_case=self, resolution=resolution, - coord_type=coord_type)) - if coord_type == 'single_layer': - forward_step = Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, - coord_type=coord_type) - damping_coeffs = None - forward_step.add_namelist_options( - {'config_zero_drying_velocity_ramp': ".true."}) - self.add_step(forward_step) - else: - damping_coeffs = [0.0025, 0.01] - for damping_coeff in damping_coeffs: - forward_step = Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, - damping_coeff=damping_coeff, - coord_type=coord_type) - forward_step.add_namelist_options( - {'config_zero_drying_velocity_ramp': ".true."}) - self.add_step(forward_step) - self.damping_coeffs = damping_coeffs - self.add_step(Viz(test_case=self, damping_coeffs=damping_coeffs)) - - def validate(self): - """ - Validate variables against a baseline - """ - damping_coeffs = self.damping_coeffs - variables = ['layerThickness', 'normalVelocity'] - if damping_coeffs is not None: - for damping_coeff in damping_coeffs: - compare_variables(test_case=self, variables=variables, - filename1=f'forward_{damping_coeff}/' - 'output.nc') - else: - compare_variables(test_case=self, variables=variables, - filename1='forward/output.nc') diff --git a/docs/developers_guide/ocean/api.rst b/docs/developers_guide/ocean/api.rst index 0bc05dd0b8..8501f75131 100644 --- a/docs/developers_guide/ocean/api.rst +++ b/docs/developers_guide/ocean/api.rst @@ -109,10 +109,6 @@ drying_slope loglaw.LogLaw.configure loglaw.LogLaw.validate - ramp.Ramp - ramp.Ramp.configure - ramp.Ramp.validate - viz.Viz viz.Viz.run From 82758b7571c9f1e8e253577918b9728c293116fa Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Mon, 16 Oct 2023 12:06:55 -0500 Subject: [PATCH 12/16] Clean-up viz step --- .../ocean/tests/drying_slope/viz/__init__.py | 114 ++---------------- 1 file changed, 13 insertions(+), 101 deletions(-) diff --git a/compass/ocean/tests/drying_slope/viz/__init__.py b/compass/ocean/tests/drying_slope/viz/__init__.py index 1c643a75b7..1b1daf7ab8 100644 --- a/compass/ocean/tests/drying_slope/viz/__init__.py +++ b/compass/ocean/tests/drying_slope/viz/__init__.py @@ -4,7 +4,7 @@ import matplotlib.pyplot as plt import numpy as np import pandas as pd -import xarray +import xarray as xr from compass.step import Step @@ -67,7 +67,7 @@ def run(self): section = self.config['drying_slope_viz'] generate_movie = section.getboolean('generate_movie') - self._plot_ssh_validation() + self._plot_ssh_validation(times=self.times) self._plot_ssh_time_series() if generate_movie: frames_per_second = section.getint('frames_per_second') @@ -78,7 +78,9 @@ def run(self): os.makedirs(os.path.join(os.getcwd(), outFolder)) except OSError: pass - self._plot_ssh_validation_for_movie(outFolder=outFolder) + for tidx, itime in enumerate(np.linspace(0, 0.5, 5 * 12 + 1)): + self._plot_ssh_validation(times=[itime], tidx=tidx, + outFolder=outFolder) self._images_to_movies(framesPerSecond=frames_per_second, outFolder=outFolder, extension=movie_format) @@ -111,7 +113,7 @@ def _plot_ssh_time_series(self, outFolder='.'): for i in range(naxes): ax = plt.subplot(naxes, 1, i + 1) - ds = xarray.open_dataset(ncFilename[i]) + ds = xr.open_dataset(ncFilename[i]) ympas = ds.ssh.where(ds.tidalInputMask).mean('nCells').values xmpas = np.linspace(0, 1.0, len(ds.xtime)) * 12.0 ax.plot(xmpas, ympas, marker='o', label='MPAS-O forward', @@ -129,7 +131,7 @@ def _plot_ssh_time_series(self, outFolder='.'): plt.close(fig) - def _plot_ssh_validation(self, outFolder='.'): + def _plot_ssh_validation(self, times, tidx=None, outFolder='.'): """ Plot ssh as a function of along-channel distance for all times for which there is validation data @@ -140,7 +142,6 @@ def _plot_ssh_validation(self, outFolder='.'): locs = 0.92 - np.divide(locs, 11.) damping_coeffs = self.damping_coeffs - times = self.times datatypes = self.datatypes if damping_coeffs is None: @@ -160,7 +161,7 @@ def _plot_ssh_validation(self, outFolder='.'): for i in range(naxes): ax = plt.subplot(naxes, 1, i + 1) - ds = xarray.open_dataset(ncFilename[i]) + ds = xr.open_dataset(ncFilename[i]) ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], ['yCell', 'ssh'])) @@ -179,7 +180,7 @@ def _plot_ssh_validation(self, outFolder='.'): # to get right time slices plottime = int((float(atime) / 0.2 + 1e-16) * 24.0) ymean = ds.isel(Time=plottime).groupby('yCell').mean( - dim=xarray.ALL_DIMS) + dim=xr.ALL_DIMS) x = ymean.yCell.values / 1000.0 y = ymean.ssh.values @@ -207,101 +208,12 @@ def _plot_ssh_validation(self, outFolder='.'): rotation='vertical') fig.text(0.5, 0.02, 'Along channel distance (km)', ha='center') - fig.savefig(f'{outFolder}/ssh_depth_section.png', dpi=200) + filename = f'{outFolder}/ssh_depth_section' + if tidx is not None: + filename = f'{filename}_t{tidx}' + fig.savefig(filename, dpi=200, format='png') plt.close(fig) - def _plot_ssh_validation_for_movie(self, outFolder='.'): - """ - Compare ssh along the channel at different time slices with the - analytical solution and ROMS results. - - Parameters - ---------- - - """ - colors = {'MPAS-O': 'k', 'analytical': 'b', 'ROMS': 'g'} - - locs = [7.2, 2.2, 0.2, 1.2, 4.2, 9.3] - locs = 0.92 - np.divide(locs, 11.) - - damping_coeffs = self.damping_coeffs - if damping_coeffs is None: - naxes = 1 - nhandles = 1 - ncFilename = ['output.nc'] - else: - naxes = len(damping_coeffs) - nhandles = naxes + 2 - ncFilename = [f'output_{damping_coeff}.nc' - for damping_coeff in damping_coeffs] - - times = self.times - datatypes = self.datatypes - - xBed = np.linspace(0, 25, 100) - yBed = 10.0 / 25.0 * xBed - - ii = 0 - # Plot profiles over the 12h simulation duration - for itime in np.linspace(0, 0.5, 5 * 12 + 1): - - plottime = int((float(itime) / 0.2 + 1e-16) * 24.0) - - fig, _ = plt.subplots(nrows=naxes, ncols=1, sharex=True) - - for i in range(naxes): - ax = plt.subplot(naxes, 1, i + 1) - ds = xarray.open_dataset(ncFilename[i]) - ds = ds.drop_vars(np.setdiff1d([j for j in ds.variables], - ['yCell', 'ssh'])) - - # Plot MPAS-O snapshots - # factor of 1e- needed to account for annoying round-off issue - # to get right time slices - ymean = ds.isel(Time=plottime).groupby('yCell').mean( - dim=xarray.ALL_DIMS) - x = ymean.yCell.values / 1000.0 - y = ymean.ssh.values - ax.plot(xBed, yBed, '-k', lw=3) - ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) - - ax.set_ylim(-1, 11) - ax.set_xlim(0, 25) - ax.invert_yaxis() - ax.spines['top'].set_visible(False) - ax.spines['right'].set_visible(False) - ax.legend(frameon=False, loc='lower left') - ax.set_title(f't = {itime:.3f} days') - if damping_coeffs is not None: - ax.text(0.5, 5, 'r = ' + str(damping_coeffs[i])) - # Plot comparison data - for atime, ay in zip(times, locs): - ax.text(1, ay, f'{atime} days', size=8, - transform=ax.transAxes) - - for datatype in datatypes: - datafile = f'./r{damping_coeffs[i]}d{atime}-'\ - f'{datatype.lower()}.csv' - data = pd.read_csv(datafile, header=None) - ax.scatter(data[0], data[1], marker='.', - color=colors[datatype], label=datatype) - - h, l0 = ax.get_legend_handles_labels() - ax.legend(h[0:nhandles], l0[0:nhandles], frameon=False, - loc='lower left') - ax.set_title(f't = {itime:.3f} days') - - ds.close() - - fig.text(0.04, 0.5, 'Channel depth (m)', va='center', - rotation='vertical') - fig.text(0.5, 0.02, 'Along channel distance (km)', ha='center') - - fig.savefig(f'{outFolder}/ssh_depth_section_{ii:03d}.png', dpi=200) - - plt.close(fig) - ii += 1 - def _images_to_movies(self, outFolder='.', framesPerSecond=30, extension='mp4', overwrite=True): """ From 46362541f22a5064790678e1c6c7cd58ea4b7665 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Mon, 16 Oct 2023 14:57:29 -0500 Subject: [PATCH 13/16] Add conv test to wetdry suite --- compass/ocean/suites/wetdry.txt | 2 ++ 1 file changed, 2 insertions(+) diff --git a/compass/ocean/suites/wetdry.txt b/compass/ocean/suites/wetdry.txt index c5c15e221e..0a1307c10c 100644 --- a/compass/ocean/suites/wetdry.txt +++ b/compass/ocean/suites/wetdry.txt @@ -14,6 +14,8 @@ ocean/drying_slope/sigma/standard/250m/loglaw ocean/drying_slope/sigma/standard/1km/loglaw ocean/drying_slope/single_layer/standard/250m/loglaw ocean/drying_slope/single_layer/standard/1km/loglaw +ocean/drying_slope/sigma/standard/convergence +ocean/drying_slope/sigma/ramp/convergence ocean/dam_break/40cm/default ocean/dam_break/40cm/ramp ocean/dam_break/120cm/default From d299637f22b2ae1fae9691585e219a44dbe2b964 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Mon, 16 Oct 2023 14:58:10 -0500 Subject: [PATCH 14/16] fixup rebase --- compass/ocean/tests/drying_slope/__init__.py | 2 +- compass/ocean/tests/drying_slope/decomp/__init__.py | 8 +++++++- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/compass/ocean/tests/drying_slope/__init__.py b/compass/ocean/tests/drying_slope/__init__.py index 45553fa801..5aa6d0d8fb 100644 --- a/compass/ocean/tests/drying_slope/__init__.py +++ b/compass/ocean/tests/drying_slope/__init__.py @@ -1,5 +1,5 @@ -from compass.ocean.tests.drying_slope.decomp import Decomp from compass.ocean.tests.drying_slope.convergence import Convergence +from compass.ocean.tests.drying_slope.decomp import Decomp from compass.ocean.tests.drying_slope.default import Default from compass.ocean.tests.drying_slope.loglaw import LogLaw from compass.testgroup import TestGroup diff --git a/compass/ocean/tests/drying_slope/decomp/__init__.py b/compass/ocean/tests/drying_slope/decomp/__init__.py index 766d933c85..5b99b7c170 100644 --- a/compass/ocean/tests/drying_slope/decomp/__init__.py +++ b/compass/ocean/tests/drying_slope/decomp/__init__.py @@ -13,9 +13,12 @@ class Decomp(TestCase): ---------- resolution : str The resolution of the test case + + coord_type : str + The type of vertical coordinate (``sigma``, ``single_layer``, etc.) """ - def __init__(self, test_group, resolution, coord_type): + def __init__(self, test_group, resolution, coord_type, method): """ Create the test case @@ -29,6 +32,9 @@ def __init__(self, test_group, resolution, coord_type): coord_type : str The type of vertical coordinate (``sigma``, ``single_layer``) + + method : str + The type of wetting-and-drying algorithm """ name = 'decomp' self.resolution = resolution From bcb9f2e61d20d08108dd9dddb505dbc63b6d1622 Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Mon, 16 Oct 2023 14:58:59 -0500 Subject: [PATCH 15/16] Add config options for tasks --- .../drying_slope/convergence/__init__.py | 10 +++++++++- .../tests/drying_slope/default/__init__.py | 19 +++++++++++++++++-- .../ocean/tests/drying_slope/drying_slope.cfg | 6 ++++++ .../tests/drying_slope/loglaw/__init__.py | 13 ++++++++++++- 4 files changed, 44 insertions(+), 4 deletions(-) diff --git a/compass/ocean/tests/drying_slope/convergence/__init__.py b/compass/ocean/tests/drying_slope/convergence/__init__.py index 1567e801bc..5e09e7c8f0 100644 --- a/compass/ocean/tests/drying_slope/convergence/__init__.py +++ b/compass/ocean/tests/drying_slope/convergence/__init__.py @@ -1,3 +1,5 @@ +from numpy import ceil + from compass.config import CompassConfigParser from compass.ocean.tests.drying_slope.analysis import Analysis from compass.ocean.tests.drying_slope.forward import Forward @@ -77,6 +79,9 @@ def _setup_steps(self, config, subdir, method): self.steps_to_run = list() self.resolutions = resolutions + section = config['drying_slope'] + ntasks_baseline = section.getint('ntasks_baseline') + min_tasks = section.getint('min_tasks') for resolution in self.resolutions: @@ -89,10 +94,13 @@ def _setup_steps(self, config, subdir, method): name=init_name, resolution=resolution, coord_type=self.coord_type)) + ntasks = max(min_tasks, + int(ceil(ntasks_baseline / resolution**2.))) forward_step = Forward(test_case=self, resolution=resolution, name=f'forward_{res_name}', input_path=f'../{init_name}', - ntasks=4, openmp_threads=1, + ntasks=ntasks, min_tasks=min_tasks, + openmp_threads=1, damping_coeff=self.damping_coeffs[0], coord_type=self.coord_type) if method == 'ramp': diff --git a/compass/ocean/tests/drying_slope/default/__init__.py b/compass/ocean/tests/drying_slope/default/__init__.py index 2344360c14..3af74a6bba 100644 --- a/compass/ocean/tests/drying_slope/default/__init__.py +++ b/compass/ocean/tests/drying_slope/default/__init__.py @@ -1,3 +1,6 @@ +from numpy import ceil + +from compass.config import CompassConfigParser from compass.ocean.tests.drying_slope.forward import Forward from compass.ocean.tests.drying_slope.initial_state import InitialState from compass.ocean.tests.drying_slope.viz import Viz @@ -32,6 +35,9 @@ def __init__(self, test_group, resolution, coord_type, method): coord_type : str The type of vertical coordinate (``sigma``, ``single_layer``) + + method : str + The type of wetting-and-drying algorithm """ name = 'default' @@ -47,9 +53,17 @@ def __init__(self, test_group, resolution, coord_type, method): self.add_step(InitialState(test_case=self, resolution=resolution, coord_type=coord_type)) damping_coeffs = None + config = CompassConfigParser() + config.add_from_package('compass.ocean.tests.drying_slope', + 'drying_slope.cfg') + section = config['drying_slope'] + ntasks_baseline = section.getint('ntasks_baseline') + min_tasks = section.getint('min_tasks') + ntasks = max(min_tasks, int(ceil(ntasks_baseline / resolution**2.))) if coord_type == 'single_layer': forward_step = Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, + ntasks=ntasks, min_tasks=min_tasks, + openmp_threads=1, coord_type=coord_type) if method == 'ramp': forward_step.add_namelist_options( @@ -60,7 +74,8 @@ def __init__(self, test_group, resolution, coord_type, method): for damping_coeff in damping_coeffs: forward_step = Forward(test_case=self, resolution=resolution, name=f'forward_{damping_coeff}', - ntasks=4, openmp_threads=1, + ntasks=ntasks, min_tasks=min_tasks, + openmp_threads=1, damping_coeff=damping_coeff, coord_type=coord_type) if method == 'ramp': diff --git a/compass/ocean/tests/drying_slope/drying_slope.cfg b/compass/ocean/tests/drying_slope/drying_slope.cfg index c858234237..a9d4c58c9d 100644 --- a/compass/ocean/tests/drying_slope/drying_slope.cfg +++ b/compass/ocean/tests/drying_slope/drying_slope.cfg @@ -16,6 +16,12 @@ nx = 6 # time step in s per km of horizontal resolution dt_per_km = 30 +# Number of tasks at 1km resolution +ntasks_baseline = 4 + +# Minimum number of tasks +min_tasks = 1 + # config options for visualizing drying slope ouptut [drying_slope_convergence] diff --git a/compass/ocean/tests/drying_slope/loglaw/__init__.py b/compass/ocean/tests/drying_slope/loglaw/__init__.py index 5024f0ce5d..0cbd50b721 100644 --- a/compass/ocean/tests/drying_slope/loglaw/__init__.py +++ b/compass/ocean/tests/drying_slope/loglaw/__init__.py @@ -1,3 +1,6 @@ +from numpy import ceil + +from compass.config import CompassConfigParser from compass.ocean.tests.drying_slope.forward import Forward from compass.ocean.tests.drying_slope.initial_state import InitialState from compass.ocean.tests.drying_slope.viz import Viz @@ -46,8 +49,16 @@ def __init__(self, test_group, resolution, coord_type, method): subdir=subdir) self.add_step(InitialState(test_case=self, coord_type=coord_type, resolution=resolution)) + config = CompassConfigParser() + config.add_from_package('compass.ocean.tests.drying_slope', + 'drying_slope.cfg') + section = config['drying_slope'] + ntasks_baseline = section.getint('ntasks_baseline') + min_tasks = section.getint('min_tasks') + ntasks = max(min_tasks, int(ceil(ntasks_baseline / resolution**2.))) forward_step = Forward(test_case=self, resolution=resolution, - ntasks=4, openmp_threads=1, + ntasks=ntasks, min_tasks=min_tasks, + openmp_threads=1, coord_type=coord_type) forward_step.add_namelist_options( {'config_implicit_bottom_drag_type': "'loglaw'"}) From 8e2af4abf09fe10321463ad14b5375a9628557de Mon Sep 17 00:00:00 2001 From: Carolyn Begeman Date: Tue, 17 Oct 2023 12:12:08 -0500 Subject: [PATCH 16/16] fixup viz --- .../ocean/tests/drying_slope/viz/__init__.py | 50 ++++++++++++++----- 1 file changed, 38 insertions(+), 12 deletions(-) diff --git a/compass/ocean/tests/drying_slope/viz/__init__.py b/compass/ocean/tests/drying_slope/viz/__init__.py index 1b1daf7ab8..875859d94f 100644 --- a/compass/ocean/tests/drying_slope/viz/__init__.py +++ b/compass/ocean/tests/drying_slope/viz/__init__.py @@ -44,6 +44,8 @@ def __init__(self, test_case, damping_coeffs=None): self.times = times self.datatypes = datatypes + self.add_input_file(filename='init.nc', + target='../initial_state/ocean.nc') if damping_coeffs is None: self.add_input_file(filename='output.nc', target='../forward/output.nc') @@ -154,6 +156,12 @@ def _plot_ssh_validation(self, times, tidx=None, outFolder='.'): ncFilename = [f'output_{damping_coeff}.nc' for damping_coeff in damping_coeffs] + ds_mesh = xr.open_dataset('init.nc') + mesh_ymean = ds_mesh.isel(Time=0).groupby('yCell').mean( + dim=xr.ALL_DIMS) + bottom_depth = mesh_ymean.bottomDepth.values + x = mesh_ymean.yCell.values / 1000.0 + xBed = np.linspace(0, 25, 100) yBed = 10.0 / 25.0 * xBed @@ -181,21 +189,39 @@ def _plot_ssh_validation(self, times, tidx=None, outFolder='.'): plottime = int((float(atime) / 0.2 + 1e-16) * 24.0) ymean = ds.isel(Time=plottime).groupby('yCell').mean( dim=xr.ALL_DIMS) - x = ymean.yCell.values / 1000.0 y = ymean.ssh.values ax.plot(x, -y, label='MPAS-O', color=colors['MPAS-O']) - ax.text(1, ay, atime + ' days', size=8, - transform=ax.transAxes) if damping_coeffs is not None: ax.text(0.5, 5, 'r = ' + str(damping_coeffs[i])) # Plot comparison data - for datatype in datatypes: - datafile = f'./r{damping_coeffs[i]}d{atime}-'\ - f'{datatype.lower()}.csv' - data = pd.read_csv(datafile, header=None) - ax.scatter(data[0], data[1], marker='.', - color=colors[datatype], label=datatype) + if tidx is not None: + plt.title(f'{atime:03f} days') + for atime, ay in zip(self.times, locs): + ax.text(1, ay, f'{atime} days', size=8, + transform=ax.transAxes) + for datatype in datatypes: + datafile = f'./r{damping_coeffs[i]}d{atime}-'\ + f'{datatype.lower()}.csv' + if os.path.exists(datafile): + data = pd.read_csv(datafile, header=None) + ax.scatter(data[0], data[1], marker='.', + color=colors[datatype], + label=datatype) + else: + ax.text(1, ay, f'{atime} days', size=8, + transform=ax.transAxes) + for datatype in datatypes: + datafile = f'./r{damping_coeffs[i]}d{atime}-'\ + f'{datatype.lower()}.csv' + if os.path.exists(datafile): + data = pd.read_csv(datafile, header=None) + ax.scatter(data[0], data[1], marker='.', + color=colors[datatype], + label=datatype) + # Plot bottom depth, but line will not be visible unless bottom + # depth is incorrect + ax.plot(x, bottom_depth, ':k') ax.legend(frameon=False, loc='lower left') ds.close() @@ -210,8 +236,8 @@ def _plot_ssh_validation(self, times, tidx=None, outFolder='.'): filename = f'{outFolder}/ssh_depth_section' if tidx is not None: - filename = f'{filename}_t{tidx}' - fig.savefig(filename, dpi=200, format='png') + filename = f'{filename}_t{tidx:03d}' + fig.savefig(f'{filename}.png', dpi=200, format='png') plt.close(fig) def _images_to_movies(self, outFolder='.', framesPerSecond=30, @@ -229,7 +255,7 @@ def _images_to_movies(self, outFolder='.', framesPerSecond=30, outFileName = f'{outFolder}/{prefix}.{extension}' if overwrite or not os.path.exists(outFileName): - imageFileTemplate = f'{outFolder}/{prefix}_%03d.png' + imageFileTemplate = f'{outFolder}/{prefix}_t%03d.png' logFileName = f'{outFolder}/logs/{prefix}.log' with open(logFileName, 'w') as logFile: args = ['ffmpeg', '-y', '-r', framesPerSecond,