summaryrefslogtreecommitdiffstats
path: root/src/common/str_map.cc
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-21 11:54:28 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-21 11:54:28 +0000
commite6918187568dbd01842d8d1d2c808ce16a894239 (patch)
tree64f88b554b444a49f656b6c656111a145cbbaa28 /src/common/str_map.cc
parentInitial commit. (diff)
downloadceph-e6918187568dbd01842d8d1d2c808ce16a894239.tar.xz
ceph-e6918187568dbd01842d8d1d2c808ce16a894239.zip
Adding upstream version 18.2.2.upstream/18.2.2
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/common/str_map.cc')
-rw-r--r--src/common/str_map.cc207
1 files changed, 207 insertions, 0 deletions
diff --git a/src/common/str_map.cc b/src/common/str_map.cc
new file mode 100644
index 000000000..638a30784
--- /dev/null
+++ b/src/common/str_map.cc
@@ -0,0 +1,207 @@
+// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
+// vim: ts=8 sw=2 smarttab
+/*
+ * Ceph - scalable distributed file system
+ *
+ * Copyright (C) 2013 Cloudwatt <libre.licensing@cloudwatt.com>
+ *
+ * Author: Loic Dachary <loic@dachary.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ */
+
+#include "include/str_map.h"
+#include "include/str_list.h"
+
+#include <boost/algorithm/string.hpp>
+
+#include "json_spirit/json_spirit.h"
+
+using namespace std;
+
+int get_json_str_map(
+ const string &str,
+ ostream &ss,
+ str_map_t *str_map,
+ bool fallback_to_plain)
+{
+ json_spirit::mValue json;
+ try {
+ // try json parsing first
+
+ json_spirit::read_or_throw(str, json);
+
+ if (json.type() != json_spirit::obj_type) {
+ ss << str << " must be a JSON object but is of type "
+ << json.type() << " instead";
+ return -EINVAL;
+ }
+
+ json_spirit::mObject o = json.get_obj();
+
+ for (map<string, json_spirit::mValue>::iterator i = o.begin();
+ i != o.end();
+ ++i) {
+ (*str_map)[i->first] = i->second.get_str();
+ }
+ } catch (json_spirit::Error_position &e) {
+ if (fallback_to_plain) {
+ // fallback to key=value format
+ get_str_map(str, str_map, "\t\n ");
+ } else {
+ return -EINVAL;
+ }
+ }
+ return 0;
+}
+
+static std::string_view trim(std::string_view str)
+{
+ static const char* whitespaces = "\t\n ";
+ auto beg = str.find_first_not_of(whitespaces);
+ if (beg == str.npos) {
+ return {};
+ }
+ auto end = str.find_last_not_of(whitespaces);
+ return str.substr(beg, end - beg + 1);
+}
+
+int get_str_map(
+ const string &str,
+ str_map_t* str_map,
+ const char *delims)
+{
+ for_each_pair(str, delims, [str_map](std::string_view key,
+ std::string_view val) {
+ // is the format 'K=V' or just 'K'?
+ if (val.empty()) {
+ str_map->emplace(std::string(key), "");
+ } else {
+ str_map->emplace(std::string(trim(key)), std::string(trim(val)));
+ }
+ });
+ return 0;
+}
+
+str_map_t get_str_map(
+ const string& str,
+ const char* delim)
+{
+ str_map_t str_map;
+ get_str_map(str, &str_map, delim);
+ return str_map;
+}
+
+string get_str_map_value(
+ const str_map_t &str_map,
+ const string &key,
+ const string *def_val)
+{
+ auto p = str_map.find(key);
+
+ // key exists in str_map
+ if (p != str_map.end()) {
+ // but value is empty
+ if (p->second.empty())
+ return p->first;
+ // and value is not empty
+ return p->second;
+ }
+
+ // key DNE in str_map and def_val was specified
+ if (def_val != nullptr)
+ return *def_val;
+
+ // key DNE in str_map, no def_val was specified
+ return string();
+}
+
+string get_str_map_key(
+ const str_map_t &str_map,
+ const string &key,
+ const string *fallback_key)
+{
+ auto p = str_map.find(key);
+ if (p != str_map.end())
+ return p->second;
+
+ if (fallback_key != nullptr) {
+ p = str_map.find(*fallback_key);
+ if (p != str_map.end())
+ return p->second;
+ }
+ return string();
+}
+
+// This function's only purpose is to check whether a given map has only
+// ONE key with an empty value (which would mean that 'get_str_map()' read
+// a map in the form of 'VALUE', without any KEY/VALUE pairs) and, in such
+// event, to assign said 'VALUE' to a given 'def_key', such that we end up
+// with a map of the form "m = { 'def_key' : 'VALUE' }" instead of the
+// original "m = { 'VALUE' : '' }".
+int get_conf_str_map_helper(
+ const string &str,
+ ostringstream &oss,
+ str_map_t* str_map,
+ const string &default_key)
+{
+ get_str_map(str, str_map);
+
+ if (str_map->size() == 1) {
+ auto p = str_map->begin();
+ if (p->second.empty()) {
+ string s = p->first;
+ str_map->erase(s);
+ (*str_map)[default_key] = s;
+ }
+ }
+ return 0;
+}
+
+std::string get_value_via_strmap(
+ const string& conf_string,
+ std::string_view default_key)
+{
+ auto mp = get_str_map(conf_string);
+ if (mp.size() != 1) {
+ return "";
+ }
+
+ // if the one-elem "map" is of the form { 'value' : '' }
+ // replace it with { 'default_key' : 'value' }
+ const auto& [k, v] = *(mp.begin());
+ if (v.empty()) {
+ return k;
+ }
+ return v;
+}
+
+std::string get_value_via_strmap(
+ const string& conf_string,
+ const string& key,
+ std::string_view default_key)
+{
+ auto mp = get_str_map(conf_string);
+ if (mp.size() != 1) {
+ return std::string{};
+ }
+
+ // if the one-elem "map" is of the form { 'value' : '' }
+ // replace it with { 'default_key' : 'value' }
+ const auto& [k, v] = *(mp.begin());
+ if (v.empty()) {
+ return k;
+ }
+ if (k == key) {
+ return k;
+ }
+ if (k == default_key) {
+ return v;
+ }
+
+ return string{};
+}