IODA
Engines/ObsStore/Group.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 /*! \addtogroup ioda_internals_engines_obsstore
8  *
9  * @{
10  * \file Group.cpp
11  * \brief Functions for ObsStore Group and Has_Groups
12  */
13 #include "./Group.hpp"
14 
15 #include <stdexcept>
16 
17 #include "./Variables.hpp"
18 #include "ioda/defs.h"
19 #include "ioda/Exception.h"
20 
21 namespace ioda {
22 namespace ObsStore {
24  : atts(std::make_shared<Has_Attributes>()), vars(std::make_shared<Has_Variables>()) {}
25 Group::~Group() = default;
26 
27 std::list<std::string> Group::list() const {
28  std::list<std::string> childList;
29  for (auto igrp = child_groups_.begin(); igrp != child_groups_.end(); ++igrp) {
30  childList.push_back(igrp->first);
31  }
32  return childList;
33 }
34 
35 void Group::listObjects(ObjectType filter, bool recurse,
36  std::map<ObjectType, std::list<std::string>>& res,
37  const std::string& prefix) const {
38  // If we either want to list groups or do any type of recursion, then we need
39  // to get the one-level child groups.
40  bool needGroups = filter == ObjectType::Ignored || filter == ObjectType::Group || recurse;
41  bool needVars = filter == ObjectType::Ignored || filter == ObjectType::Variable;
42 
43  if (needVars) {
44  // Prepare empty list if it does not already exist
45  if (!res.count(ObjectType::Variable)) res[ObjectType::Variable] = std::list<std::string>();
46 
47  // Back-insert these after prepending the prefix.
48  auto oneLevelVars = vars->list();
49  for (auto& v : oneLevelVars) res[ObjectType::Variable].push_back(prefix + v);
50  }
51 
52  if (needGroups || recurse) {
53  auto oneLevelChildren = list();
54 
55  if (needGroups) {
56  // Prepare empty list if it does not already exist
57  if (!res.count(ObjectType::Group)) res[ObjectType::Group] = std::list<std::string>();
58 
59  // Back-insert these after prepending the prefix.
60  for (auto& v : oneLevelChildren) res[ObjectType::Group].push_back(prefix + v);
61  }
62 
63  if (recurse) {
64  for (const auto& child : oneLevelChildren)
65  child_groups_.at(child)->listObjects(filter, recurse, res, std::string(child) + "/");
66  }
67  }
68 }
69 
70 bool Group::exists(const std::string& name) {
71  std::shared_ptr<Group> childGroup = open(name, false);
72  return (childGroup != nullptr);
73 }
74 
75 std::shared_ptr<Group> Group::create(const std::string& name) {
76  // split name into first group and remaining children of the first group
77  // ie, "a/b/c/d" -> "a", "b/c/d"
78  std::vector<std::string> pathSections = splitFirstLevel(name);
79 
80  // If the child exists grab it, otherwise create it.
81  std::shared_ptr<Group> childGroup;
82  if (this->exists(pathSections[0])) {
83  childGroup = this->open(pathSections[0]);
84  } else {
85  childGroup = std::make_shared<Group>();
86  childGroup->vars->setParentGroup(childGroup);
87  child_groups_.insert(
88  std::pair<std::string, std::shared_ptr<Group>>(pathSections[0], childGroup));
89  }
90 
91  // Recurse if there are more levels in the input name
92  if (pathSections.size() > 1) {
93  childGroup = childGroup->create(pathSections[1]);
94  }
95 
96  return childGroup;
97 }
98 
99 std::shared_ptr<Group> Group::open(const std::string& name, const bool throwIfNotFound) {
100  std::shared_ptr<Group> childGroup(nullptr);
101 
102  // split name into first group and remaining children of the first group
103  // ie, "a/b/c/d" -> "a", "b/c/d"
104  std::vector<std::string> pathSections = splitFirstLevel(name);
105 
106  auto igrp = child_groups_.find(pathSections[0]);
107  if (igrp == child_groups_.end()) {
108  childGroup = nullptr;
109  } else {
110  childGroup = igrp->second;
111  }
112 
113  // Recurse if there are more levels in the input name
114  if ((pathSections.size() > 1) && (childGroup != nullptr)) {
115  childGroup = childGroup->open(pathSections[1]);
116  }
117 
118  if (throwIfNotFound && (childGroup == nullptr)) {
119  throw Exception("Child group not found", ioda_Here());
120  }
121 
122  return childGroup;
123 }
124 
125 std::shared_ptr<Group> Group::createRootGroup() {
126  std::shared_ptr<Group> group = std::make_shared<Group>();
127  group->vars->setParentGroup(group);
128  return group;
129 }
130 
131 // Private methods
132 std::vector<std::string> Group::splitFirstLevel(const std::string& path) {
133  std::vector<std::string> pathSections;
134  auto pos = path.find('/');
135  pathSections.push_back(path.substr(0, pos));
136  if (pos != std::string::npos) {
137  pathSections.push_back(path.substr(pos + 1));
138  }
139  return pathSections;
140 }
141 } // namespace ObsStore
142 } // namespace ioda
143 
144 /// @}
IODA's error system.
Functions for ObsStore Group and Has_Groups.
Functions for ObsStore Variable and Has_Variables.
The ioda exception class.
Definition: Exception.h:54
void listObjects(ObjectType filter, bool recurse, std::map< ObjectType, std::list< std::string >> &res, const std::string &prefix="") const
List child objects.
static std::shared_ptr< Group > createRootGroup()
Creates a root group.
std::list< std::string > list() const
List all groups under this group.
bool exists(const std::string &name)
returns true if child group exists
static std::vector< std::string > splitFirstLevel(const std::string &path)
split a path into the first level and remainder of the path
std::shared_ptr< Has_Variables > vars
container for variables
Definition: Group.hpp:47
std::map< std::string, std::shared_ptr< Group > > child_groups_
container for child groups
Definition: Group.hpp:33
std::shared_ptr< Group > open(const std::string &name, const bool throwIfNotFound=true)
open an existing child group
std::shared_ptr< Group > create(const std::string &name)
create a new group
Common preprocessor definitions used throughout IODA.
@ ObsStore
ObsStore in-memory.
#define ioda_Here()