mirror of
				https://github.com/Icinga/icinga2.git
				synced 2025-10-26 16:53:55 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			437 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			437 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /******************************************************************************
 | |
|  * Icinga 2                                                                   *
 | |
|  * Copyright (C) 2012-2017 Icinga Development Team (https://www.icinga.com/)  *
 | |
|  *                                                                            *
 | |
|  * This program is free software; you can redistribute it and/or              *
 | |
|  * modify it under the terms of the GNU General Public License                *
 | |
|  * as published by the Free Software Foundation; either version 2             *
 | |
|  * of the License, or (at your option) any later version.                     *
 | |
|  *                                                                            *
 | |
|  * 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 General Public License for more details.                               *
 | |
|  *                                                                            *
 | |
|  * You should have received a copy of the GNU General Public License          *
 | |
|  * along with this program; if not, write to the Free Software Foundation     *
 | |
|  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.             *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| #include "base/array.hpp"
 | |
| #include "base/utility.hpp"
 | |
| #include "base/objectlock.hpp"
 | |
| #include "remote/url.hpp"
 | |
| #include "remote/url-characters.hpp"
 | |
| #include <boost/tokenizer.hpp>
 | |
| 
 | |
| using namespace icinga;
 | |
| 
 | |
| Url::Url()
 | |
| { }
 | |
| 
 | |
| Url::Url(const String& base_url)
 | |
| {
 | |
| 	String url = base_url;
 | |
| 
 | |
| 	if (url.GetLength() == 0)
 | |
| 		BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Empty URL."));
 | |
| 
 | |
| 	size_t pHelper = url.Find(":");
 | |
| 
 | |
| 	if (pHelper != String::NPos) {
 | |
| 		if (!ParseScheme(url.SubStr(0, pHelper)))
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Scheme."));
 | |
| 		url = url.SubStr(pHelper + 1);
 | |
| 	}
 | |
| 
 | |
| 	if (*url.Begin() != '/')
 | |
| 		BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL: '/' expected after scheme."));
 | |
| 
 | |
| 	if (url.GetLength() == 1) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (*(url.Begin() + 1) == '/') {
 | |
| 		pHelper = url.Find("/", 2);
 | |
| 
 | |
| 		if (pHelper == String::NPos)
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL: Missing '/' after authority."));
 | |
| 
 | |
| 		if (!ParseAuthority(url.SubStr(0, pHelper)))
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Authority"));
 | |
| 
 | |
| 		url = url.SubStr(pHelper);
 | |
| 	}
 | |
| 
 | |
| 	if (*url.Begin() == '/') {
 | |
| 		pHelper = url.FindFirstOf("#?");
 | |
| 		if (!ParsePath(url.SubStr(1, pHelper - 1)))
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Path"));
 | |
| 
 | |
| 		if (pHelper != String::NPos)
 | |
| 			url = url.SubStr(pHelper);
 | |
| 	} else
 | |
| 		BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL: Missing path."));
 | |
| 
 | |
| 	if (*url.Begin() == '?') {
 | |
| 		pHelper = url.Find("#");
 | |
| 		if (!ParseQuery(url.SubStr(1, pHelper - 1)))
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Query"));
 | |
| 
 | |
| 		if (pHelper != String::NPos)
 | |
| 			url = url.SubStr(pHelper);
 | |
| 	}
 | |
| 
 | |
| 	if (*url.Begin() == '#') {
 | |
| 		if (!ParseFragment(url.SubStr(1)))
 | |
| 			BOOST_THROW_EXCEPTION(std::invalid_argument("Invalid URL Fragment"));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| String Url::GetScheme(void) const
 | |
| {
 | |
| 	return m_Scheme;
 | |
| }
 | |
| 
 | |
| String Url::GetAuthority(void) const
 | |
| {
 | |
| 	if (m_Host.IsEmpty())
 | |
| 		return "";
 | |
| 
 | |
| 	String auth;
 | |
| 	if (!m_Username.IsEmpty()) {
 | |
| 		auth = m_Username;
 | |
| 		if (!m_Password.IsEmpty())
 | |
| 			auth += ":" + m_Password;
 | |
| 		auth += "@";
 | |
| 	}
 | |
| 
 | |
| 	auth += m_Host;
 | |
| 
 | |
| 	if (!m_Port.IsEmpty())
 | |
| 		auth += ":" + m_Port;
 | |
| 
 | |
| 	return auth;
 | |
| }
 | |
| 
 | |
| String Url::GetUsername(void) const
 | |
| {
 | |
| 	return m_Username;
 | |
| }
 | |
| 
 | |
| String Url::GetPassword(void) const
 | |
| {
 | |
| 	return m_Password;
 | |
| }
 | |
| 
 | |
| String Url::GetHost(void) const
 | |
| {
 | |
| 	return m_Host;
 | |
| }
 | |
| 
 | |
| String Url::GetPort(void) const
 | |
| {
 | |
| 	return m_Port;
 | |
| }
 | |
| 
 | |
| const std::vector<String>& Url::GetPath(void) const
 | |
| {
 | |
| 	return m_Path;
 | |
| }
 | |
| 
 | |
| const std::map<String, std::vector<String> >& Url::GetQuery(void) const
 | |
| {
 | |
| 	return m_Query;
 | |
| }
 | |
| 
 | |
| String Url::GetQueryElement(const String& name) const
 | |
| {
 | |
| 	auto it = m_Query.find(name);
 | |
| 
 | |
| 	if (it == m_Query.end())
 | |
| 		return String();
 | |
| 
 | |
| 	return it->second.back();
 | |
| }
 | |
| 
 | |
| const std::vector<String>& Url::GetQueryElements(const String& name) const
 | |
| {
 | |
| 	auto it = m_Query.find(name);
 | |
| 
 | |
| 	if (it == m_Query.end()) {
 | |
| 		static std::vector<String> emptyVector;
 | |
| 		return emptyVector;
 | |
| 	}
 | |
| 
 | |
| 	return it->second;
 | |
| }
 | |
| 
 | |
| String Url::GetFragment(void) const
 | |
| {
 | |
| 	return m_Fragment;
 | |
| }
 | |
| 
 | |
| void Url::SetScheme(const String& scheme)
 | |
| {
 | |
| 	m_Scheme = scheme;
 | |
| }
 | |
| 
 | |
| void Url::SetUsername(const String& username)
 | |
| {
 | |
| 	m_Username = username;
 | |
| }
 | |
| 
 | |
| void Url::SetPassword(const String& password)
 | |
| {
 | |
| 	m_Password = password;
 | |
| }
 | |
| 
 | |
| void Url::SetHost(const String& host)
 | |
| {
 | |
| 	m_Host = host;
 | |
| }
 | |
| 
 | |
| void Url::SetPort(const String& port)
 | |
| {
 | |
| 	m_Port = port;
 | |
| }
 | |
| 
 | |
| void Url::SetPath(const std::vector<String>& path)
 | |
| {
 | |
| 	m_Path = path;
 | |
| }
 | |
| 
 | |
| void Url::SetQuery(const std::map<String, std::vector<String> >& query)
 | |
| {
 | |
| 	m_Query = query;
 | |
| }
 | |
| 
 | |
| void Url::AddQueryElement(const String& name, const String& value)
 | |
| {
 | |
| 	auto it = m_Query.find(name);
 | |
| 	if (it == m_Query.end()) {
 | |
| 		m_Query[name] = std::vector<String>();
 | |
| 		m_Query[name].push_back(value);
 | |
| 	} else
 | |
| 		m_Query[name].push_back(value);
 | |
| }
 | |
| 
 | |
| void Url::SetQueryElements(const String& name, const std::vector<String>& values)
 | |
| {
 | |
| 	m_Query[name] = values;
 | |
| }
 | |
| 
 | |
| void Url::SetFragment(const String& fragment) {
 | |
| 	m_Fragment = fragment;
 | |
| }
 | |
| 
 | |
| String Url::Format(bool print_credentials) const
 | |
| {
 | |
| 	String url;
 | |
| 
 | |
| 	if (!m_Scheme.IsEmpty())
 | |
| 		url += m_Scheme + ":";
 | |
| 
 | |
| 	if (print_credentials && !GetAuthority().IsEmpty())
 | |
| 		url += "//" + GetAuthority();
 | |
| 	else if (!GetHost().IsEmpty())
 | |
| 		url += "//" + GetHost() + (!GetPort().IsEmpty() ? ":" + GetPort() : "");
 | |
| 
 | |
| 	if (m_Path.empty())
 | |
| 		url += "/";
 | |
| 	else {
 | |
| 		for (const String& segment : m_Path) {
 | |
| 			url += "/";
 | |
| 			url += Utility::EscapeString(segment, ACPATHSEGMENT_ENCODE, false);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	String param;
 | |
| 	if (!m_Query.empty()) {
 | |
| 		typedef std::pair<String, std::vector<String> > kv_pair;
 | |
| 
 | |
| 		for (const kv_pair& kv : m_Query) {
 | |
| 			String key = Utility::EscapeString(kv.first, ACQUERY_ENCODE, false);
 | |
| 			if (param.IsEmpty())
 | |
| 				param = "?";
 | |
| 			else
 | |
| 				param += "&";
 | |
| 
 | |
| 			// Just one (or one empty) value
 | |
| 			if (kv.second.size() == 1) {
 | |
| 				param += key;
 | |
| 				param += kv.second[0].IsEmpty() ?
 | |
| 				    String() : "=" + Utility::EscapeString(kv.second[0], ACQUERY_ENCODE, false);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			// Array
 | |
| 			String temp;
 | |
| 			for (const String s : kv.second) {
 | |
| 				if (!temp.IsEmpty())
 | |
| 					temp += "&";
 | |
| 
 | |
| 				temp += key;
 | |
| 				if (kv.second.size() > 1)
 | |
| 					temp += "[]";
 | |
| 
 | |
| 				if (!s.IsEmpty())
 | |
| 					temp += "=" + Utility::EscapeString(s, ACQUERY_ENCODE, false);
 | |
| 			}
 | |
| 			param += temp;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	url += param;
 | |
| 
 | |
| 	if (!m_Fragment.IsEmpty())
 | |
| 		url += "#" + Utility::EscapeString(m_Fragment, ACFRAGMENT_ENCODE, false);
 | |
| 
 | |
| 	return url;
 | |
| }
 | |
| 
 | |
| bool Url::ParseScheme(const String& scheme)
 | |
| {
 | |
| 	m_Scheme = scheme;
 | |
| 
 | |
| 	if (scheme.FindFirstOf(ALPHA) != 0)
 | |
| 		return false;
 | |
| 
 | |
| 	return (ValidateToken(scheme, ACSCHEME));
 | |
| }
 | |
| 
 | |
| bool Url::ParseAuthority(const String& authority)
 | |
| {
 | |
| 	String auth = authority.SubStr(2);
 | |
| 	size_t pos = auth.Find("@");
 | |
| 	if (pos != String::NPos && pos != 0) {
 | |
| 		if (!Url::ParseUserinfo(auth.SubStr(0, pos)))
 | |
| 			return false;
 | |
| 		auth = auth.SubStr(pos+1);
 | |
| 	}
 | |
| 
 | |
| 	pos = auth.Find(":");
 | |
| 	if (pos != String::NPos) {
 | |
| 		if (pos == 0 || pos == auth.GetLength() - 1 || !Url::ParsePort(auth.SubStr(pos+1)))
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	m_Host = auth.SubStr(0, pos);
 | |
| 	return ValidateToken(m_Host, ACHOST);
 | |
| }
 | |
| 
 | |
| bool Url::ParseUserinfo(const String& userinfo)
 | |
| {
 | |
| 	size_t pos = userinfo.Find(":");
 | |
| 	m_Username = userinfo.SubStr(0, pos);
 | |
| 	if (!ValidateToken(m_Username, ACUSERINFO))
 | |
| 		return false;
 | |
| 	m_Username = Utility::UnescapeString(m_Username);
 | |
| 	if (pos != String::NPos && pos != userinfo.GetLength() - 1) {
 | |
| 		m_Password = userinfo.SubStr(pos+1);
 | |
| 		if (!ValidateToken(m_Username, ACUSERINFO))
 | |
| 			return false;
 | |
| 		m_Password = Utility::UnescapeString(m_Password);
 | |
| 	} else
 | |
| 		m_Password = "";
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool Url::ParsePort(const String& port)
 | |
| {
 | |
| 	m_Port = Utility::UnescapeString(port);
 | |
| 	if (!ValidateToken(m_Port, ACPORT))
 | |
| 		return false;
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool Url::ParsePath(const String& path)
 | |
| {
 | |
| 	std::string pathStr = path;
 | |
| 	boost::char_separator<char> sep("/");
 | |
| 	boost::tokenizer<boost::char_separator<char> > tokens(pathStr, sep);
 | |
| 
 | |
| 	for (const String& token : tokens) {
 | |
| 		if (token.IsEmpty())
 | |
| 			continue;
 | |
| 
 | |
| 		if (!ValidateToken(token, ACPATHSEGMENT))
 | |
| 			return false;
 | |
| 
 | |
| 		String decodedToken = Utility::UnescapeString(token);
 | |
| 
 | |
| 		m_Path.push_back(decodedToken);
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool Url::ParseQuery(const String& query)
 | |
| {
 | |
| 	/* Tokenizer does not like String AT ALL */
 | |
