Source code for opendrift.models.oceandrift

# This file is part of OpenDrift.
#
# OpenDrift is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, version 2
#
# OpenDrift is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with OpenDrift.  If not, see <https://www.gnu.org/licenses/>.
#
# Copyright 2015, Knut-Frode Dagestad, MET Norway

from datetime import datetime, timedelta
import numpy as np
from scipy.interpolate import interp1d
import logging; logger = logging.getLogger(__name__)
from opendrift.models.basemodel import OpenDriftSimulation
from opendrift.elements import LagrangianArray
from opendrift.models.physics_methods import verticaldiffusivity_Large1994, verticaldiffusivity_Sundby1983, gls_tke, skillscore_liu_weissberg
from opendrift.config import CONFIG_LEVEL_ESSENTIAL, CONFIG_LEVEL_BASIC, CONFIG_LEVEL_ADVANCED

# Defining the oil element properties
[docs] class Lagrangian3DArray(LagrangianArray): """Extending LagrangianArray for elements moving in 3 dimensions The Particle may be buoyant and/or subject to vertical mixing buoyant bahaviour is described by terminal velocity """ variables = LagrangianArray.add_variables([ ('wind_drift_factor', {'dtype': np.float32, 'units': '1', 'description': 'Elements at surface are moved with this ' 'fraction of the vind vector, in addition to currents ' 'and Stokes drift', 'default': 0.02}), ('current_drift_factor', {'dtype': np.float32, 'units': '1', 'description': 'Elements are moved with this fraction of the ' 'current vector, in addition to currents ' 'and Stokes drift', 'default': 1}), ('terminal_velocity', {'dtype': np.float32, 'units': 'm/s', 'description': 'Terminal rise/sinking velocity (buoyancy) ' 'in the ocean column', 'default': 0.})])
[docs] class OceanDrift(OpenDriftSimulation): """Open source buoyant particle trajectory model based on OpenDrift. Developed at MET Norway Generic module for particles that move in 3 dimensions and may be to vertical turbulent mixing with the possibility for positive or negative buoyancy Particles could be e.g. oil droplets, plankton, nutrients or sediments, Model may be subclassed for more specific behaviour. """ ElementType = Lagrangian3DArray required_variables = { 'x_sea_water_velocity': {'fallback': 0}, 'y_sea_water_velocity': {'fallback': 0}, 'sea_surface_height': {'fallback': 0}, 'x_wind': {'fallback': 0}, 'y_wind': {'fallback': 0}, 'upward_sea_water_velocity': {'fallback': 0}, 'ocean_vertical_diffusivity': {'fallback': 0, 'profiles': True}, 'sea_surface_wave_significant_height': {'fallback': 0}, 'sea_surface_wave_stokes_drift_x_velocity': {'fallback': 0}, 'sea_surface_wave_stokes_drift_y_velocity': {'fallback': 0}, 'sea_surface_wave_period_at_variance_spectral_density_maximum': {'fallback': 0}, 'sea_surface_wave_mean_period_from_variance_spectral_density_second_frequency_moment': {'fallback': 0}, 'sea_surface_swell_wave_to_direction': {'fallback': 0, 'important': False}, 'sea_surface_swell_wave_peak_period_from_variance_spectral_density': {'fallback': 0, 'important': False}, 'sea_surface_swell_wave_significant_height': {'fallback': 0, 'important': False}, 'sea_surface_wind_wave_to_direction': {'fallback': 0, 'important': False}, 'sea_surface_wind_wave_mean_period': {'fallback': 0, 'important': False}, 'sea_surface_wind_wave_significant_height': {'fallback': 0, 'important': False}, 'surface_downward_x_stress': {'fallback': 0}, 'surface_downward_y_stress': {'fallback': 0}, 'turbulent_kinetic_energy': {'fallback': 0}, 'turbulent_generic_length_scale': {'fallback': 0}, 'ocean_mixed_layer_thickness': {'fallback': 50}, 'sea_floor_depth_below_sea_level': {'fallback': 10000}, 'land_binary_mask': {'fallback': None}, } def __init__(self, *args, **kwargs): if 'machine_learning_dict' in kwargs: logger.debug('Machine learning correction supplied.') mld = kwargs['machine_learning_dict'] del kwargs['machine_learning_dict'] import pickle from tensorflow import keras from tools.normalizations import generate_one_normalized_predictor, generate_one_denormalized_prediction self.generate_one_normalized_predictor = generate_one_normalized_predictor self.generate_one_denormalized_prediction = generate_one_denormalized_prediction self.trained_model = keras.models.load_model(mld['trained_model']) self.dict_normalization_params = pickle.load(open(mld['dict_normalization_params'], 'rb')) self.ml_predictors = [self.dict_normalization_params['predictors'][i]['content'] for i in self.dict_normalization_params['predictors']] else: logger.debug('No machine learning correction available.') if hasattr(self, 'required_profiles_z_range'): raise ValueError('self.required_profiles_z_range is obsolete, and replaced by config setting drift:profile_depth. Please update your model.') # Calling general constructor of parent class super(OceanDrift, self).__init__(*args, **kwargs) self._add_config({ 'drift:vertical_advection': {'type': 'bool', 'default': True, 'description': 'Advect elements with vertical component of ocean current.', 'level': CONFIG_LEVEL_BASIC}, 'drift:vertical_mixing': {'type': 'bool', 'default': False, 'level': CONFIG_LEVEL_BASIC, 'description': 'Activate vertical mixing scheme with inner loop'}, 'vertical_mixing:timestep': {'type': 'float', 'min': 0.1, 'max': 3600, 'default': 60, 'level': CONFIG_LEVEL_ADVANCED, 'units': 'seconds', 'description': 'Time step used for inner loop of vertical mixing.'}, 'vertical_mixing:diffusivitymodel': {'type': 'enum', 'default': 'environment', 'enum': ['environment', 'stepfunction', 'windspeed_Sundby1983', 'windspeed_Large1994', 'gls_tke','constant'], 'level': CONFIG_LEVEL_ADVANCED, 'units': 'seconds', 'description': 'Algorithm/source used for profile of vertical diffusivity. Environment means that diffusivity is aquired from readers or environment constants/fallback.'}, 'vertical_mixing:background_diffusivity': {'type': 'float', 'min': 0, 'max': 1, 'default': 1.2e-5, 'level': CONFIG_LEVEL_ADVANCED, 'units': 'm2s-1', 'description': 'Background diffusivity used below mixed layer for wind parameterisations.'}, 'vertical_mixing:TSprofiles': {'type': 'bool', 'default': False, 'level': CONFIG_LEVEL_ADVANCED, 'description': 'Update T and S profiles within inner loop of vertical mixing. This takes more time, but may be slightly more accurate.'}, 'drift:wind_drift_depth': {'type': 'float', 'default': 0.1, 'min': 0, 'max': 10, 'units': 'meters', 'description': 'The direct wind drift (windage) is linearly decreasing from the surface value (wind_drift_factor) until 0 at this depth.', 'level': CONFIG_LEVEL_ADVANCED}, 'drift:stokes_drift': {'type': 'bool', 'default': True, 'description': 'Advection elements with Stokes drift (wave orbital motion).', 'level': CONFIG_LEVEL_ADVANCED}, 'drift:stokes_drift_profile': {'type': 'enum', 'default': 'Phillips', 'enum': ['monochromatic', 'exponential', 'Phillips', 'windsea_swell'], 'description': 'Algorithm to calculate Stokes drift at depth from surface value', 'level': CONFIG_LEVEL_ADVANCED}, 'drift:use_tabularised_stokes_drift': {'type': 'bool', 'default': False, 'description': 'If True, Stokes drift is estimated from wind based on look-up-tables for given fetch (drift:tabularised_stokes_drift_fetch).', 'level': CONFIG_LEVEL_ADVANCED}, 'drift:tabularised_stokes_drift_fetch': {'type': 'enum', 'enum': ['5000', '25000', '50000'], 'default': '25000', 'level': CONFIG_LEVEL_ADVANCED, 'description': 'The fetch length when using tabularised Stokes drift.'}, 'general:seafloor_action': {'type': 'enum', 'default': 'lift_to_seafloor', 'enum': ['none', 'lift_to_seafloor', 'deactivate', 'previous'], 'description': '"deactivate": elements are deactivated; "lift_to_seafloor": elements are lifted to seafloor level; "previous": elements are moved back to previous position; "none"; seafloor is ignored.', 'level': CONFIG_LEVEL_ADVANCED}, 'drift:truncate_ocean_model_below_m': {'type': 'float', 'default': None, 'min': 0, 'max': 10000, 'units': 'm', 'description': 'Ocean model data are only read down to at most this depth, and extrapolated below. May be specified to read less data to improve performance.', 'level': CONFIG_LEVEL_ADVANCED}, 'seed:z': {'type': 'float', 'default': 0, 'min': -10000, 'max': 0, 'units': 'm', 'description': 'Depth below sea level where elements are released. This depth is neglected if seafloor seeding is set selected.', 'level': CONFIG_LEVEL_ESSENTIAL}, 'seed:seafloor': {'type': 'bool', 'default': False, 'description': 'Elements are seeded at seafloor, and seeding depth (z) is neglected.', 'level': CONFIG_LEVEL_ESSENTIAL}, }) self._set_config_default('drift:max_speed', 2)
[docs] def update(self): """Update positions and properties of elements.""" # Simply move particles with ambient current self.advect_ocean_current() # Advect particles due to surface wind drag, # according to element property wind_drift_factor self.advect_wind() # Stokes drift self.stokes_drift() # Turbulent Mixing if self.get_config('drift:vertical_mixing') is True: self.update_terminal_velocity() self.vertical_mixing() else: # Buoyancy self.vertical_buoyancy() # Vertical advection self.vertical_advection() # Optional machine learning correction self.machine_learning_correction()
[docs] def simulate_trajectories(self, outfile, trajectories, number=1, wind_drift_factors=None, current_drift_factors=None, time_step=None, time_step_output=None, simulation_duration=None, simulation_interval=None): import pandas as pd time_step_output = pd.Timedelta(time_step_output) simulation_interval = pd.Timedelta(simulation_interval) # Interpolate trajectories to output time step #trajectories = trajectories.traj.gridtime(time_step_output) # Find all seed combinations: position, time, wdf, cdf # Loop også over trajektorier -> origin_marker step = int(simulation_interval / time_step_output) tind = np.arange(0, trajectories.sizes['time'], step) start_lons = trajectories.isel(time=tind).isel(trajectory=0).lon.values start_lats = trajectories.isel(time=tind).isel(trajectory=0).lat.values start_times = trajectories.time[tind] self.set_config('drift:max_age_seconds', simulation_duration.total_seconds()) kwargs = {'wind_drift_factor': wind_drift_factors[0], 'current_drift_factor': current_drift_factors[0]} for (lo,la,ti) in zip(start_lons, start_lats, start_times): if np.isnan(lo): continue ti = pd.Timestamp(ti.values).to_pydatetime() self.seed_elements(lon=lo, lat=la, time=ti, number=number, **kwargs) print(self) self.run(outfile=outfile, end_time=pd.Timestamp(start_times[-1].values).to_pydatetime()+simulation_duration)
# Simulate and save to file
[docs] def wind_drift_factor_from_trajectory_lw(self, drifters, wind_drift_factors, simulation_length, simulation_interval): """Perform simulations and use skillscore to optimize wind_drift_factor drifters: list of dictionaries with numpy arrays of 'lon' and 'lat' and list of datetimes wind_drift_factors: the wind_drift_factors to use for simulations/optimalizations """ output = [] # List with one dictionary per trajectory for d in drifters: time = np.array(d['time']) ti = time[1::] - time[0:-1] if ti.min() == ti.max(): trajectory_interval = ti[0] else: raise ValueError('Trajectory interval is not constant') if 'tiv' not in locals(): tiv = trajectory_interval index_interval = simulation_interval.total_seconds()/trajectory_interval.total_seconds() if not index_interval.is_integer(): raise ValueError('Simulation interval must be a multiple of trajectory time step') else: if trajectory_interval != tiv: raise ValueError('Trajectories do not have the same time step') last_index = (time[-1]-time[0]-simulation_length).total_seconds()/trajectory_interval.total_seconds() seed_indices = np.arange(0, last_index, index_interval).astype(int) do = {'lon': d['lon'][seed_indices], 'lat': d['lat'][seed_indices], 'time': np.array(time[seed_indices]), 'trajectory_start_index': seed_indices} output.append(do) if not 'last_seed_time' in locals(): last_seed_time = do['time'][-1] else: last_seed_time = np.maximum(do['time'][-1], last_seed_time) # Seed for all starting positions in this trajectory for lo,la,ti in zip(do['lon'], do['lat'], do['time']): ow = np.ones(len(wind_drift_factors)) self.seed_elements(lon=lo*ow, lat=la*ow, time=ti, wind_drift_factor=wind_drift_factors) self.set_config('drift:max_age_seconds', simulation_length.total_seconds()+1) self.run(end_time=last_seed_time+simulation_length) index_of_first, index_of_last = \ self.index_of_activation_and_deactivation() times_model = np.array(self.get_time_array()[0]) #import matplotlib.pyplot as plt i = 0 # element number in simulation for d, o in zip(drifters, output): o['segments'] = {} o['wind_drift_factor'] = np.ones(len(o['trajectory_start_index'])) o['skillscore'] = np.ones(len(o['trajectory_start_index'])) for segnum, tsi in enumerate(o['trajectory_start_index']): o['segments'][segnum] = {'skillscore': ow} lon_obs = d['lon'][tsi:tsi+int(index_interval)] lat_obs = d['lat'][tsi:tsi+int(index_interval)] #plt.plot(lon_obs, lat_obs, 'r') for wnum, wdf in enumerate(wind_drift_factors): lon_model = self.history['lon'][i, index_of_first[i]:index_of_last[i]] lat_model = self.history['lat'][i, index_of_first[i]:index_of_last[i]] i = i + 1 ss = skillscore_liu_weissberg(lon_obs, lat_obs, lon_model, lat_model) o['segments'][segnum]['skillscore'][wnum] = ss #plt.plot(lon_model, lat_model, label='Skillscore: %s' % ss) print(o['segments'][segnum]['skillscore'], 'SS') o['wind_drift_factor'][segnum] = \ wind_drift_factors[np.argmax(o['segments'][segnum]['skillscore'])] o['skillscore'][segnum] = o['segments'][segnum]['skillscore'].max() #plt.legend() #plt.show() #stop #import xarray as xr #da = xr.DataArray( # data=skillscore, # dims=["wind_drift_factor", "time"], # coords=dict( # wind_drift_factor=wind_drift_factors, # time=time # ), # attrs=dict( # description="Liu Weissberg skillscore", # ), #) return output
[docs] def machine_learning_correction(self): if not hasattr(self, 'trained_model'): return # No machine learning correction available # ML shall only be applied every 1 hour if not hasattr(self, 'ml_timesteps'): self.ml_timesteps = 3600/self.time_step.total_seconds() if self.steps_calculation % self.ml_timesteps != 0: logger.warning('ML not applied at time step %s' % self.steps_calculation) return # Only apply every 1 hour logger.warning('Machine learning') x_vel = np.ones(self.num_elements_active())*np.NaN y_vel = np.ones(self.num_elements_active())*np.NaN st = datetime.now() list_predictors_normalized = [] list_predictors_denormalized = [] # Prepare predictor dicts for n in range(self.num_elements_active()): dict_input = {s: getattr(self.environment, s)[n] for s in self.ml_predictors} predictor_normalized = self.generate_one_normalized_predictor( dict_input, self.dict_normalization_params) list_predictors_normalized.append(predictor_normalized) # Normalize all_predictors_normalized = np.squeeze(np.array(list_predictors_normalized)) # perform a prediction from normalized predictor to normalized label logger.warning('Calculating ML correction...') predicted_normalized_residual_correction = self.trained_model.predict(all_predictors_normalized) # get the "native units" residual correction logger.warning('Denormalization...') for n in range(self.num_elements_active()): native_units_correction = self.generate_one_denormalized_prediction( predicted_normalized_residual_correction[n, :], self.dict_normalization_params) x_vel[n] = native_units_correction['residual_displacement_x']*1000/3600 y_vel[n] = native_units_correction['residual_displacement_y']*1000/3600 # Apply correction logger.warning('Applying ML correction: %s to %s m/s eastwards, %s to %s m/s northwards' % (x_vel.min(), x_vel.max(), y_vel.min(), y_vel.max())) #logger.warning('%s particles, %s' % (self.num_elements_active(), datetime.now()-st)) self.update_positions(x_vel, y_vel)
[docs] def disable_vertical_motion(self): """Deactivate any vertical processes/advection""" conf = { 'drift:vertical_advection': False, 'drift:vertical_mixing': False} for co, va in conf.items(): logger.info('Setting config: %s -> %s' % (co, va)) self.set_config(co, va)
[docs] def update_terminal_velocity(self, Tprofiles=None, Sprofiles=None, z_index=None): """Calculate terminal velocity due to bouyancy from own properties and environmental variables. Sub-modules should overload this method for particle-specific behaviour """ pass
[docs] def prepare_vertical_mixing(self): pass # To be implemented by subclasses as needed
[docs] def prepare_run(self): super(OceanDrift, self).prepare_run()
[docs] def vertical_advection(self): """Move particles vertically according to vertical ocean current Vertical advection by ocean currents is normally small compared to termical velocity """ if self.get_config('drift:vertical_advection') is False: logger.debug('Vertical advection deactivated') return in_ocean = np.where(self.elements.z<0)[0] if len(in_ocean) > 0: w = self.environment.upward_sea_water_velocity[in_ocean] self.elements.z[in_ocean] = np.minimum(0, self.elements.z[in_ocean] + self.elements.moving[in_ocean] * w * self.time_step.total_seconds()) else: logger.debug('No vertical advection for elements at surface')
[docs] def vertical_buoyancy(self): """Move particles vertically according to their buoyancy""" in_ocean = np.where(self.elements.z<0)[0] if len(in_ocean) > 0: self.elements.z[in_ocean] = np.minimum(0, self.elements.z[in_ocean] + self.elements.terminal_velocity[in_ocean] * self.time_step.total_seconds()) # check for minimum height/maximum depth for each particle accouting also for # the sea surface height Zmin = -1.*(self.environment.sea_floor_depth_below_sea_level + self.environment.sea_surface_height) # Let particles stick to bottom bottom = np.where(self.elements.z < Zmin) if len(bottom[0]) > 0: logger.debug('%s elements reached seafloor, interacting with bottom' % len(bottom[0])) self.interact_with_seafloor() self.bottom_interaction(Zmin)
[docs] def surface_stick(self): '''To be overloaded by subclasses, e.g. downward mixing of oil''' # keep particle just below the surface surface = np.where(self.elements.z >= 0) if len(surface[0]) > 0: self.elements.z[surface] = -0.01
[docs] def bottom_interaction(self, Zmin=None): '''To be overloaded by subclasses, e.g. radionuclides in sediments''' pass
[docs] def surface_wave_mixing(self, time_step_seconds): '''To be overloaded by subclasses, e.g. downward mixing of oil''' pass
[docs] def get_diffusivity_profile(self, model, depths): wind, depth = np.meshgrid(self.wind_speed(), depths) MLD = self.environment.ocean_mixed_layer_thickness background_diffusivity = self.get_config('vertical_mixing:background_diffusivity') if model == 'windspeed_Large1994': return verticaldiffusivity_Large1994(wind, depth, MLD, background_diffusivity) elif model == 'windspeed_Sundby1983': return verticaldiffusivity_Sundby1983(wind, depth, MLD, background_diffusivity) elif model == 'gls_tke': if not hasattr(self, 'gls_parameters'): logger.info('Searching readers for GLS parameters...') for reader_name, reader in self.readers.items(): if hasattr(reader, 'gls_parameters'): self.gls_parameters = reader.gls_parameters logger.info('Found gls-parameters in ' + reader_name) break # Success if not hasattr(self, 'gls_parameters'): logger.info('Did not find gls-parameters in any readers.') self.gls_parameters = None windstress = np.sqrt(self.environment.surface_downward_x_stress**2 + self.environment.surface_downward_y_stress**2) return gls_tke(windstress, depth, self.sea_water_density(), self.environment.turbulent_kinetic_energy, self.environment.turbulent_generic_length_scale, gls_parameters) else: raise ValueError('Unknown diffusivity model: ' + model)
[docs] def vertical_mixing(self, store_depths=False): """Mix particles vertically according to eddy diffusivity and buoyancy Buoyancy is expressed as terminal velocity, which is the steady-state vertical velocity due to positive or negative buoyant behaviour. It is usually a function of particle density, diameter, and shape. Vertical particle displacemend du to turbulent mixing is calculated using a random walk scheme" (Visser et al. 1996) """ if self.get_config('drift:vertical_mixing') is False: logger.debug('Turbulent mixing deactivated') return self.timer_start('main loop:updating elements:vertical mixing') dt_mix = self.get_config('vertical_mixing:timestep') # minimum height/maximum depth for each particle accouting also for # the sea surface height Zmin = -1.*(self.environment.sea_floor_depth_below_sea_level + self.environment.sea_surface_height) # Eventual model specific preparions self.prepare_vertical_mixing() # get profile of eddy diffusivity # get vertical eddy diffusivity from environment or specific model diffusivity_model = self.get_config('vertical_mixing:diffusivitymodel') diffusivity_fallback = self.get_config('environment:fallback:ocean_vertical_diffusivity') # Mixing levels to be used if analytical expression: mixing_z_analytical = -np.arange(0, self.environment.ocean_mixed_layer_thickness.max() + 2) if diffusivity_model == 'environment': if 'ocean_vertical_diffusivity' in self.environment_profiles and not ( self.environment_profiles['ocean_vertical_diffusivity'].min() == diffusivity_fallback and self.environment_profiles['ocean_vertical_diffusivity'].max() == diffusivity_fallback): Kprofiles = self.environment_profiles[ 'ocean_vertical_diffusivity'] mixing_z = self.environment_profiles['z'].copy() logger.debug('Using diffusivity from ocean model') else: logger.debug('Using diffusivity from Large1994 since model diffusivities not available') mixing_z = mixing_z_analytical Kprofiles = self.get_diffusivity_profile('windspeed_Large1994', np.abs(mixing_z)) elif diffusivity_model == 'constant': logger.debug('Using constant diffusivity specified by fallback_values[''ocean_vertical_diffusivity''] = %s m2.s-1' % (diffusivity_fallback)) mixing_z = self.environment_profiles['z'].copy() Kprofiles = diffusivity_fallback*np.ones( self.environment_profiles['ocean_vertical_diffusivity'].shape) # keep constant value for ocean_vertical_diffusivity else: logger.debug('Using functional expression for diffusivity') # Note: although analytical functions, z is discretised mixing_z = mixing_z_analytical Kprofiles = self.get_diffusivity_profile(diffusivity_model, np.abs(mixing_z)) logger.debug('Diffusivities are in range %s to %s' % (Kprofiles.min(), Kprofiles.max())) # get profiles of salinity and temperature # (to save interpolation time in the inner loop) if (self.get_config('vertical_mixing:TSprofiles') is True and 'sea_water_salinity' in self.required_variables): Sprofiles = self.environment_profiles['sea_water_salinity'] Tprofiles = \ self.environment_profiles['sea_water_temperature'] if (self.get_config('environment:fallback:sea_water_salinity') is not None and Sprofiles.min() == Sprofiles.max()): logger.debug('Salinity and temperature are fallback' 'values, skipping TSprofile') Sprofiles = None Tprofiles = None else: logger.debug('Using TSprofiles for vertical mixing') else: logger.debug('TSprofiles deactivated for vertical mixing') Sprofiles = None Tprofiles = None # prepare vertical interpolation coordinates z_i = range(mixing_z.shape[0]) if len(z_i) == 1: z_index = 0 else: z_index = interp1d(-mixing_z, z_i, bounds_error=False, fill_value=(0,len(z_i)-1)) # Extrapolation # Internal loop for fast time step of vertical mixing model. # Random walk needs faster time step than horizontal advection. logger.debug('Vertical mixing module:' + self.get_config('vertical_mixing:diffusivitymodel')) ntimes_mix = np.abs(int(self.time_step.total_seconds()/dt_mix)) logger.debug('Turbulent diffusion with random walk ' 'scheme using ' + str(ntimes_mix) + ' fast time steps of dt=' + str(dt_mix) + 's') if store_depths is not False: depths = np.zeros((ntimes_mix, self.num_elements_active())) depths[0, :] = self.elements.z # Calculating dK/dz for all profiles before the loop gradK = -np.gradient(Kprofiles, mixing_z, axis=0) gradK[np.abs(gradK)<1e-10] = 0 for i in range(0, ntimes_mix): #remember which particles belong to the exact surface surface = self.elements.z == 0 # Update the terminal velocity of particles self.update_terminal_velocity(Tprofiles=Tprofiles, Sprofiles=Sprofiles, z_index=z_index) w = self.elements.terminal_velocity # Diffusivity and its gradient at z zi = np.round(z_index(-self.elements.z)).astype(np.uint16) Kz = Kprofiles[zi, range(Kprofiles.shape[1])] dKdz = gradK[zi, range(Kprofiles.shape[1])] # Visser et al. 1997 random walk mixing # requires an inner loop time step dt such that # dt << (d2K/dz2)^-1, e.g. typically dt << 15min # # NB: In the last term Kz is evaluated in zi, while # it should be evaluated in (self.elements.z - dKdz*dt_mix) # This is not expected have large impact on the result R = 2*np.random.random(self.num_elements_active()) - 1 r = 1.0/3 # New position = old position - up_K_flux + random walk self.elements.z = self.elements.z - self.elements.moving*( dKdz*dt_mix - R*np.sqrt((Kz*dt_mix*2/r))) # Reflect from surface reflect = np.where(self.elements.z >= 0) if len(reflect[0]) > 0: self.elements.z[reflect] = -self.elements.z[reflect] # Reflect elements going below seafloor # Zmin accounts for sea_surface_height bottom = np.where(np.logical_and(self.elements.z < Zmin, self.elements.moving == 1)) if len(bottom[0]) > 0: logger.debug('%s elements penetrated seafloor, lifting up' % len(bottom[0])) self.elements.z[bottom] = 2*Zmin[bottom] - self.elements.z[bottom] # Advect due to buoyancy self.elements.z = self.elements.z + w*dt_mix*self.elements.moving # Put the particles that belonged to the surface slick # (if present) back to the surface self.elements.z[surface] = 0. # Formation of slick and wave mixing for surfaced particles # if implemented for this class self.surface_stick() self.surface_wave_mixing(dt_mix) # Let particles stick to bottom bottom = np.where(self.elements.z < Zmin) if len(bottom[0]) > 0: logger.debug('%s elements reached seafloor, interacting with bottom' % len(bottom[0])) self.interact_with_seafloor() self.bottom_interaction(Zmin) if store_depths is not False: depths[i, :] = self.elements.z self.timer_end('main loop:updating elements:vertical mixing') if store_depths is not False: return depths else: return None
[docs] def animate_vertical_distribution(self, depths=None, maxdepth=None, bins=50, filename=None, subsamplingstep=1): """Function to animate vertical distribution of particles bins: number of bins in the histogram maxdepth: maximum depth subsamplingstep: speed-up the generation of the animation reducing the number of output frames fasterwriter: speed-up the writing to outpute file """ import matplotlib.pyplot as plt import matplotlib.animation as animation start_time = datetime.now() #from timeit import default_timer as timer #start=timer() fig, (axk, axn) = plt.subplots(1, 2, gridspec_kw={'width_ratios': [1, 3]}) if depths is not None: # Debug mode, output from one cycle has been provided z = depths time_step = self.get_config('vertical_mixing:timestep') times = [self.time + i*timedelta(seconds=time_step) for i in range(z.shape[0])] else: z = self.get_property('z')[0] z = np.ma.filled(z, np.nan) K = self.get_property('ocean_vertical_diffusivity')[0] time_step = self.time_step.total_seconds() times = self.get_time_array()[0] if maxdepth is None: maxdepth = np.nanmin(z) if maxdepth > 0: maxdepth = -maxdepth # negative z if depths is not None: axk.plot(np.nanmean(self.environment_profiles['ocean_vertical_diffusivity'], 1), self.environment_profiles['z']) xmax = self.environment_profiles['ocean_vertical_diffusivity'].max() else: axk.plot(K, z, 'k.') xmax = np.nanmax(K) axk.set_ylim([maxdepth, 0]) axk.set_xlim([0, xmax*1.1]) axk.set_ylabel('Depth [m]') axk.set_xlabel('Vertical diffusivity [$m^2/s$]') subsamplingstep=max(1,subsamplingstep) times=times[0:-1:subsamplingstep] z=z[0:-1:subsamplingstep,:] hist_series = np.zeros((bins, len(times))) bin_series = np.zeros((bins+1, len(times))) for i in range(len(times)): hist_series[:,i], bin_series[:,i] = np.histogram(z[i,:][np.isfinite(z[i,:])], bins=bins, range=(maxdepth,z[np.isfinite(z)].max())) maxnum = hist_series.max() axn.clear() bc=axn.barh(bin_series[0:-1,i], hist_series[:,i], height=-maxdepth/bins, align='edge') axn.set_ylim([maxdepth, 0]) axn.set_xlim([0, maxnum]) title=axn.set_title('%s UTC' % times[i]) axn.set_xlabel('Number of particles') def update_histogram(i): for rect, y in zip(bc, hist_series[:,i]): rect.set_width(y) title.set_text('%s UTC' % times[i]) self.__save_or_plot_animation__(plt.gcf(), update_histogram, filename, frames = len(times), fps = 10, interval=50, blit=False) logger.info('Time to make animation: %s' % (datetime.now() - start_time))
[docs] def plot_vertical_distribution(self, maxdepth=None, bins=None, maxnum=None): """Function to plot vertical distribution of particles maxdepth: maximum depth considered for the profile bins: number of bins between surface and maxdepth maxnum: range of bars in histogram is [0,maxnum] """ import matplotlib.pyplot as plt from matplotlib.widgets import Slider fig = plt.figure() mainplot = fig.add_axes([.15, .3, .8, .5]) sliderax = fig.add_axes([.15, .08, .75, .05]) tslider = Slider(sliderax, 'Timestep', 0, self.steps_output-1, valinit=self.steps_output-1, valfmt='%0.0f') try: dz = self.get_config('vertical_mixing:verticalresolution') except: dz = 1. maxrange = -100 # overwrite default values if input arguments are provided if maxdepth is not None: maxrange = maxdepth if maxrange > 0: maxrange = -maxrange # negative z if bins is not None: dz = -maxrange/bins # using get_property instead of history to exclude elements thare are not yet seeded z = self.get_property('z')[0] z = np.ma.filled(z, np.nan) if maxnum is None: # Precalculatig histograms to find maxnum hist_series = np.zeros((int(-maxrange/dz), self.steps_output-1)) bin_series = np.zeros((int(-maxrange/dz)+1, self.steps_output-1)) for i in range(self.steps_output-1): hist_series[:,i], bin_series[:,i] = np.histogram(z[i,:][np.isfinite(z[i,:])], bins=int(-maxrange/dz), range=[maxrange, 0]) maxnum = hist_series.max() def update(val): tindex = int(tslider.val) mainplot.cla() mainplot.grid() mainplot.hist(z[tindex, :], bins=int(-maxrange/dz), range=[maxrange, 0], orientation='horizontal') mainplot.set_ylim([maxrange, 0]) mainplot.set_xlim([0, maxnum]) mainplot.set_xlabel('number of particles') mainplot.set_ylabel('depth [m]') x_wind = self.history['x_wind'].T[tindex, :] y_wind = self.history['y_wind'].T[tindex, :] windspeed = np.mean(np.sqrt(x_wind**2 + y_wind**2)) mainplot.set_title(str(self.get_time_array()[0][tindex]) + #' Percent at surface: %.1f %' % percent_at_surface) ' Mean windspeed: %.1f m/s' % windspeed) fig.canvas.draw_idle() update(0) # Plot initial distribution tslider.on_changed(update) plt.show() # returning objects prevents unresponsiveness when moving the Slider # https://github.com/matplotlib/matplotlib/issues/3105#issuecomment-44855888 return fig,mainplot,sliderax,tslider
[docs] def plotter_vertical_distribution_time(self, ax=None, mask=None, dz=1., maxrange=-100, bins=None, step=1): """Function to plot vertical distribution of particles. Use mask to plot any selection of particles. """ from pylab import axes, draw from matplotlib import dates, pyplot if ax is None: fig = pyplot.figure() ax = fig.gca() show = True else: show = False if mask is None: # create a mask that is True for all particles mask = self.history['z'].T[0] == self.history['z'].T[0] if bins is None: bins=int(-maxrange/dz) ax.hist(self.history['z'].T[step,mask], bins=bins, range=[maxrange, 0], orientation='horizontal') ax.set_ylim([maxrange, 0]) ax.grid() #ax.set_xlim([0, mask.sum()*.15]) ax.set_xlabel('Number of particles') ax.set_ylabel('Depth [m]') x_wind = self.history['x_wind'].T[step, :] y_wind = self.history['x_wind'].T[step, :] windspeed = np.mean(np.sqrt(x_wind**2 + y_wind**2)) ax.set_title(str(self.get_time_array()[0][step]) + ' Mean windspeed: %.1f m/s' % windspeed) if show is True: pyplot.show()