2013-08-13 23:09:46 +01:00
|
|
|
/**
|
2020-08-17 10:48:10 +01:00
|
|
|
* SPDX-FileCopyrightText: 2013 Albert Vaca <albertvaka@gmail.com>
|
2013-08-13 23:09:46 +01:00
|
|
|
*
|
2020-08-17 10:48:10 +01:00
|
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
2013-08-13 23:09:46 +01:00
|
|
|
*/
|
|
|
|
|
2013-11-23 00:39:10 +00:00
|
|
|
#include "shareplugin.h"
|
2013-08-13 23:09:46 +01:00
|
|
|
|
2014-09-22 00:25:15 +01:00
|
|
|
#include <QStandardPaths>
|
2015-03-10 04:59:36 +00:00
|
|
|
#include <QProcess>
|
2013-11-23 00:39:10 +00:00
|
|
|
#include <QDir>
|
2013-09-10 18:01:46 +01:00
|
|
|
#include <QDesktopServices>
|
2014-03-03 03:53:11 +00:00
|
|
|
#include <QDBusConnection>
|
2016-07-05 13:27:53 +01:00
|
|
|
#include <QTemporaryFile>
|
2019-05-11 20:52:00 +01:00
|
|
|
#include <QDateTime>
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2015-03-10 04:59:36 +00:00
|
|
|
#include <KLocalizedString>
|
|
|
|
#include <KJobTrackerInterface>
|
|
|
|
#include <KPluginFactory>
|
2019-12-09 22:14:19 +00:00
|
|
|
#include <KIO/Job>
|
2015-04-01 15:06:12 +01:00
|
|
|
#include <KIO/MkpathJob>
|
2020-08-12 00:33:56 +01:00
|
|
|
#include <KApplicationTrader>
|
2019-12-20 00:23:49 +00:00
|
|
|
#include <KFileUtils>
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2016-07-05 13:27:53 +01:00
|
|
|
#include "core/filetransferjob.h"
|
2019-07-17 22:06:58 +01:00
|
|
|
#include "core/daemon.h"
|
2020-05-26 17:55:47 +01:00
|
|
|
#include "plugin_share_debug.h"
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2019-06-12 21:16:54 +01:00
|
|
|
K_PLUGIN_CLASS_WITH_JSON(SharePlugin, "kdeconnect_share.json")
|
2013-08-13 23:09:46 +01:00
|
|
|
|
2013-11-23 00:39:10 +00:00
|
|
|
SharePlugin::SharePlugin(QObject* parent, const QVariantList& args)
|
2013-08-13 23:09:46 +01:00
|
|
|
: KdeConnectPlugin(parent, args)
|
2019-06-02 15:02:21 +01:00
|
|
|
, m_compositeJob()
|
2013-08-13 23:09:46 +01:00
|
|
|
{
|
2013-11-23 00:39:10 +00:00
|
|
|
}
|
|
|
|
|
2014-09-21 21:24:37 +01:00
|
|
|
QUrl SharePlugin::destinationDir() const
|
2013-11-23 00:39:10 +00:00
|
|
|
{
|
2015-04-15 16:45:54 +01:00
|
|
|
const QString defaultDownloadPath = QStandardPaths::writableLocation(QStandardPaths::DownloadLocation);
|
2019-10-27 17:08:51 +00:00
|
|
|
QUrl dir = QUrl::fromLocalFile(config()->getString(QStringLiteral("incoming_path"), defaultDownloadPath));
|
2013-11-23 00:39:10 +00:00
|
|
|
|
2016-11-26 14:38:08 +00:00
|
|
|
if (dir.path().contains(QLatin1String("%1"))) {
|
2015-04-01 15:06:12 +01:00
|
|
|
dir.setPath(dir.path().arg(device()->name()));
|
2014-09-21 21:24:37 +01:00
|
|
|
}
|
2013-11-23 00:39:10 +00:00
|
|
|
|
2015-04-01 15:06:12 +01:00
|
|
|
KJob* job = KIO::mkpath(dir);
|
|
|
|
bool ret = job->exec();
|
|
|
|
if (!ret) {
|
|
|
|
qWarning() << "couldn't create" << dir;
|
|
|
|
}
|
2013-11-23 00:39:10 +00:00
|
|
|
|
2015-04-01 15:06:12 +01:00
|
|
|
return dir;
|
2013-08-13 23:09:46 +01:00
|
|
|
}
|
|
|
|
|
2018-11-26 17:02:42 +00:00
|
|
|
QUrl SharePlugin::getFileDestination(const QString filename) const
|
|
|
|
{
|
|
|
|
const QUrl dir = destinationDir().adjusted(QUrl::StripTrailingSlash);
|
|
|
|
QUrl destination(dir);
|
2019-06-10 15:40:28 +01:00
|
|
|
destination.setPath(dir.path() + QStringLiteral("/") + filename, QUrl::DecodedMode);
|
2018-11-26 17:02:42 +00:00
|
|
|
if (destination.isLocalFile() && QFile::exists(destination.toLocalFile())) {
|
2019-12-20 00:23:49 +00:00
|
|
|
destination.setPath(dir.path() + QStringLiteral("/") + KFileUtils::suggestName(dir, filename), QUrl::DecodedMode);
|
2018-11-26 17:02:42 +00:00
|
|
|
}
|
|
|
|
return destination;
|
|
|
|
}
|
|
|
|
|
2017-10-02 12:21:41 +01:00
|
|
|
static QString cleanFilename(const QString &filename)
|
|
|
|
{
|
|
|
|
int idx = filename.lastIndexOf(QLatin1Char('/'));
|
|
|
|
return idx>=0 ? filename.mid(idx + 1) : filename;
|
|
|
|
}
|
|
|
|
|
2019-05-11 20:52:00 +01:00
|
|
|
void SharePlugin::setDateModified(const QUrl& destination, const qint64 timestamp)
|
|
|
|
{
|
|
|
|
QFile receivedFile(destination.toLocalFile());
|
2019-05-19 18:13:07 +01:00
|
|
|
if (!receivedFile.exists() || !receivedFile.open(QIODevice::ReadWrite | QIODevice::Text)) {
|
|
|
|
return;
|
2019-05-11 20:52:00 +01:00
|
|
|
}
|
|
|
|
receivedFile.setFileTime(QDateTime::fromMSecsSinceEpoch(timestamp), QFileDevice::FileTime(QFileDevice::FileModificationTime));
|
|
|
|
}
|
|
|
|
|
2022-03-03 21:41:18 +00:00
|
|
|
void SharePlugin::setDateCreated(const QUrl& destination, const qint64 timestamp)
|
|
|
|
{
|
|
|
|
QFile receivedFile(destination.toLocalFile());
|
|
|
|
if (!receivedFile.exists() || !receivedFile.open(QIODevice::ReadWrite | QIODevice::Text)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
receivedFile.setFileTime(QDateTime::fromMSecsSinceEpoch(timestamp), QFileDevice::FileTime(QFileDevice::FileBirthTime));
|
|
|
|
}
|
|
|
|
|
2018-03-04 19:48:51 +00:00
|
|
|
bool SharePlugin::receivePacket(const NetworkPacket& np)
|
2013-08-13 23:09:46 +01:00
|
|
|
{
|
2013-09-27 02:01:13 +01:00
|
|
|
/*
|
2014-10-10 19:47:35 +01:00
|
|
|
//TODO: Write a test like this
|
2018-03-04 19:48:51 +00:00
|
|
|
if (np.type() == PACKET_TYPE_PING) {
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2014-09-21 23:46:19 +01:00
|
|
|
qCDebug(KDECONNECT_PLUGIN_SHARE) << "sending file" << (QDesktopServices::storageLocation(QDesktopServices::HomeLocation) + "/.bashrc");
|
2013-09-16 14:21:22 +01:00
|
|
|
|
2018-03-04 19:48:51 +00:00
|
|
|
NetworkPacket out(PACKET_TYPE_SHARE_REQUEST);
|
2013-09-20 14:54:30 +01:00
|
|
|
out.set("filename", mDestinationDir + "itworks.txt");
|
2013-09-16 14:21:22 +01:00
|
|
|
AutoClosingQFile* file = new AutoClosingQFile(QDesktopServices::storageLocation(QDesktopServices::HomeLocation) + "/.bashrc"); //Test file to transfer
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2013-09-20 14:54:30 +01:00
|
|
|
out.setPayload(file, file->size());
|
2013-09-10 18:01:46 +01:00
|
|
|
|
2018-03-04 19:48:51 +00:00
|
|
|
device()->sendPacket(out);
|
2013-09-10 18:01:46 +01:00
|
|
|
|
|
|
|
return true;
|
2013-09-16 14:21:22 +01:00
|
|
|
|
2013-09-20 14:54:30 +01:00
|
|
|
}
|
2013-09-27 02:01:13 +01:00
|
|
|
*/
|
2013-10-01 01:44:49 +01:00
|
|
|
|
2014-09-21 23:46:19 +01:00
|
|
|
qCDebug(KDECONNECT_PLUGIN_SHARE) << "File transfer";
|
2013-09-16 14:21:22 +01:00
|
|
|
|
2018-11-26 17:02:42 +00:00
|
|
|
if (np.hasPayload() || np.has(QStringLiteral("filename"))) {
|
|
|
|
// qCDebug(KDECONNECT_PLUGIN_SHARE) << "receiving file" << filename << "in" << dir << "into" << destination;
|
2017-10-02 12:21:41 +01:00
|
|
|
const QString filename = cleanFilename(np.get<QString>(QStringLiteral("filename"), QString::number(QDateTime::currentMSecsSinceEpoch())));
|
2018-11-26 17:02:42 +00:00
|
|
|
QUrl destination = getFileDestination(filename);
|
2019-05-11 20:52:00 +01:00
|
|
|
|
2018-11-26 17:02:42 +00:00
|
|
|
if (np.hasPayload()) {
|
2022-03-03 21:41:18 +00:00
|
|
|
qint64 dateCreated = np.get<qint64>(QStringLiteral("creationTime"), QDateTime::currentMSecsSinceEpoch());
|
2019-05-11 20:52:00 +01:00
|
|
|
qint64 dateModified = np.get<qint64>(QStringLiteral("lastModified"), QDateTime::currentMSecsSinceEpoch());
|
2020-02-16 16:31:14 +00:00
|
|
|
const bool open = np.get<bool>(QStringLiteral("open"), false);
|
2020-08-12 00:33:56 +01:00
|
|
|
|
2019-06-02 15:02:21 +01:00
|
|
|
if (!m_compositeJob) {
|
|
|
|
m_compositeJob = new CompositeFileTransferJob(device()->id());
|
2021-01-31 21:22:36 +00:00
|
|
|
m_compositeJob->setProperty("destUrl", destinationDir().toString());
|
2021-11-16 12:28:33 +00:00
|
|
|
m_compositeJob->setProperty("immediateProgressReporting", true);
|
2019-06-02 15:02:21 +01:00
|
|
|
KIO::getJobTracker()->registerJob(m_compositeJob);
|
|
|
|
}
|
|
|
|
|
2018-11-26 17:02:42 +00:00
|
|
|
FileTransferJob* job = np.createPayloadTransferJob(destination);
|
2019-06-10 15:40:28 +01:00
|
|
|
job->setOriginName(device()->name() + QStringLiteral(": ") + filename);
|
2022-03-03 21:41:18 +00:00
|
|
|
connect(job, &KJob::result, this, [this, dateCreated, dateModified, open] (KJob* job) -> void {
|
|
|
|
finished(job, dateCreated, dateModified, open);
|
|
|
|
});
|
2019-06-02 15:02:21 +01:00
|
|
|
m_compositeJob->addSubjob(job);
|
|
|
|
|
|
|
|
if (!m_compositeJob->isRunning()) {
|
|
|
|
m_compositeJob->start();
|
|
|
|
}
|
2018-11-26 17:02:42 +00:00
|
|
|
} else {
|
|
|
|
QFile file(destination.toLocalFile());
|
|
|
|
file.open(QIODevice::WriteOnly);
|
|
|
|
file.close();
|
2015-04-02 13:48:50 +01:00
|
|
|
}
|
2016-11-26 14:38:08 +00:00
|
|
|
} else if (np.has(QStringLiteral("text"))) {
|
|
|
|
QString text = np.get<QString>(QStringLiteral("text"));
|
2019-02-08 23:00:01 +00:00
|
|
|
|
2020-08-12 00:33:56 +01:00
|
|
|
KService::Ptr service = KApplicationTrader::preferredService(QStringLiteral("text/plain"));
|
2019-02-08 23:00:01 +00:00
|
|
|
const QString defaultApp = service ? service->desktopEntryName() : QString();
|
|
|
|
|
|
|
|
if (defaultApp == QLatin1String("org.kde.kate") || defaultApp == QLatin1String("org.kde.kwrite")) {
|
2013-09-27 02:01:13 +01:00
|
|
|
QProcess* proc = new QProcess();
|
|
|
|
connect(proc, SIGNAL(finished(int)), proc, SLOT(deleteLater()));
|
2019-06-10 15:40:28 +01:00
|
|
|
proc->start(defaultApp.section(QStringLiteral("."), 2,2), QStringList(QStringLiteral("--stdin")));
|
2013-09-27 02:01:13 +01:00
|
|
|
proc->write(text.toUtf8());
|
|
|
|
proc->closeWriteChannel();
|
|
|
|
} else {
|
|
|
|
QTemporaryFile tmpFile;
|
2019-02-08 23:00:01 +00:00
|
|
|
tmpFile.setFileTemplate(QStringLiteral("kdeconnect-XXXXXX.txt"));
|
2013-09-27 02:01:13 +01:00
|
|
|
tmpFile.setAutoRemove(false);
|
|
|
|
tmpFile.open();
|
|
|
|
tmpFile.write(text.toUtf8());
|
|
|
|
tmpFile.close();
|
2015-12-07 02:27:40 +00:00
|
|
|
|
2017-08-01 22:57:50 +01:00
|
|
|
const QString fileName = tmpFile.fileName();
|
|
|
|
Q_EMIT shareReceived(fileName);
|
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(fileName));
|
2013-09-27 02:01:13 +01:00
|
|
|
}
|
2016-11-26 14:38:08 +00:00
|
|
|
} else if (np.has(QStringLiteral("url"))) {
|
|
|
|
QUrl url = QUrl::fromEncoded(np.get<QByteArray>(QStringLiteral("url")));
|
2013-09-27 02:01:13 +01:00
|
|
|
QDesktopServices::openUrl(url);
|
2017-08-01 22:57:50 +01:00
|
|
|
Q_EMIT shareReceived(url.toString());
|
2013-09-27 02:01:13 +01:00
|
|
|
} else {
|
2014-09-21 23:46:19 +01:00
|
|
|
qCDebug(KDECONNECT_PLUGIN_SHARE) << "Error: Nothing attached!";
|
2013-09-10 18:01:46 +01:00
|
|
|
}
|
2013-09-20 14:54:30 +01:00
|
|
|
|
2013-09-10 18:01:46 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-03-03 21:41:18 +00:00
|
|
|
void SharePlugin::finished(KJob* job, const qint64 dateModified, const qint64 dateCreated, const bool open)
|
2013-09-10 18:01:46 +01:00
|
|
|
{
|
2015-09-07 13:54:33 +01:00
|
|
|
FileTransferJob* ftjob = qobject_cast<FileTransferJob*>(job);
|
2017-08-01 22:57:50 +01:00
|
|
|
if (ftjob && !job->error()) {
|
|
|
|
Q_EMIT shareReceived(ftjob->destination().toString());
|
2022-03-03 21:41:18 +00:00
|
|
|
setDateCreated(ftjob->destination(), dateCreated);
|
2019-05-11 20:52:00 +01:00
|
|
|
setDateModified(ftjob->destination(), dateModified);
|
2017-08-01 22:57:50 +01:00
|
|
|
qCDebug(KDECONNECT_PLUGIN_SHARE) << "File transfer finished." << ftjob->destination();
|
2020-02-16 16:31:14 +00:00
|
|
|
if (open) {
|
|
|
|
QDesktopServices::openUrl(ftjob->destination());
|
|
|
|
}
|
2017-08-01 22:57:50 +01:00
|
|
|
} else {
|
|
|
|
qCDebug(KDECONNECT_PLUGIN_SHARE) << "File transfer failed." << (ftjob ? ftjob->destination() : QUrl());
|
|
|
|
}
|
2013-10-01 01:44:49 +01:00
|
|
|
}
|
2013-08-13 23:09:46 +01:00
|
|
|
|
2013-11-23 00:39:10 +00:00
|
|
|
void SharePlugin::openDestinationFolder()
|
2013-10-01 01:44:49 +01:00
|
|
|
{
|
2013-11-23 00:39:10 +00:00
|
|
|
QDesktopServices::openUrl(destinationDir());
|
2013-08-13 23:09:46 +01:00
|
|
|
}
|
2014-03-03 03:53:11 +00:00
|
|
|
|
2019-06-17 22:35:53 +01:00
|
|
|
void SharePlugin::shareUrl(const QUrl& url, bool open)
|
2014-03-03 03:53:11 +00:00
|
|
|
{
|
2018-03-04 19:48:51 +00:00
|
|
|
NetworkPacket packet(PACKET_TYPE_SHARE_REQUEST);
|
2019-06-17 22:35:53 +01:00
|
|
|
if (url.isLocalFile()) {
|
2019-07-17 22:06:58 +01:00
|
|
|
QSharedPointer<QFile> ioFile(new QFile(url.toLocalFile()));
|
2022-03-03 21:41:18 +00:00
|
|
|
QFileInfo info(*ioFile);
|
2019-07-17 22:06:58 +01:00
|
|
|
|
|
|
|
if (!ioFile->exists()) {
|
|
|
|
Daemon::instance()->reportError(i18n("Could not share file"), i18n("%1 does not exist", url.toLocalFile()));
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
packet.setPayload(ioFile, ioFile->size());
|
|
|
|
packet.set<QString>(QStringLiteral("filename"), QUrl(url).fileName());
|
2022-03-03 21:41:18 +00:00
|
|
|
packet.set<qint64>(QStringLiteral("creationTime"), info.birthTime().toMSecsSinceEpoch());
|
|
|
|
packet.set<qint64>(QStringLiteral("lastModified"), info.lastModified().toMSecsSinceEpoch());
|
2019-07-17 22:06:58 +01:00
|
|
|
packet.set<bool>(QStringLiteral("open"), open);
|
|
|
|
}
|
2014-03-03 03:53:11 +00:00
|
|
|
} else {
|
2018-03-04 19:48:51 +00:00
|
|
|
packet.set<QString>(QStringLiteral("url"), url.toString());
|
2014-03-03 03:53:11 +00:00
|
|
|
}
|
2018-03-04 19:48:51 +00:00
|
|
|
sendPacket(packet);
|
2014-03-03 03:53:11 +00:00
|
|
|
}
|
|
|
|
|
2018-11-28 18:56:22 +00:00
|
|
|
void SharePlugin::shareUrls(const QStringList& urls) {
|
2019-02-28 13:21:29 +00:00
|
|
|
for(const QString& url : urls) {
|
2019-06-17 22:35:53 +01:00
|
|
|
shareUrl(QUrl(url), false);
|
2018-11-28 18:56:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-03 21:16:07 +00:00
|
|
|
void SharePlugin::shareText(const QString& text)
|
|
|
|
{
|
|
|
|
NetworkPacket packet(PACKET_TYPE_SHARE_REQUEST);
|
|
|
|
packet.set<QString>(QStringLiteral("text"), text);
|
|
|
|
sendPacket(packet);
|
|
|
|
}
|
|
|
|
|
2014-03-03 03:53:11 +00:00
|
|
|
QString SharePlugin::dbusPath() const
|
|
|
|
{
|
2019-06-10 15:40:28 +01:00
|
|
|
return QStringLiteral("/modules/kdeconnect/devices/") + device()->id() + QStringLiteral("/share");
|
2014-03-03 03:53:11 +00:00
|
|
|
}
|
2014-06-16 19:02:07 +01:00
|
|
|
|
|
|
|
#include "shareplugin.moc"
|