IODA
obsgroup/test.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 UCAR
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 #include <cstdlib>
8 #include <iostream>
9 #include <numeric>
10 
11 #include "Eigen/Dense"
12 #include "ioda/Engines/Factory.h"
13 #include "ioda/Exception.h"
14 #include "ioda/ObsGroup.h"
15 #include "ioda/defs.h"
16 #include "unsupported/Eigen/CXX11/Tensor"
17 
18 int main(int argc, char** argv) {
19  try {
20  using namespace ioda;
21  auto f = Engines::constructFromCmdLine(argc, argv, "test-obsgroup1.hdf5");
22 
23  // ATMS profiles. Generally 11 scan lines per granule,
24  // 96 scan positions per line. CRTM retrieves along
25  // 101 pressure levels, so 100 pressure layers.
26  // Time is expressed as a unit of scan line, so it is not a
27  // fundamental dimension here.
28 
29  // We create the base dimensions along with the ObsGroup.
30  // Adding attributes to the dimensions is done later.
31  const int atms_scanpos = 96;
32  const int atms_numchannels = 22;
33  const int crtm_numlevels = 101;
34  const int crtm_numlayers = 100;
35  const int atms_initlines = 66; // Scan lines can vary, but for this example we write 11.
36  // This function makes a brand new ObsGroup with pre-allocated dimensions.
37  // For each dimension, we provide a name, whether it is Horizontal, Vertical, etc. (
38  // for Locations and GeoVaLs stuff), an initial size, a maximum size, and
39  // a "hint" for setting chunking properties of derived variables.
40 
41  auto og = ObsGroup::generate(
42  f, {NewDimensionScale<int>("ScanPosition", atms_scanpos, atms_scanpos, atms_scanpos),
43  NewDimensionScale<int>("ScanLine", atms_initlines, -1, 11),
44  NewDimensionScale<int>("Level", crtm_numlevels, crtm_numlevels, crtm_numlevels),
45  NewDimensionScale<int>("Layer", crtm_numlayers, crtm_numlayers, crtm_numlayers),
46  NewDimensionScale<int>("Channel", atms_numchannels, atms_numchannels,
47  atms_numchannels)});
48 
49  // We want to use variable chunking and turn on GZIP compression.
51  params.chunk = true;
52  params.compressWithGZIP();
53 
54  // We want to set a default fill value.
55  // TODO(rhoneyager): See if we always want to set this to a sensible default.
56  auto params_double = params;
57  params_double.setFillValue<double>(-999);
58  auto params_float = params;
59  params_float.setFillValue<float>(-999);
60  auto params_int = params;
61  params_int.setFillValue<int>(-999);
62 
63  /*
64  // TODO: Make this automatic.
65 
66  // Let's deliberately fill in the scan position, scan line and channel numbers so
67  /// that Panoply plots look nice.
68  Eigen::ArrayXi eScanPos(atms_scanpos);
69  eScanPos.setLinSpaced(1, atms_scanpos);
70  og.all_dims["ScanPosition"].writeWithEigenRegular(eScanPos);
71 
72  Eigen::ArrayXi eScanLine(atms_initlines);
73  eScanLine.setLinSpaced(1, atms_initlines);
74  og.all_dims["ScanLine"].writeWithEigenRegular(eScanLine);
75 
76  std::array<int, atms_numchannels> vChannel{ 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,
77  17,18,19,20,21,22 };
78  og.all_dims["Channel"].write<int>(vChannel);
79  */
80 
81  // Writing in the data
82 
83  std::array<std::string, atms_numchannels> vCentrFreq{
84  "23.8", "31.4", "50.3", "51.76", "52.8", "53.596", "54.40", "54.94",
85  "55.50", "57.29034", "57.29034", "57.29034", "57.29034", "57.29034", "57.29034", "88.20",
86  "165.5", "183.31", "183.31", "183.31", "183.31", "183.31"};
87  og.vars.createWithScales<std::string>("CenterFreq@MetaData", {og.vars["Channel"]}, params)
88  .write<std::string>(vCentrFreq)
89  .atts.add<std::string>("long_name", std::string("Center frequency of instrument channel"))
90  .add("units", std::string("GHz"));
91 
92  std::array<int, atms_numchannels> vPol{0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1};
93  og.vars.createWithScales<int>("MetaData/Polarization", {og.vars["Channel"]}, params_int)
94  .write<int>(vPol)
95  .atts.add<std::string>("long_name", std::string("Polarization of instrument channel"))
96  .add<int>("valid_range", {0, 6});
97 
98  // Let's assume a magical swath that has a bottom corner of (0,0), with 0.5 degree spacing
99  // in both latitude and longitude.
100  // Let's also make the TBs smoothly varying for each channel. This is total garbage data,
101  // but it plots nicely.
102 
103  Eigen::ArrayXXf eLat(atms_initlines, atms_scanpos);
104  Eigen::ArrayXXf eLon(atms_initlines, atms_scanpos);
105  // Note the different index order for a 3-D tensor. This is to align Eigen and
106  // the row-major conventions.
107  Eigen::Tensor<float, 3, Eigen::RowMajor> eBTs(atms_initlines, atms_scanpos, atms_numchannels);
108  for (int i = 0; i < atms_initlines; ++i)
109  for (int j = 0; j < atms_scanpos; ++j) {
110  eLat(i, j) = static_cast<float>(i) / 2.f;
111  eLon(i, j) = (static_cast<float>(j) / 2.f) + (static_cast<float>(i) / 6.f);
112  for (int k = 0; k < atms_numchannels; ++k) {
113  const float degtorad = 3.141592654f / 180.f;
114  eBTs(i, j, k) =
115  150 + (150 * sin(degtorad * static_cast<float>(i))) +
116  (100 * cos(degtorad * static_cast<float>(15 + (4 * j) + (8 * k)))) +
117  (15 * sin(2 * degtorad * static_cast<float>(i)) * cos(4 * degtorad * static_cast<float>(j)));
118  }
119  }
120 
121  og.vars
122  .createWithScales<float>("Latitude@MetaData", {og.vars["ScanLine"], og.vars["ScanPosition"]},
123  params_float)
124  .writeWithEigenRegular(eLat)
125  .atts.add<std::string>("long_name", std::string("Latitude"))
126  .add<std::string>("units", std::string("degrees_north"))
127  .add<float>("valid_range", {-90, 90});
128 
129  og.vars
130  .createWithScales<float>("Longitude@MetaData", {og.vars["ScanLine"], og.vars["ScanPosition"]},
131  params_float)
132  .writeWithEigenRegular(eLon)
133  .atts.add<std::string>("long_name", std::string("Longitude"))
134  .add<std::string>("units", std::string("degrees_east"))
135  .add<float>("valid_range", {-360, 360});
136 
137  // Note again the ordering.
138  og.vars
139  .createWithScales<float>("ObsValue/Inst_Brightness_Temperature_Uncorrected",
140  {og.vars["ScanLine"], og.vars["ScanPosition"], og.vars["Channel"]},
141  params_float)
142  .writeWithEigenTensor(eBTs)
143  .atts.add<std::string>("long_name", std::string("Raw instrument brightness temperature"))
144  .add<std::string>("units", std::string("K"))
145  .add<float>("valid_range", {120, 500})
146  // Default display settings
147  .add<std::string>("coordinates", std::string("Longitude Latitude Channel"));
148 
149  // Some tests
150  Expects(!og.vars["ScanLine"].getDimensionScaleName().empty());
151  Expects(og.vars["ObsValue/Inst_Brightness_Temperature_Uncorrected"].isDimensionScaleAttached(
152  0, og.vars["ScanLine"]));
153  } catch (const std::exception& e) {
155  return 1;
156  }
157  return 0;
158 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::ObsGroup and related classes.
static ObsGroup generate(Group &emptyGroup, const NewDimensionScales_t &fundamentalDims, std::shared_ptr< const detail::DataLayoutPolicy > layout=nullptr)
Create an empty ObsGroup and populate it with the fundamental dimensions.
Definition: ObsGroup.cpp:72
Common preprocessor definitions used throughout IODA.
IODA_DL Group constructFromCmdLine(int argc, char **argv, const std::string &defaultFilename)
This is a wrapper function around the constructBackend function for creating a backend based on comma...
Definition: Factory.cpp:21
IODA_DL void unwind_exception_stack(const std::exception &e, std::ostream &out=std::cerr, int level=0)
Convenience function for unwinding an exception stack.
Definition: Exception.cpp:48
int main(int argc, char **argv)
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57