/** * Copyright 2015 Vineet Garg * * 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) version 3 or any later version * accepted by the membership of KDE e.V. (or its successor approved * by the membership of KDE e.V.), which shall act as a proxy * defined in Section 14 of version 3 of the license. * * 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, see . */ // This class tests the behaviour of the class LanLinkProvider, be sure to kill process kdeconnectd to avoid any port binding issues #include "../core/backends/lan/lanlinkprovider.h" #include "../core/backends/lan/server.h" #include "../core/backends/lan/socketlinereader.h" #include "../core/kdeconnectconfig.h" #include #include #include #include /* * This class tests the working of LanLinkProvider under different conditions that when identity package is received over TCP, over UDP and same when the device is paired. * It depends on KdeConnectConfig since LanLinkProvider internally uses it. */ class LanLinkProviderTest : public QObject { Q_OBJECT public Q_SLOTS: void initTestCase(); private Q_SLOTS: void pairedDeviceTcpPackageReceived(); void pairedDeviceUdpPackageReceived(); void unpairedDeviceTcpPackageReceived(); void unpairedDeviceUdpPackageReceived(); private: const int PORT = 8520; // Add some private fields here LanLinkProvider mLanLinkProvider; QEventLoop mLoop; Server* mServer; SocketLineReader* mReader; QUdpSocket* mUdpSocket; QString mIdentityPackage; // Attributes for test device QString deviceId; QString name; QCA::PrivateKey privateKey; QSslCertificate certificate; QSslCertificate generateCertificate(QString&, QCA::PrivateKey&); void addTrustedDevice(); void removeTrustedDevice(); void setSocketAttributes(QSslSocket *socket); void testIdentityPackage(QByteArray& identityPackage); }; void LanLinkProviderTest::initTestCase() { removeTrustedDevice(); // Remove trusted device if left by chance by any test deviceId = QString("testdevice"); name = QString("Test Device"); privateKey = QCA::KeyGenerator().createRSA(2048); certificate = generateCertificate(deviceId, privateKey); mLanLinkProvider.onStart(); mIdentityPackage = QString("{\"id\":1439365924847,\"type\":\"kdeconnect.identity\",\"body\":{\"deviceId\":\"testdevice\",\"deviceName\":\"Test Device\",\"protocolVersion\":6,\"deviceType\":\"phone\",\"tcpPort\":") + QString::number(PORT) + QString("}}"); } void LanLinkProviderTest::pairedDeviceTcpPackageReceived() { KdeConnectConfig* kcc = KdeConnectConfig::instance(); addTrustedDevice(); QUdpSocket* mUdpServer = new QUdpSocket; mUdpServer->bind(QHostAddress::Any, 1714, QUdpSocket::ShareAddress); connect(mUdpServer, SIGNAL(readyRead()), &mLoop, SLOT(quit())); mLanLinkProvider.onNetworkChange(); mLoop.exec(); disconnect(mUdpServer, SIGNAL(readyRead()), &mLoop, SLOT(quit())); // This avoids strange behaviour of mLoop due to incoming udp package QByteArray datagram; datagram.resize(mUdpServer->pendingDatagramSize()); QHostAddress sender; mUdpServer->readDatagram(datagram.data(), datagram.size(), &sender); testIdentityPackage(datagram); QJsonDocument jsonDocument = QJsonDocument::fromJson(datagram); QJsonObject body = jsonDocument.object().value("body").toObject(); int tcpPort = body.value("tcpPort").toInt(); QSslSocket socket; connect(&socket, SIGNAL(connected()), &mLoop, SLOT(quit())); connect(&socket, SIGNAL(error(QAbstractSocket::SocketError)), &mLoop, SLOT(quit())); socket.connectToHost(sender, tcpPort); mLoop.exec(); QVERIFY2(socket.isOpen(), "Socket disconnected immediately"); socket.write(mIdentityPackage.toLatin1()); socket.waitForBytesWritten(2000); connect(&socket, SIGNAL(encrypted()), &mLoop, SLOT(quit())); connect(&socket, SIGNAL(sslErrors(const QList&)), &mLoop, SLOT(quit())); setSocketAttributes(&socket); socket.addCaCertificate(kcc->certificate()); socket.setPeerVerifyMode(QSslSocket::VerifyPeer); socket.setPeerVerifyName(kcc->name()); socket.startServerEncryption(); mLoop.exec(); QCOMPARE(socket.sslErrors().size(), 0); QVERIFY2(socket.isValid(), "Server socket disconnected"); QVERIFY2(socket.isEncrypted(), "Server socket not yet encrypted"); QVERIFY2(!socket.peerCertificate().isNull(), "Peer certificate is null"); removeTrustedDevice(); delete mUdpServer; } void LanLinkProviderTest::pairedDeviceUdpPackageReceived() { KdeConnectConfig* kcc = KdeConnectConfig::instance(); addTrustedDevice(); mServer = new Server(this); mUdpSocket = new QUdpSocket(this); mServer->listen(QHostAddress::Any, PORT); connect(mServer, SIGNAL(newConnection()), &mLoop, SLOT(quit())); qint64 bytesWritten = mUdpSocket->writeDatagram(mIdentityPackage.toLatin1(), QHostAddress::LocalHost, 1714); // write an identity package to udp socket here, we do not broadcast it here QCOMPARE(bytesWritten, mIdentityPackage.size()); // We should have an incoming connection now, wait for incoming connection mLoop.exec(); QSslSocket* serverSocket = mServer->nextPendingConnection(); QVERIFY2(serverSocket != 0, "Server socket is null"); QVERIFY2(serverSocket->isOpen(), "Server socket already closed"); mReader = new SocketLineReader(serverSocket, this); connect(mReader, SIGNAL(readyRead()), &mLoop, SLOT(quit())); mLoop.exec(); QByteArray receivedPackage = mReader->readLine(); testIdentityPackage(receivedPackage); // Received identiy package from LanLinkProvider now start ssl connect(serverSocket, SIGNAL(encrypted()), &mLoop, SLOT(quit())); connect(serverSocket, SIGNAL(sslErrors(const QList&)), &mLoop, SLOT(quit())); setSocketAttributes(serverSocket); serverSocket->addCaCertificate(kcc->certificate()); serverSocket->setPeerVerifyMode(QSslSocket::VerifyPeer); serverSocket->setPeerVerifyName(kcc->deviceId()); serverSocket->startClientEncryption(); // Its TCP server. but SSL client mLoop.exec(); QCOMPARE(serverSocket->sslErrors().size(), 0); QVERIFY2(serverSocket->isValid(), "Server socket disconnected"); QVERIFY2(serverSocket->isEncrypted(), "Server socket not yet encrypted"); QVERIFY2(!serverSocket->peerCertificate().isNull(), "Peer certificate is null"); removeTrustedDevice(); delete mServer; delete mUdpSocket; } void LanLinkProviderTest::unpairedDeviceTcpPackageReceived() { QUdpSocket* mUdpServer = new QUdpSocket; mUdpServer->bind(QHostAddress::Any, 1714, QUdpSocket::ShareAddress); connect(mUdpServer, SIGNAL(readyRead()), &mLoop, SLOT(quit())); mLanLinkProvider.onNetworkChange(); mLoop.exec(); disconnect(mUdpServer, SIGNAL(readyRead()), &mLoop, SLOT(quit())); QByteArray datagram; datagram.resize(mUdpServer->pendingDatagramSize()); QHostAddress sender; mUdpServer->readDatagram(datagram.data(), datagram.size(), &sender); testIdentityPackage(datagram); QJsonDocument jsonDocument = QJsonDocument::fromJson(datagram); QJsonObject body = jsonDocument.object().value("body").toObject(); int tcpPort = body.value("tcpPort").toInt(); QSslSocket socket; connect(&socket, SIGNAL(connected()), &mLoop, SLOT(quit())); connect(&socket, SIGNAL(error(QAbstractSocket::SocketError)), &mLoop, SLOT(quit())); socket.connectToHost(sender, tcpPort); mLoop.exec(); QVERIFY2(socket.isOpen(), "Socket disconnected immediately"); socket.write(mIdentityPackage.toLatin1()); socket.waitForBytesWritten(2000); connect(&socket, SIGNAL(encrypted()), &mLoop, SLOT(quit())); // We don't take care for sslErrors signal here, but signal will emit still we will get successful connection setSocketAttributes(&socket); socket.setPeerVerifyMode(QSslSocket::QueryPeer); socket.startServerEncryption(); mLoop.exec(); QVERIFY2(socket.isValid(), "Server socket disconnected"); QVERIFY2(socket.isEncrypted(), "Server socket not yet encrypted"); QVERIFY2(!socket.peerCertificate().isNull(), "Peer certificate is null"); delete mUdpServer; } void LanLinkProviderTest::unpairedDeviceUdpPackageReceived() { mServer = new Server(this); mUdpSocket = new QUdpSocket(this); mServer->listen(QHostAddress::Any, PORT); connect(mServer, SIGNAL(newConnection()), &mLoop, SLOT(quit())); qint64 bytesWritten = mUdpSocket->writeDatagram(mIdentityPackage.toLatin1(), QHostAddress::LocalHost, 1714); // write an identity package to udp socket here, we do not broadcast it here QCOMPARE(bytesWritten, mIdentityPackage.size()); // We should have an incoming connection now, wait for incoming connection mLoop.exec(); QSslSocket* serverSocket = mServer->nextPendingConnection(); QVERIFY2(serverSocket != 0, "Server socket is null"); QVERIFY2(serverSocket->isOpen(), "Server socket already closed"); mReader = new SocketLineReader(serverSocket, this); connect(mReader, SIGNAL(readyRead()), &mLoop, SLOT(quit())); mLoop.exec(); QByteArray receivedPackage = mReader->readLine(); QVERIFY2(!receivedPackage.isEmpty(), "Empty package received"); testIdentityPackage(receivedPackage); // Received identity package from LanLinkProvider now start ssl connect(serverSocket, SIGNAL(encrypted()), &mLoop, SLOT(quit())); connect(serverSocket, SIGNAL(disconnected()), &mLoop, SLOT(quit())); setSocketAttributes(serverSocket); serverSocket->setPeerVerifyMode(QSslSocket::QueryPeer); serverSocket->startClientEncryption(); // Its TCP server. but SSL client mLoop.exec(); QVERIFY2(serverSocket->isValid(), "Server socket disconnected"); QVERIFY2(serverSocket->isEncrypted(), "Server socket not yet encrypted"); QVERIFY2(!serverSocket->peerCertificate().isNull(), "Peer certificate is null"); delete mServer; delete mUdpSocket; } void LanLinkProviderTest::testIdentityPackage(QByteArray &identityPackage) { QJsonDocument jsonDocument = QJsonDocument::fromJson(identityPackage); QJsonObject jsonObject = jsonDocument.object(); QJsonObject body = jsonObject.value("body").toObject(); QCOMPARE(jsonObject.value("type").toString(), QString("kdeconnect.identity")); QVERIFY2(body.contains("deviceName"), "Device name not found in identity package"); QVERIFY2(body.contains("deviceId"), "Device id not found in identity package"); QVERIFY2(body.contains("protocolVersion"), "Protocol version not found in identity package"); QVERIFY2(body.contains("deviceType"), "Device type not found in identity package"); } QSslCertificate LanLinkProviderTest::generateCertificate(QString& commonName, QCA::PrivateKey& privateKey) { QDateTime startTime = QDateTime::currentDateTime(); QDateTime endTime = startTime.addYears(10); QCA::CertificateInfo certificateInfo; certificateInfo.insert(QCA::CommonName,commonName); certificateInfo.insert(QCA::Organization,"KDE"); certificateInfo.insert(QCA::OrganizationalUnit,"Kde connect"); QCA::CertificateOptions certificateOptions(QCA::PKCS10); certificateOptions.setSerialNumber(10); certificateOptions.setInfo(certificateInfo); certificateOptions.setValidityPeriod(startTime, endTime); certificateOptions.setFormat(QCA::PKCS10); QSslCertificate certificate = QSslCertificate(QCA::Certificate(certificateOptions, privateKey).toPEM().toLatin1()); return certificate; } void LanLinkProviderTest::setSocketAttributes(QSslSocket *socket) { socket->setPrivateKey(QSslKey(privateKey.toPEM().toLatin1(), QSsl::Rsa)); socket->setLocalCertificate(certificate); } void LanLinkProviderTest::addTrustedDevice() { KdeConnectConfig *kcc = KdeConnectConfig::instance(); kcc->addTrustedDevice(deviceId, name, QString("phone")); kcc->setDeviceProperty(deviceId, QString("certificate"), QString::fromLatin1(certificate.toPem())); kcc->setDeviceProperty(deviceId, QString("publicKey"), privateKey.toPublicKey().toPEM()); } void LanLinkProviderTest::removeTrustedDevice() { KdeConnectConfig *kcc = KdeConnectConfig::instance(); kcc->removeTrustedDevice(deviceId); } QTEST_GUILESS_MAIN(LanLinkProviderTest) #include "lanlinkprovidertest.moc"