import numpy as np
from sasktran2 import Atmosphere
from sasktran2.atmosphere import (
InterpolatedDerivativeMapping,
NativeGridDerivative,
)
from sasktran2.optical.base import OpticalProperty
from sasktran2.util.interpolation import linear_interpolating_matrix
from .base import Constituent
[docs]
class VMRAltitudeAbsorber(Constituent):
def __init__(
self,
optical_property: OpticalProperty,
altitudes_m: np.array,
vmr: np.array,
out_of_bounds_mode: str = "zero",
) -> None:
"""
An atmospheric constituent that is specified through volume mixing ratio (VMR) on an altitude grid.
The altitude grid need not match the global atmospheric grid, in the case they are different the VMRs
will be linearly interpolated to the atmospheric grid locations.
The constituent has an associated OpticalProperty that defines the absorption cross section as a function
of wavelength.
Parameters
----------
optical_property : OpticalProperty
An object that provides absorption cross sections as a function of wavelength
altitudes_m : np.array
Altitudes in [m] that the VMR is specified on
vmr : np.array
Volume mixing ratio
out_of_bounds_mode : str, optional
One of ['zero', 'extend'], 'zero' will set the VMR to be 0 on any altitudes that are
out of bounds. 'extend' will extend the last or first VMR value. By default 'zero'
"""
super().__init__()
self._out_of_bounds_mode = out_of_bounds_mode
self._altitudes_m = altitudes_m
self._vmr = vmr
self._optical_property = optical_property
@property
def vmr(self):
return self._vmr
@vmr.setter
def vmr(self, vmr: np.array):
self._vmr = vmr
def add_to_atmosphere(self, atmo: Atmosphere):
self._optical_quants = self._optical_property.atmosphere_quantities(atmo)
if atmo.pressure_pa is None or atmo.temperature_k is None:
msg = "Both pressure_pa and temperature_k have to be specified in the atmosphere to use VMRAltitudeAbsorber"
raise ValueError(msg)
number_density = atmo.state_equation.dry_air_numberdensity["N"]
interp_matrix = linear_interpolating_matrix(
self._altitudes_m,
atmo.model_geometry.altitudes(),
self._out_of_bounds_mode.lower(),
)
interp_vmr = interp_matrix @ self._vmr
atmo.storage.total_extinction += (
self._optical_quants.extinction
* (number_density * interp_vmr)[:, np.newaxis]
)
def register_derivative(self, atmo: Atmosphere, name: str):
dry_air_number_density = atmo.state_equation.dry_air_numberdensity
number_density = dry_air_number_density["N"]
interp_matrix = linear_interpolating_matrix(
self._altitudes_m,
atmo.model_geometry.altitudes(),
self._out_of_bounds_mode.lower(),
)
derivs = {}
derivs["vmr"] = InterpolatedDerivativeMapping(
NativeGridDerivative(
d_extinction=self._optical_quants.extinction
* number_density[:, np.newaxis],
d_ssa=self._optical_quants.extinction
* (self._optical_quants.ssa - atmo.storage.ssa)
/ atmo.storage.total_extinction
* number_density[:, np.newaxis],
),
interpolating_matrix=interp_matrix,
interp_dim="altitude",
result_dim=f"{name}_altitude",
)
interp_vmr = interp_matrix @ self._vmr
deriv_names = []
d_vals = []
if atmo.calculate_pressure_derivative:
deriv_names.append("pressure_pa")
d_vals.append(dry_air_number_density["dN_dP"])
if atmo.calculate_temperature_derivative:
deriv_names.append("temperature_k")
d_vals.append(dry_air_number_density["dN_dT"])
if atmo.calculate_specific_humidity_derivative:
deriv_names.append("specific_humidity")
d_vals.append(dry_air_number_density["dN_dsh"])
# Contributions from the change in number density due to a constant
# VMR and changing pressure/temperature
for deriv_name, vert_factor in zip(deriv_names, d_vals, strict=False):
derivs[deriv_name] = InterpolatedDerivativeMapping(
NativeGridDerivative(
d_extinction=self._optical_quants.extinction,
d_ssa=self._optical_quants.extinction
* (self._optical_quants.ssa - atmo.storage.ssa)
/ atmo.storage.total_extinction,
),
interpolating_matrix=np.eye(len(number_density))
* (vert_factor * interp_vmr)[np.newaxis, :],
interp_dim="altitude",
result_dim="altitude",
summable=True,
)
if len(deriv_names) > 0:
optical_derivs = self._optical_property.optical_derivatives(atmo=atmo)
for key, val in optical_derivs.items():
# We only get d_extinction from the optical property, have to set d_ssa accordingly
val.d_ssa = (
val.d_extinction
* (self._optical_quants.ssa - atmo.storage.ssa)
/ atmo.storage.total_extinction
)
# Assume that all optical derivs are summable, I can't think of any that aren't right now.
# Give it a unique key to help with the summing
derivs[f"{key}_xs"] = InterpolatedDerivativeMapping(
val,
interpolating_matrix=np.eye(len(number_density))
* (number_density * interp_vmr)[np.newaxis, :],
interp_dim="altitude",
result_dim="altitude",
summable=True,
name=key,
)
return derivs