Skip to content

Commit

Permalink
Support new flexible YAML settings file format.
Browse files Browse the repository at this point in the history
  • Loading branch information
josephbirkner committed Jul 5, 2024
1 parent 5f232c6 commit b5c0678
Show file tree
Hide file tree
Showing 2 changed files with 88 additions and 20 deletions.
10 changes: 8 additions & 2 deletions libs/httpcl/include/httpcl/http-settings.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
#include <map>
#include <vector>
#include <string>
#include "yaml-cpp/yaml.h"


namespace httpcl
Expand Down Expand Up @@ -41,6 +42,10 @@ struct Config
std::string keychain;
};

std::optional<std::string> scope;
std::regex urlPattern;
std::string urlPatternString;

std::map<std::string, std::string> cookies;
std::optional<BasicAuthentication> auth;
std::optional<Proxy> proxy;
Expand All @@ -67,7 +72,7 @@ struct Config
};

/**
* Loads settings from HTTP_SETTINGS_FILE.
* Loads/stores settings from/to HTTP_SETTINGS_FILE.
* Allows returning config for a specific URL.
*/
struct Settings
Expand All @@ -85,7 +90,8 @@ struct Settings
/**
* Map from URL pattern to some config values.
*/
std::map<std::string, Config> settings;
std::vector<Config> settings;
YAML::Node document;
};

struct secret
Expand Down
98 changes: 80 additions & 18 deletions libs/httpcl/src/http-settings.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -112,9 +112,51 @@ struct convert<Config::Proxy>
}

namespace {
YAML::Node configToNode(Config const& config, std::string const& url=".*") {

std::string convertToRegex(const std::string& scope) {
std::string regexPattern = "^";
for (char c : scope) {
switch (c) {
case '*':
regexPattern += ".*";
break;
case '.':
regexPattern += "\\.";
break;
case '\\':
regexPattern += "\\\\";
break;
case '^':
case '$':
case '|':
case '(':
case ')':
case '[':
case ']':
case '{':
case '}':
case '?':
case '+':
case '-':
case '!':
regexPattern += '\\';
regexPattern += c;
break;
default:
regexPattern += c;
break;
}
}
regexPattern += ".*$";
return regexPattern;
}

YAML::Node configToNode(Config const& config) {
YAML::Node result;
result["url"] = url;
if (config.scope)
result["scope"] = *config.scope;
else
result["url"] = config.urlPatternString;

if (!config.cookies.empty())
result["cookies"] = config.cookies;
Expand All @@ -139,16 +181,21 @@ YAML::Node configToNode(Config const& config, std::string const& url=".*") {
return result;
}

std::pair<Config, std::string> configFromNode(YAML::Node const& node)
Config configFromNode(YAML::Node const& node)
{
std::string urlPattern;
Config conf;

if (auto entryParam = node["url"])
urlPattern = entryParam.as<std::string>();
else
throw std::runtime_error(
"HTTP Settings: Missing 'url' field in: " + YAML::Dump(node));
if (auto entryParam = node["url"]) {
conf.urlPattern = conf.urlPatternString = entryParam.as<std::string>();
}
else {
if (auto entryParamScope = node["scope"])
conf.scope = entryParamScope.as<std::string>();
else
conf.scope = "*";
conf.urlPatternString = convertToRegex(*conf.scope);
conf.urlPattern = conf.urlPatternString;
}

if (auto cookies = node["cookies"])
conf.cookies = cookies.as<std::map<std::string, std::string>>();
Expand All @@ -172,7 +219,7 @@ std::pair<Config, std::string> configFromNode(YAML::Node const& node)
if (auto apiKey = node["api-key"])
conf.apiKey = apiKey.as<std::string>();

return {std::move(conf), std::move(urlPattern)};
return conf;
}
}

Expand Down Expand Up @@ -302,12 +349,20 @@ void Settings::load()

try {
log().debug("Loading HTTP settings from '{}'...", cookieJar);
auto node = YAML::LoadFile(cookieJar);
document = YAML::LoadFile(cookieJar);
YAML::Node httpSettingsNode;
uint32_t idx = 0;

for (auto const& entry : node.as<std::vector<YAML::Node>>()) {
auto [conf, urlPattern] = configFromNode(entry);
settings[urlPattern] = std::move(conf);
if (document.IsMap()) {
httpSettingsNode = document["http-settings"];
}
else {
// Keep supporting the old format, where the root structure is a settings array.
httpSettingsNode = document;
}

for (auto const& entry : httpSettingsNode.as<std::vector<YAML::Node>>()) {
settings.emplace_back(configFromNode(entry));
++idx;
}

Expand All @@ -330,9 +385,16 @@ void Settings::store()
try {
auto node = YAML::Node();

for (const auto& [key, config] : settings)
for (const auto& config : settings)
node.push_back(configToNode(config));

if (document && document.IsMap()) {
document["http-settings"] = node;
}
else {
document = node;
}

log().debug("Saving HTTP settings to '{}'...", cookieJar);
std::ofstream os(cookieJar);
os << node;
Expand All @@ -345,16 +407,16 @@ void Settings::store()
Config::Config(const std::string& yamlConf)
{
YAML::Node parsedYaml = YAML::Load(yamlConf);
*this = configFromNode(parsedYaml).first;
*this = configFromNode(parsedYaml);
}

Config Settings::operator[] (const std::string &url) const
{
Config result;

for (auto const& [pattern, config] : settings)
for (auto const& config : settings)
{
if (!std::regex_match(url, std::regex(pattern)))
if (!std::regex_match(url, config.urlPattern))
continue;
result |= config;
}
Expand Down

0 comments on commit b5c0678

Please sign in to comment.