bes Updated for version 3.21.1
The Backend Server (BES) is the lower two tiers of the Hyrax data server
BESContainerStorageVolatile.cc
1// BESContainerStorageVolatile.cc
2
3// This file is part of bes, A C++ back-end server implementation framework
4// for the OPeNDAP Data Access Protocol.
5
6// Copyright (c) 2004-2009 University Corporation for Atmospheric Research
7// Author: Patrick West <pwest@ucar.edu> and Jose Garcia <jgarcia@ucar.edu>
8//
9// This library is free software; you can redistribute it and/or
10// modify it under the terms of the GNU Lesser General Public
11// License as published by the Free Software Foundation; either
12// version 2.1 of the License, or (at your option) any later version.
13//
14// This library is distributed in the hope that it will be useful,
15// but WITHOUT ANY WARRANTY; without even the implied warranty of
16// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17// Lesser General Public License for more details.
18//
19// You should have received a copy of the GNU Lesser General Public
20// License along with this library; if not, write to the Free Software
21// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22//
23// You can contact University Corporation for Atmospheric Research at
24// 3080 Center Green Drive, Boulder, CO 80301
25
26// (c) COPYRIGHT University Corporation for Atmospheric Research 2004-2005
27// Please read the full copyright statement in the file COPYRIGHT_UCAR.
28//
29// Authors:
30// pwest Patrick West <pwest@ucar.edu>
31// jgarcia Jose Garcia <jgarcia@ucar.edu>
32
33#include "config.h"
34
35#include "BESContainerStorageVolatile.h"
36#include "BESFileContainer.h"
37#include "BESInternalError.h"
38#include "BESSyntaxUserError.h"
39#include "BESInfo.h"
40#include "TheBESKeys.h"
41#include "BESUtil.h"
42#include "BESServiceRegistry.h"
43#include "BESDebug.h"
44
45// FIXME This is a lie, but it will help with debugging/design for the RemoteResources
46// FIXME design/fix. Remove this once that's done. jhrg 8/7/20
47#define MODULE "ngap"
48#define prolog std::string("BESContainerStorageVolatile::").append(__func__).append("() - ")
49
50using std::endl;
51using std::string;
52using std::list;
53using std::ostream;
54
70 BESContainerStorage(n), _root_dir(""), _follow_sym_links(false)
71{
72 string key = "BES.Data.RootDirectory";
73 bool found = false;
74 TheBESKeys::TheKeys()->get_value(key, _root_dir, found);
75 if (_root_dir == "") {
76 string s = key + " not defined in BES configuration file";
77 throw BESSyntaxUserError(s, __FILE__, __LINE__);
78 }
79
80 found = false;
81 key = (string) "BES.FollowSymLinks";
82 string s_str;
83 TheBESKeys::TheKeys()->get_value(key, s_str, found);
84 s_str = BESUtil::lowercase(s_str);
85 if (found && (s_str == "yes" || s_str == "on" || s_str == "true")) {
86 _follow_sym_links = true;
87 }
88}
89
90BESContainerStorageVolatile::~BESContainerStorageVolatile()
91{
93}
94
106{
107 BESContainer *ret_container = 0;
108
109 BESContainerStorageVolatile::Container_citer i;
110 i = _container_list.find(sym_name);
111 if (i != _container_list.end()) {
112 BESContainer *c = (*i).second;
113 ret_container = c->ptr_duplicate();
114 }
115
116 return ret_container;
117}
118
135void BESContainerStorageVolatile::add_container(const string &sym_name, const string &real_name, const string &type)
136{
137 // The type must be specified so that we can find the request handler
138 // that knows how to handle the container.
139 // Changed sym_name to real_name to make the message clearer. jhrg 11/14/19
140 if (type.empty())
141 throw BESInternalError(string("Unable to add container '").append(real_name).append("', the type of data must be specified."), __FILE__, __LINE__);
142
143 // if the container already exists then throw an error
144 BESContainerStorageVolatile::Container_citer i = _container_list.find(sym_name);
145 if (i != _container_list.end()) {
146 throw BESInternalError(string("A container with the name '").append(sym_name).append("' already exists"), __FILE__, __LINE__);
147 }
148
149 // make sure that the path to the container exists. If follow_sym_links
150 // is false and there is a symbolic link in the path then an error will
151 // be thrown. If the path does not exist, an error will be thrown.
152 BESUtil::check_path(real_name, _root_dir, _follow_sym_links);
153
154 // add the root directory to the real_name passed
155 // Removed the 'false' since that is the default value. jhrg 3/3/25
156 string fully_qualified_real_name = BESUtil::assemblePath(_root_dir, real_name);
157
158 BESDEBUG("container","BESContainerStorageVolatile::add_container() - "
159 << " _root_dir: " << _root_dir
160 << " real_name: " << real_name
161 << " symbolic name: " << sym_name
162 << " fully_qualified_real_name: " << fully_qualified_real_name
163 << " type: " << type
164 << endl);
165
166 // Create the file container with the new information
167 BESContainer *c = new BESFileContainer(sym_name, fully_qualified_real_name, type);
168 c->set_relative_name(real_name);
169
170 // add it to the container list
171 _container_list[sym_name] = c;
172}
173
192{
193 if (!c) {
194 throw BESInternalError("Unable to add container, container passed is null", __FILE__, __LINE__);
195 }
196 if (c->get_container_type().empty()) {
197 throw BESInternalError("Unable to add container, the type of data must be specified.", __FILE__, __LINE__);
198 }
199
200 string sym_name = c->get_symbolic_name();
201
202 BESContainerStorageVolatile::Container_citer i = _container_list.find(sym_name);
203 if (i != _container_list.end()) {
204 throw BESInternalError(string("A container with the name '").append(sym_name).append("' already exists"), __FILE__, __LINE__);
205 }
206
207 _container_list[sym_name] = c;
208}
209
217{
218 BESDEBUG(MODULE, prolog << "BEGIN: " << s_name << endl);
219 bool ret = false;
220 BESContainerStorageVolatile::Container_iter i = _container_list.find(s_name);
221 if (i != _container_list.end()) {
222 BESContainer *c = (*i).second;
223 _container_list.erase(i);
224 if (c) {
225 BESDEBUG(MODULE, prolog << "delete the container: "<< (void *) c << endl);
226 delete c;
227 }
228 ret = true;
229 }
230 BESDEBUG(MODULE, prolog << "END" << endl);
231 return ret;
232}
233
242{
243 BESDEBUG(MODULE, prolog << "BEGIN" << endl);
244 while (_container_list.size() != 0) {
245 Container_iter ci = _container_list.begin();
246 BESContainer *c = (*ci).second;
247 _container_list.erase(ci);
248 if (c) {
249 BESDEBUG(MODULE, prolog << "delete the container: "<< (void *) c << endl);
250 delete c;
251 }
252 }
253 BESDEBUG(MODULE, prolog << "END" << endl);
254 return true;
255}
256
264bool BESContainerStorageVolatile::isData(const string &inQuestion, list<string> &provides)
265{
266 bool isit = false;
267 BESContainer *c = look_for(inQuestion);
268 if (c) {
269 isit = true;
270 string node_type = c->get_container_type();
271 BESServiceRegistry::TheRegistry()->services_handled(node_type, provides);
272 }
273 return isit;
274}
275
291{
292 info.add_tag("name", get_name());
293 string::size_type root_len = _root_dir.size();
294 BESContainerStorageVolatile::Container_iter i = _container_list.begin();
295 BESContainerStorageVolatile::Container_iter e = _container_list.end();
296 for (; i != e; i++) {
297 BESContainer *c = (*i).second;
298 string sym = c->get_symbolic_name();
299 string real = c->get_real_name();
300 if (real.size() > root_len) {
301 if (real.compare(0, root_len, _root_dir) == 0) {
302 real = real.substr(root_len, real.size() - root_len);
303 }
304 }
305 string type = c->get_container_type();
306 show_container(sym, real, type, info);
307 }
308}
309
317void BESContainerStorageVolatile::dump(ostream &strm) const
318{
319 strm << BESIndent::LMarg << "BESContainerStorageVolatile::dump - (" << (void *) this << ")" << endl;
320 BESIndent::Indent();
321 strm << BESIndent::LMarg << "name: " << get_name() << endl;
322 if (_container_list.size()) {
323 strm << BESIndent::LMarg << "containers:" << endl;
324 BESIndent::Indent();
325 BESContainerStorageVolatile::Container_citer i = _container_list.begin();
326 BESContainerStorageVolatile::Container_citer ie = _container_list.end();
327 for (; i != ie; i++) {
328 BESContainer *c = (*i).second;
329 c->dump(strm);
330 }
331 BESIndent::UnIndent();
332 }
333 else {
334 strm << BESIndent::LMarg << "containers: none" << endl;
335 }
336 BESIndent::UnIndent();
337}
338
virtual BESContainer * look_for(const std::string &sym_name)
looks for the specified container using the symbolic name passed
virtual void show_containers(BESInfo &info)
show information for each container in this persistent store
virtual bool del_container(const std::string &s_name)
removes a container with the given symbolic name from the list and deletes it.
virtual bool isData(const std::string &inQuestion, std::list< std::string > &provides)
determine if the given container is data and what services are available for it
BESContainerStorageVolatile(const std::string &n)
create an instance of this persistent store with the given name.
virtual bool del_containers()
removes all container
virtual void dump(std::ostream &strm) const
dumps information about this object
virtual void add_container(BESContainer *c)
add the passed container to the list of containers in volatile storage
virtual void show_container(const std::string &sym_name, const std::string &real_name, const std::string &type, BESInfo &info)
add information for a container to the informational response object
virtual const std::string & get_name() const
retrieve the name of this persistent store
BESContainerStorage(const std::string &name)
create an instance of BESContainerStorage with the given name.
A container is something that holds data. E.G., a netcdf file or a database entry.
std::string get_symbolic_name() const
retrieve the symbolic name for this container
virtual BESContainer * ptr_duplicate()=0
pure abstract method to duplicate this instances of BESContainer
void dump(std::ostream &strm) const override
dumps information about this object
std::string get_container_type() const
retrieve the type of data this container holds, such as cedar or netcdf.
void set_relative_name(const std::string &relative)
Set the relative name of the object in this container.
std::string get_real_name() const
retrieve the real name for this container, such as a file name.
Holds real data, container type and constraint for symbolic name read from persistence.
informational response object
Definition BESInfo.h:63
exception thrown if internal error encountered
error thrown if there is a user syntax error in the request or any other user error
static void check_path(const std::string &path, const std::string &root, bool follow_sym_links)
Is the combination of root + path a pathname the BES can/should access?
Definition BESUtil.cc:385
static std::string lowercase(const std::string &s)
Definition BESUtil.cc:257
static std::string assemblePath(const std::string &firstPart, const std::string &secondPart, bool leadingSlash=false, bool trailingSlash=false)
Assemble path fragments making sure that they are separated by a single '/' character.
Definition BESUtil.cc:804
void get_value(const std::string &s, std::string &val, bool &found)
Retrieve the value of a given key, if set.
static TheBESKeys * TheKeys()
Access to the singleton.
Definition TheBESKeys.cc:85