// Copyright (C) 2017 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only // Qt-Security score:critical reason:authorization-protocol #include "qoauth1signature.h" #include "qoauth1signature_p.h" #include #include #include #include QT_BEGIN_NAMESPACE Q_STATIC_LOGGING_CATEGORY(loggingCategory, "qt.networkauth.oauth1.signature") /*! \class QOAuth1Signature \inmodule QtNetworkAuth \ingroup oauth \brief Implements OAuth 1 signature methods. \since 5.8 OAuth-authenticated requests can have two sets of credentials: those passed via the "oauth_consumer_key" parameter and those in the "oauth_token" parameter. In order for the server to verify the authenticity of the request and prevent unauthorized access, the client needs to prove that it is the rightful owner of the credentials. This is accomplished using the shared-secret (or RSA key) part of each set of credentials. OAuth specifies three methods for the client to establish its rightful ownership of the credentials: "HMAC-SHA1", "RSA-SHA1", and "PLAINTEXT". Each generates a "signature" with which the request is "signed"; the first two use a digest of the data signed in generating this, though the last does not. The "RSA-SHA1" method is not supported here; it would use an RSA key rather than the shared-secret associated with the client credentials. */ /*! \enum QOAuth1Signature::HttpRequestMethod Indicates the HTTP request method. \value Head HEAD method. \value Get GET method. \value Put PUT method. \value Post POST method. \value Delete DELETE method. \value Custom Identifies a custom method. \value Unknown Method not set. */ static_assert(static_cast(QOAuth1Signature::HttpRequestMethod::Head) == static_cast(QNetworkAccessManager::HeadOperation) && static_cast(QOAuth1Signature::HttpRequestMethod::Get) == static_cast(QNetworkAccessManager::GetOperation) && static_cast(QOAuth1Signature::HttpRequestMethod::Put) == static_cast(QNetworkAccessManager::PutOperation) && static_cast(QOAuth1Signature::HttpRequestMethod::Post) == static_cast(QNetworkAccessManager::PostOperation) && static_cast(QOAuth1Signature::HttpRequestMethod::Delete) == static_cast(QNetworkAccessManager::DeleteOperation), "Invalid QOAuth1Signature::HttpRequestMethod enumeration values"); QOAuth1SignaturePrivate QOAuth1SignaturePrivate::shared_null; QOAuth1SignaturePrivate::QOAuth1SignaturePrivate(const QUrl &url, QOAuth1Signature::HttpRequestMethod method, const QMultiMap ¶meters, const QString &clientSharedKey, const QString &tokenSecret) : method(method), http://code.qt.io/url(url), clientSharedKey(clientSharedKey), tokenSecret(tokenSecret), parameters(parameters) {} QByteArray QOAuth1SignaturePrivate::signatureBaseString() const { // https://tools.ietf.org/html/rfc5849#section-3.4.1 QByteArray base; switch (method) { case QOAuth1Signature::HttpRequestMethod::Head: base.append("HEAD"); break; case QOAuth1Signature::HttpRequestMethod::Get: base.append("GET"); break; case QOAuth1Signature::HttpRequestMethod::Put: base.append("PUT"); break; case QOAuth1Signature::HttpRequestMethod::Post: base.append("POST"); break; case QOAuth1Signature::HttpRequestMethod::Delete: base.append("DELETE"); break; case QOAuth1Signature::HttpRequestMethod::Custom: if (!customVerb.isEmpty()) { base.append(customVerb); } else { qCCritical(loggingCategory, "QOAuth1Signature: HttpRequestMethod::Custom requires " "the verb to be set via setCustomMethodString"); } break; default: qCCritical(loggingCategory, "QOAuth1Signature: HttpRequestMethod not supported"); } base.append('&'); base.append(QUrl::toPercentEncoding(url.toString(QUrl::RemoveQuery)) + "&"); QMultiMap p = parameters; { // replace '+' with spaces now before decoding so that '%2B' gets left as '+' const QString query = url.query().replace(QLatin1Char('+'), QLatin1Char(' ')); const auto queryItems = QUrlQuery(query).queryItems(QUrl::FullyDecoded); for (auto it = queryItems.begin(), end = queryItems.end(); it != end; ++it) p.insert(it->first, it->second); } base.append(encodeHeaders(p)); return base; } QByteArray QOAuth1SignaturePrivate::secret() const { QByteArray secret; secret.append(QUrl::toPercentEncoding(clientSharedKey)); secret.append('&'); secret.append(QUrl::toPercentEncoding(tokenSecret)); return secret; } QByteArray QOAuth1SignaturePrivate::parameterString(const QMultiMap ¶meters) { QByteArray ret; auto previous = parameters.end(); for (auto it = parameters.begin(), end = parameters.end(); it != end; previous = it++) { if (previous != parameters.end()) { if (Q_UNLIKELY(previous.key() == it.key())) qCWarning(loggingCategory, "duplicated key %s", qPrintable(it.key())); ret.append("&"); } ret.append(QUrl::toPercentEncoding(it.key())); ret.append("="); ret.append(QUrl::toPercentEncoding(it.value().toString())); } return ret; } QByteArray QOAuth1SignaturePrivate::encodeHeaders(const QMultiMap &headers) { return QUrl::toPercentEncoding(QString::fromLatin1(parameterString(headers))); } /*! Creates a QOAuth1Signature using \list \li \a url as the target address \li \a method as the HTTP method used to send the request \li and the given user \a parameters to augment the request. \endlist */ QOAuth1Signature::QOAuth1Signature(const QUrl &url, QOAuth1Signature::HttpRequestMethod method, const QMultiMap ¶meters) : d(new QOAuth1SignaturePrivate(url, method, parameters)) {} /*! Creates a QOAuth1Signature using \list \li \a url as the target address \li \a clientSharedKey as the user token used to verify the signature \li \a tokenSecret as the negotiated token used to verify the signature \li \a method as the HTTP method used to send the request \li and the given user \a parameters to augment the request. \endlist */ QOAuth1Signature::QOAuth1Signature(const QUrl &url, const QString &clientSharedKey, const QString &tokenSecret, HttpRequestMethod method, const QMultiMap ¶meters) : d(new QOAuth1SignaturePrivate(url, method, parameters, clientSharedKey, tokenSecret)) {} /*! Creates a copy of \a other. */ QOAuth1Signature::QOAuth1Signature(const QOAuth1Signature &other) : d(other.d) {} /*! Move-constructs a QOAuth1Signature instance, taking over the private data \a other was using. */ QOAuth1Signature::QOAuth1Signature(QOAuth1Signature &&other) : d(std::move(other.d)) { } /*! Destroys the QOAuth1Signature. */ QOAuth1Signature::~QOAuth1Signature() {} /*! Returns the request method. */ QOAuth1Signature::HttpRequestMethod QOAuth1Signature::httpRequestMethod() const { return d->method; } /*! Sets the request \a method. */ void QOAuth1Signature::setHttpRequestMethod(QOAuth1Signature::HttpRequestMethod method) { d->method = method; } /*! \since 5.13 Returns the custom method string. \sa httpRequestMethod() */ QByteArray QOAuth1Signature::customMethodString() const { return d->customVerb; } /*! \since 5.13 Sets a custom request method. Will set the httpRequestMethod to QOAuth1Signature::HttpRequestMethod::Custom and store the \a verb to use it for the generation of the signature. \note Using this method is required when working with custom verbs. Setting only the request method will fail, as the signure needs to know the actual verb. \sa setHttpRequestMethod(), HttpRequestMethod */ void QOAuth1Signature::setCustomMethodString(const QByteArray &verb) { d->method = QOAuth1Signature::HttpRequestMethod::Custom; d->customVerb = verb; } /*! Returns the URL. */ QUrl QOAuth1Signature::url() const { return d->url; } /*! Sets the URL to \a url. */ void QOAuth1Signature::setUrl(const QUrl &url) { d->url = url; } /*! Returns the parameters. */ QMultiMap QOAuth1Signature::parameters() const { return d->parameters; } /*! Sets the \a parameters. */ void QOAuth1Signature::setParameters(const QMultiMap ¶meters) { d->parameters.clear(); for (auto it = parameters.cbegin(), end = parameters.cend(); it != end; ++it) d->parameters.insert(it.key(), it.value()); } /*! Adds the request \a body to the signature. When a POST request body contains arguments they should be included in the signed data. */ void QOAuth1Signature::addRequestBody(const QUrlQuery &body) { const auto list = body.queryItems(); for (auto it = list.begin(), end = list.end(); it != end; ++it) d->parameters.replace(it->first, it->second); } /*! Inserts a new pair \a key, \a value into the signature. When a POST request body contains arguments they should be included in the signed data. */ void QOAuth1Signature::insert(const QString &key, const QVariant &value) { d->parameters.replace(key, value); } /*! Retrieves the list of keys of parameters included in the signed data. */ QList QOAuth1Signature::keys() const { return d->parameters.uniqueKeys(); } /*! Removes \a key and any associated value from the signed data. */ QVariant QOAuth1Signature::take(const QString &key) { return d->parameters.take(key); } /*! Returns the value associated with \a key, if present in the signed data, otherwise \a defaultValue. */ QVariant QOAuth1Signature::value(const QString &key, const QVariant &defaultValue) const { return d->parameters.value(key, defaultValue); } /*! Returns the user secret used to generate the signature. */ QString QOAuth1Signature::clientSharedKey() const { return d->clientSharedKey; } /*! Sets \a secret as the user secret used to generate the signature. */ void QOAuth1Signature::setClientSharedKey(const QString &secret) { d->clientSharedKey = secret; } /*! Returns the negotiated secret used to generate the signature. */ QString QOAuth1Signature::tokenSecret() const { return d->tokenSecret; } /*! Sets \a secret as the negotiated secret used to generate the signature. */ void QOAuth1Signature::setTokenSecret(const QString &secret) { d->tokenSecret = secret; } /*! Generates the HMAC-SHA1 signature using the client shared secret and, where available, token secret. */ QByteArray QOAuth1Signature::hmacSha1() const { return QMessageAuthenticationCode::hash(d->signatureBaseString(), d->secret(), QCryptographicHash::Sha1); } /*! Generates the RSA-SHA1 signature. \note Currently this method is not supported. */ QByteArray QOAuth1Signature::rsaSha1() const { qCCritical(loggingCategory, "RSA-SHA1 signing method not supported"); return QByteArray(); } /*! Generates the PLAINTEXT signature. */ QByteArray QOAuth1Signature::plainText() const { return plainText(d->clientSharedKey, d->tokenSecret); } /*! Generates a PLAINTEXT signature from the client secret \a clientSharedKey and the token secret \a tokenSecret. */ QByteArray QOAuth1Signature::plainText(const QString &clientSharedKey, const QString &tokenSecret) { QByteArray ret; ret += clientSharedKey.toUtf8() + '&' + tokenSecret.toUtf8(); return ret; } /*! Swaps signature \a other with this signature. This operation is very fast and never fails. */ void QOAuth1Signature::swap(QOAuth1Signature &other) { qSwap(d, other.d); } QOAuth1Signature &QOAuth1Signature::operator=(const QOAuth1Signature &other) { if (d != other.d) { QOAuth1Signature tmp(other); tmp.swap(*this); } return *this; } /*! Move-assigns \a other to this signature and returns a reference to this QOAuth1Signature. */ QOAuth1Signature &QOAuth1Signature::operator=(QOAuth1Signature &&other) { QOAuth1Signature moved(std::move(other)); swap(moved); return *this; } QT_END_NAMESPACE