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:

\[\begin{gather*} idx = (iz-1) \cdot nx \cdot ny + (iy-1) \cdot nx + ix \end{gather*}\]

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:

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.