Temperate Ecohydrology#
Building on the previous demos around integrated hydrology, the first “typical” research application of ATS is in temperate ecohydrology. Here, by temperate we really mean that frozen soil is not a consideration, and by ecohydrology we mean that one wishes to represent evapotranspiration as a significant sink.
In this demo, we first describe how to bring in the needed input data. We describe the spinup process, and desribe how, once data is downloaded, it can be put into files in the right formats to allow ATS to use that forcing data.
Then we show a few different options for how to represent evapotranspiration, including:
prescribed P - ET which assumes that ET is known from data or another model
prescribed transpiration where we assume that transpiration is known, and it is distributed to a rooting zone
Priestley-Taylor ET: a simple model for total ET
Priestley-Taylor ET decomposed into four channels: a next version of P-T is to split ET into four compoments – canopy-water evaporation (of e.g. intercepted rainfall), bare ground evaporation, canopy transpiration, and snow sublimation.
Author: Ethan Coon
%matplotlib inline
import sys,os
sys.path.append(os.path.join(os.environ['ATS_SRC_DIR'],'tools', 'utils'))
sys.path.append(os.path.join(os.environ['ATS_SRC_DIR'],'docs', 'documentation', 'source', 'ats_demos'))
import ats_xdmf
import colors
import numpy as np
from matplotlib import pyplot as plt
from matplotlib import colorbar
import pandas
import run_demos
import h5py
Spinup#
For applications of temperate ecohydrology, spinup is all about getting antecedent soil moisture conditions correct. There can be no expectation of getting reasonable hydrographs if the antecedent soil moisture field is wrong.
One way of beginning this process is to solve, to pseudo-steady state, the integrated hydrology problem with a constant source term that is given by the mean annual (or mean seasonal) precipitation minus evapotranspiration (P-ET) rate. This can be solved for long times, using an adaptive timestep that grows as the solution approaches steady state. This sets the mean background conditions for simulations with daily or other high-resolution inputs.
The first step is to provide an initial condition for that run which is hydrostatic with the water table at the surface. Starting from fully saturated and drawing down to the steady state is faster and more efficient than starting from dry and filling up to the steady state. To do this, we don’t assume that this is a hillslope with fixed slope, but imagine that it is set by topography and therefore cannot be provided analytically.
To do this, we use the hydrostatic initial condition option. Given that initial condition, we solve to steady state. In this solve, we often use a too-large Manning’s coefficient. This slows up the surface water, which is safe to do because we’re looking at such long time scales. Surface water is still much faster than subsurface water, and as long as we aren’t trying to get high frequency hydrographs correct, this makes for much faster simulations (with longer timestep sizes) with little loss in predictive power.
This run is for 20,000 days and is plotted below. The image is a bit hard to see without exaggerating in the vertical – a better way of seeing these runs is to open it in VisIt.
# run the demo
run_demos.run_demo_local('spinup')
# plot the saturation at steady-state
directory = "spinup.demo"
vis = ats_xdmf.VisFile(directory, time_unit='d')
vis.loadMeshPolygons()
fig, ax = plt.subplots(1,1,figsize=(12,8))
def plot(i, ax):
cax = colorbar.make_axes(ax, 'right')[0]
cax.set_ylabel("saturation [-]")
sat = vis.get("saturation_liquid", vis.cycles[i])
poly = vis.getMeshPolygons(cmap='jet_r', linewidth=0)
poly.set_array(sat)
poly.set_clim(0.4,1)
ax.add_collection(poly)
ax.axis('auto')
plot(0, ax)
# plot the runoff and, more clearly, the total water content in the subsurface to see if we are at steadystate
fig, ax = plt.subplots(1,2, figsize=(13,3))
flux = pandas.read_csv(os.path.join(directory, 'surface_outlet_flux.dat'), comment='#')
ax[0].semilogy(flux['time [s]'][:]/86400., flux['surface outlet flux'][:]/55000., 'b')
ax[0].set_xlabel('time [d]')
ax[0].set_ylabel('flux [m^3 / s]')
wc = pandas.read_csv(os.path.join(directory, 'water_content.dat'), comment='#')
ax[1].plot(wc['time [s]'][:]/86400., wc[list(wc.keys())[1]][:], 'b')
ax[1].set_xlabel('time [d]')
ax[1].set_ylabel('water content [mol]')
plt.show()
Prescribed P - ET#
Often precipitation and evapotranspiration are known from either a preexisting land model run, remotely sensed or point data (e.g. eddy flux towers and weather stations) or other approaches. Here we do a few examples to generate datasets for use in ATS and then drive simulations with this prescribed data.
Data processing#
Below is an example of how to generate, from raw data, the input files needed by ATS for P, E, and T for these examples. Included in this repository is output from a CLM 4.5 run for the Asu Basin outside Manaus, Brazil. This run provides daily data of a variety of the water balance fluxes, including precipitation, evaporation, and transpiration. We read this file and generate the two HDF5 forcing files for use in ATS in the prescribed P-ET examples.
#
# load the data
#
data = np.genfromtxt("data/flux_output_w_fluxtowerR.txt", names=True)
# note this read with column headers -- the file contains:
print(data.dtype)
# daily data, converted to [s]. Note there are some units options in ATS, but it is easiest to
# write data in SI units.
times = np.array(range(len(data)))
times = times * 86400.
# precip, evap, and trans are all in [mm/s], and so must be converted to [m/s] for use in ATS
precip = 1.e-3 * data['atmospheric_rainmms']
evap = 1.e-3 * (data['vegetation_evaporationmms'] + data['ground_evaporationmms'])
trans = 1.e-3 * data['vegetation_transpirationmms']
# plot the data
fig, axs = plt.subplots(2,1,figsize=(13,6))
def plot(ax):
fac = 1.e3 * 86400
ax.plot(times/86400., precip*fac, 'b', label='precip')
ax.plot(times/86000., evap*fac, 'r', label='evap')
ax.plot(times/86400., trans*fac, color='forestgreen', label='trans')
ax.legend(loc='upper right')
ax.set_xlabel('time [d]')
ax.set_ylabel('water flux [mm/day]')
plot(axs[0])
plot(axs[1])
axs[1].set_ylim(0, 6)
# write to disc
with h5py.File('data/prescribed_pet.h5','w') as fid:
fid.create_dataset('time [s]', data=times)
fid.create_dataset('precipitation [m s^-1]', data=precip)
fid.create_dataset('evaporation [m s^-1]', data=evap)
fid.create_dataset('transpiration [m s^-1]', data=trans)
fid.create_dataset('p_minus_et [m s^-1]', data=(precip - evap - trans))
fid.create_dataset('p_minus_e [m s^-1]', data=(precip - evap))
plt.show()
[('atmospheric_rainmms', '<f8'), ('surface_runoffmms', '<f8'), ('subsurface_runoffmms', '<f8'), ('infiltrationmms', '<f8'), ('interceptionmms', '<f8'), ('ground_evaporationmms', '<f8'), ('vegetation_evaporationmms', '<f8'), ('vegetation_transpirationmms', '<f8'), ('heat_flux_into_soilWm2', '<f8')]
Daily forced simulations#
Given the steady-state initial condition and the forcing data, we now run simulations forced by daily P-ET data. Here we consider two choices – first a run where P - ET is prescribed as a single entity as a function of time, and a second run where P - E is prescribed as a function of time while T is separated out and extracted from the subsurface rather than from the surface.
In the former case, the net water source is provided to the surface system (prescribed_surface_evaporation.xml
). This is often problematic. Specifically, in most places, T can be order P, and pulling that much water from the surface isn’t always possible – the top grid cell dries out and cannot get water to the surface fast enough. Therefore a limiter must be applied to the ET to ensure it does not dry out a grid cell too much. This is especially problematic in 3D integrated hydrology simulations, where often the ET is calculated via a column model using an “average” depth to water table, which is significantly less than the depth to water table at, for instance, the top of a hillslope in a 3D run.
In the latter case, P-E is provided as a source to the surface system, while T is then distributed to a subsurface sink using a root profile for Plant Functional Type (PFT)-based vegetation (prescribed_transpiration.xml
). This is much more effective because the PFT rooting-depth based transpiration allows T to pull from depth, and has a native limiter on it based on soil moisture (or is it mafic potential?). Note that the E may still need to be limited, but typically, as E < P,T, this is less frequently a problem and typically full evaporation is taken and the limiter can be quite less of an issue.
In these runs we did need a limiter for the surface ET case, but not for the distributed T case.
Note the limiters used here are quite naive – better limiters should be implemented before this approach is used for real science.
Note that the hydrographs in the distributed case are drier, because ET was not limited and full transpiration was taken.
# run the demo
run_demos.run_demo_local(['prescribed_surface_evaporation', 'prescribed_transpiration'])
directory_p_et = 'prescribed_surface_evaporation.demo'
directory_p_e_t = 'prescribed_transpiration.demo'
# plot the hydrographs
fig, ax = plt.subplots(1,1)
def plot(ax, directory, name, color):
flux = pandas.read_csv(os.path.join(directory, 'water_balance.dat'), comment='#')
ax.plot(flux['time [d]'][:], flux['runoff generation [mol d^-1]'][:]/55000., color=color, label=name)
plot(ax, directory_p_et, 'surface', 'b')
plot(ax, directory_p_e_t, 'distributed', 'r')
ax.legend()
ax.set_xlabel('time [days]')
ax.set_ylabel('flow rate [m^3 / s]')
plt.show()
Empirical evapotranspiration#
Many empirical models of evapotranspiration have been defined. One of the most commonly used is the Priestley-Taylor model, which is itself a variant of the Penman-Monteith equation. Our implementation follows that of USGS’s PRMS model.
Under the first example, we impose incoming shortwave radiation as the net radiation balance. The ground surface temperature, needed to compute the ground conduction term, is given by yesterday’s air temperature. Snow is included as a simple bucket model, with sources due to precipitation and sinks due to an empirical melting model (also based on PRMS) is given by a melting degree-day approach. The Priestley-Taylor equations provide potential transpiration – this flux is then allocated to grid cells in the shallow subsurface based on a model from CLM version 4.5 that includes rooting distribution parameters and a limiter based on mafic potential of full turgor and that of the wilting point.
def plot(testname, axs=None, label=None):
dirname = testname+'.demo'
obs = pandas.read_csv(os.path.join(dirname, 'water_balance.dat'), comment='#')
if axs is None:
fig,axs = plt.subplots(1,2, figsize=(12,6))
axs[0].plot(obs['time [d]'], obs['runoff generation [mol d^-1]'] / 55500, label=label)
axs[0].set_xlabel('time [d]')
axs[0].set_ylabel('runoff [m^3 / d]')
axs[1].plot(obs['time [d]'], obs['evapotranspiration [m d^-1]'])
axs[1].set_xlabel('time [d]')
axs[1].set_ylabel('transpiration [m / d]')
return axs
# run the demos
run_demos.run_demo_local(['priestley_taylor',
'priestley_taylor_canopy_evapotranspiration',
'priestley_taylor_canopy_evapotranspiration_relperm_trf'])
axs = plot('priestley_taylor', label='PT')
plot('priestley_taylor_canopy_evapotranspiration', axs, label='PT_can')
plot('priestley_taylor_canopy_evapotranspiration_relperm_trf', axs, label='PT_can_TRF')
axs[0].legend()
plt.show()
Distributed energy equation#
The empirical forms above are derived from a surface energy balance where not all terms are fully specified, or specified empirically. For instance, ground conductance of energy is parameterized.
The next set of models solve an energy transport equation on the surface and subsurface domains. Both examples here solve for diffusion of energy, advection of energy on the surface domain (where advected energy is much larger than lateral diffusion of energy), and a surface energy balance to compute ET through energy conservation.
Full energy The full energy example includes freeze-thaw and phase change, cryosuction, and all other “Arctic” processes
Simple energy The simple energy example simplifies the full energy equation by assuming no phase change
See also the next set of demos for a full study of energy equations.
# run the demos
run_demos.run_demo_local(['full_energy', 'simple_energy'])
plot('full_energy')
array([<Axes: xlabel='time [d]', ylabel='runoff [m^3 / d]'>,
<Axes: xlabel='time [d]', ylabel='transpiration [m / d]'>],
dtype=object)
plot('simple_energy')
---------------------------------------------------------------------------
FileNotFoundError Traceback (most recent call last)
Cell In[14], line 1
----> 1 plot('simple_energy')
Cell In[9], line 3, in plot(testname, axs, label)
1 def plot(testname, axs=None, label=None):
2 dirname = testname+'.demo'
----> 3 obs = pandas.read_csv(os.path.join(dirname, 'water_balance.dat'), comment='#')
4 if axs is None:
5 fig,axs = plt.subplots(1,2, figsize=(12,6))
File ~/code/mambaforge/lib/python3.10/site-packages/pandas/io/parsers/readers.py:912, in read_csv(filepath_or_buffer, sep, delimiter, header, names, index_col, usecols, dtype, engine, converters, true_values, false_values, skipinitialspace, skiprows, skipfooter, nrows, na_values, keep_default_na, na_filter, verbose, skip_blank_lines, parse_dates, infer_datetime_format, keep_date_col, date_parser, date_format, dayfirst, cache_dates, iterator, chunksize, compression, thousands, decimal, lineterminator, quotechar, quoting, doublequote, escapechar, comment, encoding, encoding_errors, dialect, on_bad_lines, delim_whitespace, low_memory, memory_map, float_precision, storage_options, dtype_backend)
899 kwds_defaults = _refine_defaults_read(
900 dialect,
901 delimiter,
(...)
908 dtype_backend=dtype_backend,
909 )
910 kwds.update(kwds_defaults)
--> 912 return _read(filepath_or_buffer, kwds)
File ~/code/mambaforge/lib/python3.10/site-packages/pandas/io/parsers/readers.py:577, in _read(filepath_or_buffer, kwds)
574 _validate_names(kwds.get("names", None))
576 # Create the parser.
--> 577 parser = TextFileReader(filepath_or_buffer, **kwds)
579 if chunksize or iterator:
580 return parser
File ~/code/mambaforge/lib/python3.10/site-packages/pandas/io/parsers/readers.py:1407, in TextFileReader.__init__(self, f, engine, **kwds)
1404 self.options["has_index_names"] = kwds["has_index_names"]
1406 self.handles: IOHandles | None = None
-> 1407 self._engine = self._make_engine(f, self.engine)
File ~/code/mambaforge/lib/python3.10/site-packages/pandas/io/parsers/readers.py:1661, in TextFileReader._make_engine(self, f, engine)
1659 if "b" not in mode:
1660 mode += "b"
-> 1661 self.handles = get_handle(
1662 f,
1663 mode,
1664 encoding=self.options.get("encoding", None),
1665 compression=self.options.get("compression", None),
1666 memory_map=self.options.get("memory_map", False),
1667 is_text=is_text,
1668 errors=self.options.get("encoding_errors", "strict"),
1669 storage_options=self.options.get("storage_options", None),
1670 )
1671 assert self.handles is not None
1672 f = self.handles.handle
File ~/code/mambaforge/lib/python3.10/site-packages/pandas/io/common.py:859, in get_handle(path_or_buf, mode, encoding, compression, memory_map, is_text, errors, storage_options)
854 elif isinstance(handle, str):
855 # Check whether the filename is to be opened in binary mode.
856 # Binary mode does not support 'encoding' and 'newline'.
857 if ioargs.encoding and "b" not in ioargs.mode:
858 # Encoding
--> 859 handle = open(
860 handle,
861 ioargs.mode,
862 encoding=ioargs.encoding,
863 errors=errors,
864 newline="",
865 )
866 else:
867 # Binary mode
868 handle = open(handle, ioargs.mode)
FileNotFoundError: [Errno 2] No such file or directory: 'simple_energy.demo/water_balance.dat'
Prescribed LAI, calculated transpiration#
More realistic than prescribing transpiration is to actively calculate transpiration as a function of photosynthesis. This can be done through vegetation models, which actively simulate a carbon cycle and predict LAI and photosynthesis, or through a biophysics model, which takes as input a prescribed LAI as a function of time (including, e.g. seasonal variations) and calculates photosynthesis, stomatal conductance, and transpiration.
These approaches need no limiters – the limiter is in the stomatal conductance term, which is itself a function of soil moisture or mafic potential.
TO DO Provide demos for any/all of: native vegetation model, ATS+CLM, ELM+ATS, ATS+ELMKernels, and eventually ATS+EcoSys.