1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
|
// OpenVPN -- An application to securely tunnel IP networks
// over a single port, with support for SSL/TLS-based
// session authentication and key exchange,
// packet encryption, packet authentication, and
// packet compression.
//
// Copyright (C) 2012- OpenVPN Inc.
//
// SPDX-License-Identifier: MPL-2.0 OR AGPL-3.0-only WITH openvpn3-openssl-exception
//
#ifndef OPENVPN_OPTIONS_SERVPUSH_H
#define OPENVPN_OPTIONS_SERVPUSH_H
#include <string>
#include <sstream>
#include <ostream>
#include <vector>
#include <utility> // for std::move
#include <openvpn/common/exception.hpp>
#include <openvpn/common/options.hpp>
#include <openvpn/common/jsonlib.hpp>
#ifdef HAVE_JSON
#include <openvpn/common/jsonhelper.hpp>
#endif
namespace openvpn {
class ServerPushList : public std::vector<std::string>
{
public:
void parse(const std::string &opt_name, const OptionList &opt)
{
const auto *push = opt.get_index_ptr(opt_name);
if (push)
{
reserve(size() + push->size());
for (auto &i : *push)
{
const Option &o = opt[i];
o.touch();
push_back(o.get(1, 512));
}
}
}
#ifdef HAVE_JSON
// Parse JSON representation of a push list.
// Each push list array element can be one of:
// 1. simple JSON string,
// 2. dictionary containing an "item" string, or
// 3. dictionary containing an "item" array of strings.
void parse(const std::string &title, const Json::Value &push_list) // push_list is JSON array
{
reserve(16); // arbitrary, just a guess
const auto &ja = json::cast_array(push_list, false, title).asArray();
for (size_t i = 0; i < ja.size(); ++i)
{
const Json::Value &jv = ja[i];
if (jv.isString())
push_back(jv.asStringRef());
else if (jv.isObject())
{
const Json::Value &ji = jv["item"];
if (ji.isString())
push_back(ji.asStringRef());
else if (ji.isArray())
{
const auto &ia = ji.asArray();
for (size_t j = 0; j < ia.size(); ++j)
{
const Json::Value &iv = ia[j];
if (iv.isString())
push_back(iv.asStringRef());
else
throw json::json_parse(json::fmt_name(i, title) + " object contains 'item' array that includes non-string element at index=" + std::to_string(j));
}
}
else
throw json::json_parse(json::fmt_name(i, title) + " object must contain 'item' string or array");
}
else
throw json::json_parse(json::fmt_name(i, title) + " must be of type string or object");
}
}
#endif
void extend(const std::vector<std::string> &other)
{
reserve(size() + other.size());
for (auto &e : other)
push_back(e);
}
// do a roundtrip to csv and back to OptionList
OptionList to_option_list() const
{
std::ostringstream os;
output_csv(os);
return OptionList::parse_from_csv_static(os.str(), nullptr);
}
void output_csv(std::ostream &os) const
{
for (auto &e : *this)
{
os << ',';
output_arg(e, os);
}
}
static void output_arg(const std::string &e, std::ostream &os)
{
const bool must_quote = (e.find_first_of(',') != std::string::npos);
Option::escape_string(os, e, must_quote);
}
};
} // namespace openvpn
#endif
|