123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- //
- // HTTPDigestCredentials.cpp
- //
- // Library: Net
- // Package: HTTP
- // Module: HTTPDigestCredentials
- //
- // Copyright (c) 2011, Anton V. Yabchinskiy (arn at bestmx dot ru).
- // Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
- // and Contributors.
- //
- // SPDX-License-Identifier: BSL-1.0
- //
- #include "Poco/DateTime.h"
- #include "Poco/DateTimeFormat.h"
- #include "Poco/DateTimeFormatter.h"
- #include "Poco/Exception.h"
- #include "Poco/MD5Engine.h"
- #include "Poco/Net/HTTPDigestCredentials.h"
- #include "Poco/Net/HTTPRequest.h"
- #include "Poco/Net/HTTPResponse.h"
- #include "Poco/NumberFormatter.h"
- #include "Poco/StringTokenizer.h"
- namespace
- {
- std::string digest(Poco::DigestEngine& engine,
- const std::string& a,
- const std::string& b,
- const std::string& c = std::string(),
- const std::string& d = std::string(),
- const std::string& e = std::string(),
- const std::string& f = std::string())
- {
- engine.reset();
- engine.update(a);
- engine.update(':');
- engine.update(b);
- if (!c.empty())
- {
- engine.update(':');
- engine.update(c);
- if (!d.empty())
- {
- engine.update(':');
- engine.update(d);
- engine.update(':');
- engine.update(e);
- engine.update(':');
- engine.update(f);
- }
- }
- return Poco::DigestEngine::digestToHex(engine.digest());
- }
- std::string formatNonceCounter(int counter)
- {
- return Poco::NumberFormatter::formatHex(counter, 8);
- }
- }
- namespace Poco {
- namespace Net {
- const std::string HTTPDigestCredentials::SCHEME = "Digest";
- const std::string HTTPDigestCredentials::DEFAULT_ALGORITHM("MD5");
- const std::string HTTPDigestCredentials::DEFAULT_QOP("");
- const std::string HTTPDigestCredentials::NONCE_PARAM("nonce");
- const std::string HTTPDigestCredentials::REALM_PARAM("realm");
- const std::string HTTPDigestCredentials::QOP_PARAM("qop");
- const std::string HTTPDigestCredentials::ALGORITHM_PARAM("algorithm");
- const std::string HTTPDigestCredentials::USERNAME_PARAM("username");
- const std::string HTTPDigestCredentials::OPAQUE_PARAM("opaque");
- const std::string HTTPDigestCredentials::URI_PARAM("uri");
- const std::string HTTPDigestCredentials::RESPONSE_PARAM("response");
- const std::string HTTPDigestCredentials::AUTH_PARAM("auth");
- const std::string HTTPDigestCredentials::CNONCE_PARAM("cnonce");
- const std::string HTTPDigestCredentials::NC_PARAM("nc");
- int HTTPDigestCredentials::_nonceCounter(0);
- Poco::FastMutex HTTPDigestCredentials::_nonceMutex;
- HTTPDigestCredentials::HTTPDigestCredentials()
- {
- }
- HTTPDigestCredentials::HTTPDigestCredentials(const std::string& username, const std::string& password):
- _username(username),
- _password(password)
- {
- }
- HTTPDigestCredentials::~HTTPDigestCredentials()
- {
- }
- void HTTPDigestCredentials::reset()
- {
- _requestAuthParams.clear();
- _nc.clear();
- }
- void HTTPDigestCredentials::setUsername(const std::string& username)
- {
- _username = username;
- }
- void HTTPDigestCredentials::setPassword(const std::string& password)
- {
- _password = password;
- }
- void HTTPDigestCredentials::clear()
- {
- _username.clear();
- _password.clear();
- }
- void HTTPDigestCredentials::authenticate(HTTPRequest& request, const HTTPResponse& response)
- {
- authenticate(request, HTTPAuthenticationParams(response));
- }
- void HTTPDigestCredentials::authenticate(HTTPRequest& request, const HTTPAuthenticationParams& responseAuthParams)
- {
- createAuthParams(request, responseAuthParams);
- request.setCredentials(SCHEME, _requestAuthParams.toString());
- }
- void HTTPDigestCredentials::updateAuthInfo(HTTPRequest& request)
- {
- updateAuthParams(request);
- request.setCredentials(SCHEME, _requestAuthParams.toString());
- }
- void HTTPDigestCredentials::proxyAuthenticate(HTTPRequest& request, const HTTPResponse& response)
- {
- proxyAuthenticate(request, HTTPAuthenticationParams(response, HTTPAuthenticationParams::PROXY_AUTHENTICATE));
- }
- void HTTPDigestCredentials::proxyAuthenticate(HTTPRequest& request, const HTTPAuthenticationParams& responseAuthParams)
- {
- createAuthParams(request, responseAuthParams);
- request.setProxyCredentials(SCHEME, _requestAuthParams.toString());
- }
- void HTTPDigestCredentials::updateProxyAuthInfo(HTTPRequest& request)
- {
- updateAuthParams(request);
- request.setProxyCredentials(SCHEME, _requestAuthParams.toString());
- }
- std::string HTTPDigestCredentials::createNonce()
- {
- Poco::FastMutex::ScopedLock lock(_nonceMutex);
- MD5Engine md5;
- Timestamp::TimeVal now = Timestamp().epochMicroseconds();
- md5.update(&_nonceCounter, sizeof(_nonceCounter));
- md5.update(&now, sizeof(now));
- ++_nonceCounter;
- return DigestEngine::digestToHex(md5.digest());
- }
- void HTTPDigestCredentials::createAuthParams(const HTTPRequest& request, const HTTPAuthenticationParams& responseAuthParams)
- {
- // Not implemented: "domain" auth parameter and integrity protection.
- if (!responseAuthParams.has(NONCE_PARAM) || !responseAuthParams.has(REALM_PARAM))
- throw InvalidArgumentException("Invalid HTTP authentication parameters");
- const std::string& algorithm = responseAuthParams.get(ALGORITHM_PARAM, DEFAULT_ALGORITHM);
- if (icompare(algorithm, DEFAULT_ALGORITHM) != 0)
- throw NotImplementedException("Unsupported digest algorithm", algorithm);
- const std::string& nonce = responseAuthParams.get(NONCE_PARAM);
- const std::string& qop = responseAuthParams.get(QOP_PARAM, DEFAULT_QOP);
- const std::string& realm = responseAuthParams.getRealm();
- _requestAuthParams.clear();
- _requestAuthParams.set(USERNAME_PARAM, _username);
- _requestAuthParams.set(NONCE_PARAM, nonce);
- _requestAuthParams.setRealm(realm);
- if (responseAuthParams.has(OPAQUE_PARAM))
- {
- _requestAuthParams.set(OPAQUE_PARAM, responseAuthParams.get(OPAQUE_PARAM));
- }
- if (qop.empty())
- {
- updateAuthParams(request);
- }
- else
- {
- Poco::StringTokenizer tok(qop, ",", Poco::StringTokenizer::TOK_TRIM);
- bool qopSupported = false;
- for (Poco::StringTokenizer::Iterator it = tok.begin(); it != tok.end(); ++it)
- {
- if (icompare(*it, AUTH_PARAM) == 0)
- {
- qopSupported = true;
- _requestAuthParams.set(CNONCE_PARAM, createNonce());
- _requestAuthParams.set(QOP_PARAM, *it);
- updateAuthParams(request);
- break;
- }
- }
- if (!qopSupported)
- throw NotImplementedException("Unsupported QoP requested", qop);
- }
- }
- void HTTPDigestCredentials::updateAuthParams(const HTTPRequest& request)
- {
- MD5Engine engine;
- const std::string& qop = _requestAuthParams.get(QOP_PARAM, DEFAULT_QOP);
- const std::string& realm = _requestAuthParams.getRealm();
- const std::string& nonce = _requestAuthParams.get(NONCE_PARAM);
- _requestAuthParams.set(URI_PARAM, request.getURI());
- if (qop.empty())
- {
- const std::string ha1 = digest(engine, _username, realm, _password);
- const std::string ha2 = digest(engine, request.getMethod(), request.getURI());
- _requestAuthParams.set(RESPONSE_PARAM, digest(engine, ha1, nonce, ha2));
- }
- else if (icompare(qop, AUTH_PARAM) == 0)
- {
- const std::string& cnonce = _requestAuthParams.get(CNONCE_PARAM);
- const std::string ha1 = digest(engine, _username, realm, _password);
- const std::string ha2 = digest(engine, request.getMethod(), request.getURI());
- const std::string nc = formatNonceCounter(updateNonceCounter(nonce));
- _requestAuthParams.set(NC_PARAM, nc);
- _requestAuthParams.set(RESPONSE_PARAM, digest(engine, ha1, nonce, nc, cnonce, qop, ha2));
- }
- }
- bool HTTPDigestCredentials::verifyAuthInfo(const HTTPRequest& request) const
- {
- HTTPAuthenticationParams params(request);
- return verifyAuthParams(request, params);
- }
- bool HTTPDigestCredentials::verifyAuthParams(const HTTPRequest& request, const HTTPAuthenticationParams& params) const
- {
- const std::string& nonce = params.get(NONCE_PARAM);
- const std::string& realm = params.getRealm();
- const std::string& qop = params.get(QOP_PARAM, DEFAULT_QOP);
- std::string response;
- MD5Engine engine;
- if (qop.empty())
- {
- const std::string ha1 = digest(engine, _username, realm, _password);
- const std::string ha2 = digest(engine, request.getMethod(), request.getURI());
- response = digest(engine, ha1, nonce, ha2);
- }
- else if (icompare(qop, AUTH_PARAM) == 0)
- {
- const std::string& cnonce = params.get(CNONCE_PARAM);
- const std::string& nc = params.get(NC_PARAM);
- const std::string ha1 = digest(engine, _username, realm, _password);
- const std::string ha2 = digest(engine, request.getMethod(), request.getURI());
- response = digest(engine, ha1, nonce, nc, cnonce, qop, ha2);
- }
- return response == params.get(RESPONSE_PARAM);
- }
- int HTTPDigestCredentials::updateNonceCounter(const std::string& nonce)
- {
- NonceCounterMap::iterator iter = _nc.find(nonce);
- if (iter == _nc.end())
- {
- iter = _nc.insert(NonceCounterMap::value_type(nonce, 0)).first;
- }
- iter->second++;
- return iter->second;
- }
- } } // namespace Poco::Net
|