0
0
mirror of https://github.com/OpenVPN/openvpn3.git synced 2024-09-20 12:12:15 +02:00
openvpn3/openvpn/http/urlparm.hpp
David Sommerseth dde1574596
Reformatting source code to new coding style
This is the result after running 'clang-format -i' on all C++ files and
headers, with the defined formatting rules in .clang-format.

Only the openvpn/common/unicode-impl.hpp has been excluded, as that is
mostly a copy of an external project.

Signed-off-by: David Sommerseth <davids@openvpn.net>
2023-01-18 19:24:15 +01:00

215 lines
5.9 KiB
C++

// 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-2022 OpenVPN Inc.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License Version 3
// as published by the Free Software Foundation.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program in the COPYING file.
// If not, see <http://www.gnu.org/licenses/>.
#ifndef OPENVPN_HTTP_URLPARM_H
#define OPENVPN_HTTP_URLPARM_H
#include <string>
#include <sstream>
#include <vector>
#include <openvpn/common/number.hpp>
#include <openvpn/http/urlencode.hpp>
#include <openvpn/http/webexcept.hpp>
#include <openvpn/common/string.hpp>
namespace openvpn {
namespace URL {
OPENVPN_EXCEPTION(url_parameter_error);
struct Parm
{
Parm()
{
}
Parm(const std::string &name_arg, const std::string &value_arg)
: name(name_arg), value(value_arg)
{
}
std::string to_string() const
{
std::ostringstream out;
out << name << '=' << value;
return out.str();
}
std::string name;
std::string value;
};
class ParmList : public std::vector<Parm>
{
public:
ParmList(const std::string &uri)
{
try
{
const std::vector<std::string> req_parms = string::split(uri, '?', 1);
request_ = req_parms[0];
if (req_parms.size() == 2)
{
const std::vector<std::string> kv_list = string::split(req_parms[1], '&');
for (auto &kvstr : kv_list)
{
const std::vector<std::string> kv = string::split(kvstr, '=', 1);
Parm p;
p.name = decode(kv[0]);
if (kv.size() == 2)
p.value = decode(kv[1]);
push_back(std::move(p));
}
}
}
catch (const std::exception &e)
{
throw HTTP::WebException(HTTP::Status::BadRequest, e.what());
}
}
const Parm *get(const std::string &key) const
{
for (auto &p : *this)
{
if (key == p.name)
return &p;
}
return nullptr;
}
std::string get_value(const std::string &key) const
{
const Parm *p = get(key);
if (p)
return p->value;
else
return "";
}
const std::string &get_value_required(const std::string &key) const
{
const Parm *p = get(key);
if (p)
return p->value;
else
throw url_parameter_error(key + " : not found");
}
template <typename T>
T get_num(const std::string &name, const std::string &short_name, const T default_value) const
{
const Parm *p = get(name);
if (!p && !short_name.empty())
p = get(short_name);
if (p)
return parse_number_throw<T>(p->value, name);
else
return default_value;
}
template <typename T>
T get_num_required(const std::string &name, const std::string &short_name) const
{
const Parm *p = get(name);
if (!p && !short_name.empty())
p = get(short_name);
if (!p)
throw url_parameter_error(name + " : not found");
return parse_number_throw<T>(p->value, name);
}
bool get_bool(const std::string &name, const std::string &short_name, const bool default_value) const
{
const Parm *p = get(name);
if (!p && !short_name.empty())
p = get(short_name);
if (p)
{
if (p->value == "0")
return false;
else if (p->value == "1")
return true;
else
throw url_parameter_error(name + ": parameter must be 0 or 1");
}
else
return default_value;
}
std::string get_string(const std::string &name, const std::string &short_name) const
{
const Parm *p = get(name);
if (!p && !short_name.empty())
p = get(short_name);
if (p)
return p->value;
else
return "";
}
std::string get_string_required(const std::string &name, const std::string &short_name) const
{
const Parm *p = get(name);
if (!p && !short_name.empty())
p = get(short_name);
if (!p)
throw url_parameter_error(name + " : not found");
return p->value;
}
std::string to_string() const
{
std::ostringstream out;
for (size_t i = 0; i < size(); ++i)
out << '[' << i << "] " << (*this)[i].to_string() << std::endl;
return out.str();
}
std::string request(const bool remove_leading_slash) const
{
std::string ret = request_;
if (remove_leading_slash)
{
if (ret.length() > 0 && ret[0] == '/')
ret = ret.substr(1);
else
throw HTTP::WebException(HTTP::Status::BadRequest, "URI missing leading slash");
}
if (ret.empty())
throw HTTP::WebException(HTTP::Status::BadRequest, "URI resource is empty");
return ret;
}
const std::string &request() const
{
return request_;
}
private:
std::string request_;
};
} // namespace URL
} // namespace openvpn
#endif