| 	std::string queryStr = query;
 | |
| 	boost::char_separator<char> sep("&");
 | |
| 	boost::tokenizer<boost::char_separator<char> > tokens(queryStr, sep);
 | |
| 
 | |
| 	for (const String& token : tokens) {
 | |
| 		size_t pHelper = token.Find("=");
 | |
| 
 | |
| 		if (pHelper == 0)
 | |
| 			// /?foo=bar&=bar == invalid
 | |
| 			return false;
 | |
| 
 | |
| 		String key = token.SubStr(0, pHelper);
 | |
| 		String value = Empty;
 | |
| 
 | |
| 		if (pHelper != String::NPos && pHelper != token.GetLength() - 1)
 | |
| 			value = token.SubStr(pHelper+1);
 | |
| 
 | |
| 		if (!ValidateToken(value, ACQUERY))
 | |
| 			return false;
 | |
| 
 | |
| 		value = Utility::UnescapeString(value);
 | |
| 
 | |
| 		pHelper = key.Find("[]");
 | |
| 
 | |
| 		if (pHelper == 0 || (pHelper != String::NPos && pHelper != key.GetLength()-2))
 | |
| 			return false;
 | |
| 
 | |
| 		key = key.SubStr(0, pHelper);
 | |
| 
 | |
| 		if (!ValidateToken(key, ACQUERY))
 | |
| 			return false;
 | |
| 
 | |
| 		key = Utility::UnescapeString(key);
 | |
| 
 | |
| 		auto it = m_Query.find(key);
 | |
| 
 | |
| 		if (it == m_Query.end()) {
 | |
| 			m_Query[key] = std::vector<String>();
 | |
| 			m_Query[key].push_back(value);
 | |
| 		} else
 | |
| 			m_Query[key].push_back(value);
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool Url::ParseFragment(const String& fragment)
 | |
| {
 | |
| 	m_Fragment = Utility::UnescapeString(fragment);
 | |
| 
 | |
| 	return ValidateToken(fragment, ACFRAGMENT);
 | |
| }
 | |
| 
 | |
| bool Url::ValidateToken(const String& token, const String& symbols)
 | |
| {
 | |
| 	for (const char ch : token) {
 | |
| 		if (symbols.FindFirstOf(ch) == String::NPos)
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 |