/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2015, 2021 Vladimir Golovnev * * 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 Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. */ #pragma once #include #include #include #include #include #include #include template class Digest32 { public: using UnderlyingType = lt::digest32; Digest32() = default; Digest32(const Digest32 &other) = default; Digest32(Digest32 &&other) = default; Digest32(const UnderlyingType &nativeDigest) : m_dataPtr {new Data(nativeDigest)} { } static constexpr int length() { return UnderlyingType::size(); } bool isValid() const { return m_dataPtr->isValid(); } Digest32 &operator=(const Digest32 &other) = default; Digest32 &operator=(Digest32 &&other) = default; operator UnderlyingType() const { return m_dataPtr->nativeDigest(); } QString toString() const { return m_dataPtr->hashString(); } static Digest32 fromString(const QString &digestString) { return Digest32(QSharedDataPointer(new Data(digestString))); } private: class Data; explicit Digest32(QSharedDataPointer dataPtr) : m_dataPtr {dataPtr} { } QSharedDataPointer m_dataPtr {new Data}; }; template class Digest32::Data : public QSharedData { public: Data() = default; explicit Data(UnderlyingType nativeDigest) : m_isValid {true} , m_nativeDigest {nativeDigest} { } explicit Data(const QString &digestString) { if (digestString.size() != (length() * 2)) return; const QByteArray raw = QByteArray::fromHex(digestString.toLatin1()); if (raw.size() != length()) // QByteArray::fromHex() will skip over invalid characters return; m_isValid = true; m_hashString = digestString; m_nativeDigest.assign(raw.constData()); } bool isValid() const { return m_isValid; } UnderlyingType nativeDigest() const { return m_nativeDigest; } QString hashString() const { if (m_hashString.isEmpty() && isValid()) { const QByteArray raw = QByteArray::fromRawData(m_nativeDigest.data(), length()); m_hashString = QString::fromLatin1(raw.toHex()); } return m_hashString; } private: bool m_isValid = false; UnderlyingType m_nativeDigest; mutable QString m_hashString; }; template bool operator==(const Digest32 &left, const Digest32 &right) { return (static_cast::UnderlyingType>(left) == static_cast::UnderlyingType>(right)); } template bool operator!=(const Digest32 &left, const Digest32 &right) { return !(left == right); } template bool operator<(const Digest32 &left, const Digest32 &right) { return static_cast::UnderlyingType>(left) < static_cast::UnderlyingType>(right); } #if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)) template std::size_t qHash(const Digest32 &key, const std::size_t seed = 0) { return ::qHash(static_cast::UnderlyingType>(key), seed); } #else template uint qHash(const Digest32 &key, const uint seed = 0) { return static_cast((std::hash::UnderlyingType> {})(key)) ^ seed; } #endif