aboutsummaryrefslogtreecommitdiff
path: root/src/client/QXmppEncryptedHttpFileSharingProvider.cpp
blob: b6e82621d38398b8edcc1c5a55883dbbe99d2cac (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// SPDX-FileCopyrightText: 2022 Jonah Brüchert <jbb@kaidan.im>
//
// SPDX-License-Identifier: LGPL-2.1-or-later

#include "QXmppEncryptedHttpFileSharingProvider.h"

#include "QXmppClient.h"
#include "QXmppFileEncryption.h"
#include "QXmppHttpUploadManager.h"
#include "QXmppUpload.h"
#include "QXmppUtils.h"

#include "QcaInitializer_p.h"
#include <QMimeDatabase>

using namespace QXmpp;
using namespace QXmpp::Private;

class QXmppSfsEncryptedHttpUpload : public QXmppUpload
{
    Q_OBJECT

public:
    QXmppSfsEncryptedHttpUpload(std::shared_ptr<QXmppHttpUpload> &&httpUpload, const QByteArray &key, const QByteArray &iv)
        : inner(std::move(httpUpload))
    {
        connect(inner.get(), &QXmppHttpUpload::finished, this, [=](const QXmppHttpUpload::Result &result) {
            Q_EMIT uploadFinished(std::visit([=](auto &&value) -> QXmpp::Private::UploadResult {
                using T = std::decay_t<decltype(value)>;
                if constexpr (std::is_same_v<T, QUrl>) {
                    QXmppEncryptedFileSource encryptedSource;
                    encryptedSource.setKey(key);
                    encryptedSource.setIv(iv);
                    encryptedSource.setHttpSources({ QXmppHttpFileSource(value) });
                    return std::any(encryptedSource);
                } else if constexpr (std::is_same_v<T, Cancelled>) {
                    return Cancelled {};
                } else if constexpr (std::is_same_v<T, QXmppError>) {
                    return value;
                }
            },
                                             result));
        });
    }

    float progress() override { return inner->progress(); }
    void cancel() override { inner->cancel(); }
    bool isFinished() override { return inner->isFinished(); }
    quint64 bytesTransferred() override { return inner->bytesSent(); }
    quint64 bytesTotal() override { return inner->bytesTotal(); }

private:
    std::shared_ptr<QXmppHttpUpload> inner;
};

///
/// \class QXmppEncryptedHttpFileSharingProvider
///
/// Support for storing files encrypted on an HTTP server
///
/// \since QXmpp 1.5
///

class QXmppEncryptedHttpFileSharingProviderPrivate
{
public:
    QXmpp::Private::QcaInitializer init;
    QXmppHttpUploadManager *manager;
    QXmppHttpFileSharingProvider *httpProvider;
};

///
/// \brief Create a new QXmppEncryptedHttpFileSharingProvider
/// \param client
/// \param netManager QNetworkAccessManager that can be reused all over your application.
///
QXmppEncryptedHttpFileSharingProvider::QXmppEncryptedHttpFileSharingProvider(QXmppClient *client, QNetworkAccessManager *netManager)
    : d(std::make_unique<QXmppEncryptedHttpFileSharingProviderPrivate>())
{
    qRegisterMetaType<QXmpp::Private::UploadResult>();
    Q_ASSERT(client);
    d->manager = client->findExtension<QXmppHttpUploadManager>();
    Q_ASSERT(d->manager);
    d->httpProvider = new QXmppHttpFileSharingProvider(client, netManager);
}

QXmppEncryptedHttpFileSharingProvider::~QXmppEncryptedHttpFileSharingProvider() = default;

std::shared_ptr<QXmppDownload> QXmppEncryptedHttpFileSharingProvider::downloadFile(const std::any &source, std::unique_ptr<QIODevice> &&target)
{
    QXmppEncryptedFileSource encryptedSource;
    try {
        encryptedSource = std::any_cast<QXmppEncryptedFileSource>(source);
    } catch (const std::bad_any_cast &) {
        qFatal("QXmppEncryptedHttpFileSharingProvider::downloadFile can only handle QXmppEncryptedFileSource sources");
    }

    auto httpSource = encryptedSource.httpSources().front();
    auto output = std::make_unique<Encryption::DecryptionDevice>(std::move(target), encryptedSource.cipher(), encryptedSource.iv(), encryptedSource.key());
    return d->httpProvider->downloadFile(httpSource, std::move(output));
}

std::shared_ptr<QXmppUpload> QXmppEncryptedHttpFileSharingProvider::uploadFile(
    std::unique_ptr<QIODevice> data,
    const QXmppFileMetadata &info)
{
    auto cipher = Aes256CbcPkcs7;
    auto key = Encryption::generateKey(cipher);
    auto iv = Encryption::generateInitializationVector(cipher);

    auto encDevice = std::make_unique<Encryption::EncryptionDevice>(std::move(data), cipher, key, iv);
    auto encryptedSize = encDevice->size();

    auto upload = d->manager->uploadFile(
        std::move(encDevice),
        QXmppUtils::generateStanzaHash(10),
        QMimeDatabase().mimeTypeForName("application/octet-stream"),
        encryptedSize);

    return std::make_shared<QXmppSfsEncryptedHttpUpload>(std::move(upload), key, iv);
}

#include "QXmppEncryptedHttpFileSharingProvider.moc"