Observation input
Observation input consists of a number of different inputs:
Observation files with correct name and content
Command Line Options
Environment Variables
Observation files
Observation files for TSMP-PDAF are netCDF files which follow a
certain naming convention. Each observation file has a fixed prefix
followed by the formatted number of the assimilation cycle which is
determined by da_interval
. Additionally, the command line
input delt_obs
determines the
number of assimilation cycles/iterations/steps that are run purely
forward before checking an observation file again.
The prefix is chosen with the command line
option obs_filename
.
Example: If the chosen prefix is myobs
, the observation files are
named myobs.00001, myobs.00002, myobs.00003
, etc.
General observation file variables
All observation files contain the following variable:
dim_obs
dim_obs
: (int) The observation files contain one dimension named
dim_obs
which should be set equal to the number of observations for
the respective assimilation cycle. The observation files for ParFlow
should contain the following variables which all should have the
dimension dim_obs
(except variable dr
in CLM observations files):
ParFlow observation file variables
If TSMP-PDAF is only applied with ParFlow, the following variables have to be specified in the observation files:
obs_pf
obs_pf
: (real) Observations for ParFlow (either pressure or soil
moisture)
obserr_pf
obserr_pf
: (real) Observation errors which can be different for
individual observations (optional). If this variable is not
present. the command line option rms_obs
(see Command line
options) will be used to define
the observation error (equal for all observations).
ix
ix
: (integer) Position of the observation in the ParFlow grid in
x-direction.
iy
iy
: (integer) Position of the observation in the ParFlow grid in
y-direction.
iz
iz
: (integer) Position of the observation in the ParFlow grid in
z-direction.
idx
idx
: (integer) Index of the observation in the ParFlow grid.
The positions are always relative to the south-east corner cell at the
lowest model layer. The index idx
can be calculated as follows:
where nx
and ny
are the number of grid cells in x- and y-direction
respectively and ix
, iy
and iz
are the positions of the
observation in x-, y- and z-direction.
CLM observation file variables
If TSMP-PDAF is only applied with CLM, different variables have to be specified in the observation files:
obs_clm
obs_clm
: (real) Observations for CLM (soil moisture)
obserr_clm
obserr_clm
: (real) Observation errors which can be different for
individual observations (optional). If this variable is not
present. the command line option rms_obs
(see Command line
options) will be used to define
the observation error (equal for all observations).
lon
lon
: (real) Longitude of the observation.
lat
lat
: (real) Latitude of the observation.
layer
layer
: (integer) CLM layer where the observation is located (counted
from uppermost CLM layer).
var_id
var_id
: (integer) Only used for multiscalar data assimilation. Size
of var_id
is same as dim_obs
. var_id
has same values over model
grid cells, which have range of similar observations values from raw
data over them. The values can be grouped starting from 1 for similar
observation values to other integers (2,3,4,5 etc.) for other similar
observations. If there are no observations over some grid cells than a
negative var_id
(integer) is assigned to them.
dr
dr
: (real) Snapping distance for the observation.
Attention This variable should have a length of 2
(one snapping
distance for longitudes and one for latitudes).
Each of the CLM observations is snapped to the nearest CLM grid cell
based on the given lon
, lat
and the snapping distance dr
which
should be smaller than the minimum grid cell size.
Multiscalar Data Assimilation
The multiscalar data assimilation has been implemented for Local
Ensemble Transform Kalman Filter(LETKF) filter (filtertype=5
). For
multiscalar data assimilation, we need to specify in enkfpf.par
the
entry point_obs
(in [DA]
) to value 0 (integer) for using
multiscalar data assimilation (eg. using SMAP satellite data over a
large area which is not point data).
Then in the observation files we need to specify variable var_id
values. Size of var_id
is same as dim_obs
. var_id
has same
values over model grid cells, which have range of similar observations
values from raw data over them. The values can be grouped starting
from 1 for similar observation values to other integers (2,3,4,5 etc.)
for other similar obersvations. If there are no observations over some
grid cells than a negative var_id
(integer) is assigned to them.
Observation time flexibility
Currently, in TSMP-PDAF there are a number input options that determine, when observations are read-in and assimilated:
Parflow:
TimeInfo.BaseUnit
Observation file NetCDF variable
no_obs
The goal of this section is to show how these inputs can be used to create a flexible observation time input.
First, da_interval
and TimeInfo.BaseUnit
define the forward
integration step that is executed during on iteration of the main
TSMP-PDAF loop. This is the smallest time interval in your model and
it determines the counter, also for observation file names. So
da_interval
should be small enough that each possible observation
can be attributed to one timestep.
Next, delt_obs
is used to specify if data assimilation is not to be
called at every iteration/cycle of da_interval
. If two available
observations are only da_interval
apart, delt_obs
has to be set to
1
. However, suppose observations daily at noon, and da_interval
is
one hour, then delt_obs
could be chosen as 24
.
Now, PDAF will be called every delt_obs
iterations/cycles. However,
there is one last way of manipulating observation input, by setting
the NetCDF variable no_obs
in the observation file to 0
. In this
case, PDAF will skip the assimilation during that observation files
cycle. So, in principle, one could set delt_obs
to 1
, thus needing
an observation file at every cycle. At each timestep that is not
supposed to be assimilated, one could supply an alsmost-empty
observation file with just a single variable no_obs
that has the
value 0
. By this workaround, one obtains a fairly flexible
observation time input!
Example code for setting variable no_obs
to zero in an existing NetCDF
file:
import scipy.io as scio
f = scio.netcdf_file("swc_obs.00001","a", mmap=False) # append
f.variables["no_obs"].assignValue(0)
f.close()
Example code for generating a minimal NetCDF observation file with
just one variable no_obs
set to zero:
import numpy as np
import scipy.io as scio
f = scio.netcdf_file("swc_obs.00001","w", mmap=False) # write
f.createDimension("dim_noobs", 1)
f.createVariable("no_obs", np.int32, ["dim_noobs"])
f.variables["no_obs"].assignValue(0)
f.close()
Specifying type of observation at compile time
The following environment variables let the user specify the expected observational input (i.e. ParFlow or CLM observations) at compile time (during the build-process). This may save some time during execution as certain parts of the source code are not accessed at all.
CLM observations: Set
ParFlow observations: Set
Example for setting environment variables
The aforementioned environment variables can be set in the PDAF-build
script
TSMP/bldsva/intf_DA/pdaf1_1/arch/JURECA/build_interface_pdaf1_1_JURECA.ksh
(or replace JURECA
with other machine).
Source code from script build_interface_pdaf1_1_JURECA.ksh
:
if [[ $withCLM == "true" && $withCOS == "false" && $withPFL == "true" ]] ; then
importFlags+=$importFlagsCLM
importFlags+=$importFlagsOAS
importFlags+=$importFlagsPFL
importFlags+=$importFlagsDA
cppdefs+=" ${pf}-Duse_comm_da ${pf}-DCOUP_OAS_PFL ${pf}-DMAXPATCH_PFT=1 "
cppdefs+=" ${pf}-DOBS_ONLY_PARFLOW " # Remove for observations from both ParFlow + CLM
if [[ $readCLM == "true" ]] ; then ; cppdefs+=" ${pf}-DREADCLM " ; fi
if [[ $freeDrain == "true" ]] ; then ; cppdefs+=" ${pf}-DFREEDRAINAGE " ; fi
libs+=$libsCLM
libs+=$libsOAS
libs+=$libsPFL
obj+=' $(OBJCLM) $(OBJPF) '
fi
Here we see the flags that are set when the compilation flag -c clm-pfl
is used. Interesting is the second line starting with
cppdefs+=...
. Here OBS_ONLY_PARFLOW
is set. In analogous fashion,
one of the other environment variables mentioned before could be set.