IODA
Factory.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020-2021 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 
8 #include "ioda/Engines/Factory.h"
9 
10 #include <iostream>
11 #include <string>
12 
13 #include "ioda/Engines/HH.h"
14 #include "ioda/Engines/ObsStore.h"
15 #include "ioda/Exception.h"
16 #include "ioda/Group.h"
17 #include "ioda/defs.h"
18 
19 namespace ioda {
20 namespace Engines {
21 Group constructFromCmdLine(int argc, char** argv, const std::string& defaultFilename) {
22  /* Options:
23  Start with --ioda-engine-options. All options after this are
24  positional options.
25 
26  --ioda-engine-options name parameters...
27 
28  Currently supported engine names:
29  1. HDF5 with file backend [HDF5-file]
30  2. HDF5 in-memory backend [HDF5-mem]
31 3. ObsStore in-memory backend [obs-store]
32 
33  Engine parameters:
34  1. Needs a file name, file opening properties [create, open],
35  access mode [read, read_write, create, truncate].
36  2. Needs a file name, file size increment length (in MB), flush_on_close (bool).
37  3. None
38  */
40  BackendNames backendName;
41 
42  using std::string;
43  std::vector<string> opts(argc);
44  for (size_t i = 0; i < (size_t)argc; ++i) opts[i] = string(argv[i]);
45 
46  // Find the "--ioda-engine-options" option.
47  // If none is specified, use a default.
48  auto it = std::find(opts.cbegin(), opts.cend(), "--ioda-engine-options");
49  if (it == opts.cend()) {
50  // No options --> create a file using the defaultFileName
51  backendName = BackendNames::Hdf5File;
52  params.fileName = defaultFilename;
55  } else {
56  ++it;
57  if (it == opts.cend()) {
58  throw Exception("Bad option --ioda-engine-options. Got the "
59  "--ioda-engine-options token but nothing else.", ioda_Here());
60  }
61 
62  // convert array of c-string options into vector of strings
63  auto readOpts = [&opts, &it](size_t n) -> std::vector<string> {
64  std::vector<string> res(n);
65  for (size_t i = 0; i < n; ++i) {
66  ++it;
67  if (it == opts.cend())
68  throw Exception("Bad option --ioda-engine-options. "
69  "Wrong number of elements.", ioda_Here()).add("Expected", n);
70  res[i] = std::string(*it);
71  }
72  return res;
73  };
74 
75  string sEngine = *it;
76  if (sEngine == "HDF5-file") {
77  auto engineOpts = readOpts(3);
78  backendName = BackendNames::Hdf5File;
79  params.fileName = engineOpts[0];
80 
81  enum class open_or_create {
82  open,
83  create
84  } action
85  = (engineOpts[1] == "create") ? open_or_create::create : open_or_create::open;
86  if (action == open_or_create::open) {
87  // open action
89  params.openMode = (engineOpts[2] == "read_write") ? BackendOpenModes::Read_Write
91  } else {
92  // create action
94  params.createMode = (engineOpts[2] == "truncate") ? BackendCreateModes::Truncate_If_Exists
96  }
97  } else if (sEngine == "HDF5-mem") {
98  auto engineOpts = readOpts(3);
99  // TODO(ryan): Allow open / create
100  // enum class open_or_create { open, create } action
101  // = (engineOpts[1] == "create") ? open_or_create::create :
102  // open_or_create::open;
103 
104  backendName = BackendNames::Hdf5Mem;
105  params.fileName = engineOpts[0];
108 
109  string sAllocLen_MB = engineOpts[1];
110  string sFlush = engineOpts[2];
111 
112  params.allocBytes = gsl::narrow<size_t>(((size_t)std::stoul(sAllocLen_MB)) * 1024 * 1024);
113  params.flush = (sFlush == "true");
114  } else if (sEngine == "obs-store") {
115  backendName = BackendNames::ObsStore;
116  } else {
117  throw Exception("Bad option --ioda-engine-options. "
118  "Unknown engine.", ioda_Here()).add("Engine", sEngine);
119  }
120  }
121  return constructBackend(backendName, params);
122 }
123 
125  Group backend;
126  if (name == BackendNames::Hdf5File) {
127  if (params.action == BackendFileActions::Open)
128  return HH::openFile(params.fileName, params.openMode);
129  if (params.action == BackendFileActions::Create)
131  throw Exception("Unknown BackendFileActions value", ioda_Here());
132  }
133  if (name == BackendNames::Hdf5Mem) {
134  if (params.action == BackendFileActions::Open)
135  return HH::openMemoryFile(params.fileName, params.openMode, params.flush, params.allocBytes);
136  if (params.action == BackendFileActions::Create)
137  return HH::createMemoryFile(params.fileName, params.createMode, params.flush,
138  params.allocBytes);
139  throw Exception("Unknown BackendFileActions value", ioda_Here());
140  }
142 
143  // If we get to here, then we have a backend name that is
144  // not implemented yet.
145  throw Exception("Backend not implemented yet", ioda_Here());
146 }
147 
148 std::ostream& operator<<(std::ostream& os, const ioda::Engines::BackendCreateModes& mode)
149 {
150  using namespace ioda::Engines;
151  static const std::map<BackendCreateModes, std::string> names {
152  {BackendCreateModes::Truncate_If_Exists, "Truncate_If_Exists"},
153  {BackendCreateModes::Fail_If_Exists, "Fail_If_Exists"}
154  };
155 
156  if (names.count(mode) == 0) throw Exception("Unhandled backend creation mode", ioda_Here());
157  os << "ioda::Engines::BackendCreateModes::" << names.at(mode);
158  return os;
159 }
160 
161 std::ostream& operator<<(std::ostream& os, const ioda::Engines::BackendOpenModes& mode)
162 {
163  using namespace ioda::Engines;
164  static const std::map<BackendOpenModes, std::string> names {
165  {BackendOpenModes::Read_Only, "Read_Only"},
166  {BackendOpenModes::Read_Write, "Read_Write"}
167  };
168  if (names.count(mode) == 0) throw Exception("Unhandled backend open mode", ioda_Here());
169  os << "ioda::Engines::BackendOpenModes::" << names.at(mode);
170  return os;
171 }
172 
173 } // namespace Engines
174 } // namespace ioda
175 
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::Group and related classes.
HDF5 engine.
ObsStore engine.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
Common preprocessor definitions used throughout IODA.
IODA_DL Group createFile(const std::string &filename, BackendCreateModes mode, HDF5_Version_Range compat=defaultVersionRange())
Create a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:120
IODA_DL Group openFile(const std::string &filename, BackendOpenModes mode, HDF5_Version_Range compat=defaultVersionRange())
Open a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:147
IODA_DL Group openMemoryFile(const std::string &filename, BackendOpenModes mode=BackendOpenModes::Read_Only, bool flush_on_close=false, size_t increment_len_bytes=1000000, HDF5_Version_Range compat=defaultVersionRange())
Map an HDF5 file in memory and open a ioda::Group.
Definition: HH.cpp:171
IODA_DL Group createMemoryFile(const std::string &filename, BackendCreateModes mode, bool flush_on_close=false, size_t increment_len_bytes=1000000, HDF5_Version_Range compat=defaultVersionRange())
Create a ioda::Group backed by the HDF5 in-memory-store.
Definition: HH.cpp:86
std::pair< HDF5_Version, HDF5_Version > HDF5_Version_Range
Definition: HH.h:42
@ V18
Use the latest HDF5 v1.8 format for storing objects.
@ V110
Use the latest HDF5 v1.10 format for storing objects.
IODA_DL Group createRootGroup()
Create a ioda::Group backed by an OsbStore Group object.
Definition: ObsStore.cpp:24
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
BackendNames
Backend names.
Definition: Factory.h:28
IODA_DL Group constructBackend(BackendNames name, BackendCreationParameters &params)
This is a simple factory style function that will instantiate a different backend based on a given na...
Definition: Factory.cpp:124
BackendCreateModes
Definition: Factory.h:44
@ Open
Open an existing file.
@ Hdf5Mem
HDF5 in-memory "file".
@ Hdf5File
HDF5 file access.
@ ObsStore
ObsStore in-memory.
@ Fail_If_Exists
If the file already exists, fail with an error.
@ Truncate_If_Exists
If the file already exists, overwrite it.
@ Read_Write
Open the file in read-write mode.
@ Read_Only
Open the file in read-only mode.
The backends that implement the ioda-engines functionality.
Definition: Capabilities.h:21
IODA_DL std::ostream & operator<<(std::ostream &os, const BackendCreateModes &mode)
stream operator
Definition: Factory.cpp:148
#define ioda_Here()
Used to specify backend creation-time properties.
Definition: Factory.h:59