Newer
Older
XinYang_IOS / Carthage / Checkouts / OpenVPNAdapter / Sources / OpenVPN3 / openvpn / http / urlparm.hpp
@zhangfeng zhangfeng on 7 Dec 2023 5 KB 1.8.0
//    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-2020 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_;
    };

  }
}

#endif