| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734 |
- /*
- * Copyright (C) by Dominik Schmidt <dev@dominik-schmidt.de>
- * Copyright (C) by Klaas Freitag <freitag@owncloud.com>
- * Copyright (C) by Roeland Jago Douma <roeland@famdouma.nl>
- *
- * 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.
- */
- #include "socketapi.h"
- #include "socketapi_p.h"
- #include "socketapi/socketuploadjob.h"
- #include "conflictdialog.h"
- #include "conflictsolver.h"
- #include "config.h"
- #include "configfile.h"
- #include "deletejob.h"
- #include "folderman.h"
- #include "folder.h"
- #include "encryptfolderjob.h"
- #include "theme.h"
- #include "common/syncjournalfilerecord.h"
- #include "syncengine.h"
- #include "syncfileitem.h"
- #include "filesystem.h"
- #include "version.h"
- #include "account.h"
- #include "accountstate.h"
- #include "account.h"
- #include "accountmanager.h"
- #include "capabilities.h"
- #include "common/asserts.h"
- #include "guiutility.h"
- #ifndef OWNCLOUD_TEST
- #include "sharemanager.h"
- #endif
- #include <array>
- #include <QBitArray>
- #include <QUrl>
- #include <QMetaMethod>
- #include <QMetaObject>
- #include <QStringList>
- #include <QScopedPointer>
- #include <QFile>
- #include <QDir>
- #include <QApplication>
- #include <QLocalSocket>
- #include <QStringBuilder>
- #include <QMessageBox>
- #include <QInputDialog>
- #include <QFileDialog>
- #include <QAction>
- #include <QJsonArray>
- #include <QJsonDocument>
- #include <QJsonObject>
- #include <QWidget>
- #include <QClipboard>
- #include <QDesktopServices>
- #include <QProcess>
- #include <QStandardPaths>
- #ifdef Q_OS_MAC
- #include <CoreFoundation/CoreFoundation.h>
- #endif
- // This is the version that is returned when the client asks for the VERSION.
- // The first number should be changed if there is an incompatible change that breaks old clients.
- // The second number should be changed when there are new features.
- #define MIRALL_SOCKET_API_VERSION "1.1"
- namespace {
- constexpr auto encryptJobPropertyFolder = "folder";
- constexpr auto encryptJobPropertyPath = "path";
- }
- namespace {
- const QLatin1Char RecordSeparator()
- {
- return QLatin1Char('\x1e');
- }
- QStringList split(const QString &data)
- {
- // TODO: string ref?
- return data.split(RecordSeparator());
- }
- #if GUI_TESTING
- using namespace OCC;
- QList<QObject *> allObjects(const QList<QWidget *> &widgets)
- {
- QList<QObject *> objects;
- std::copy(widgets.constBegin(), widgets.constEnd(), std::back_inserter(objects));
- objects << qApp;
- return objects;
- }
- QObject *findWidget(const QString &queryString, const QList<QWidget *> &widgets = QApplication::allWidgets())
- {
- auto objects = allObjects(widgets);
- QList<QObject *>::const_iterator foundWidget;
- if (queryString.contains('>')) {
- qCDebug(lcSocketApi) << "queryString contains >";
- auto subQueries = queryString.split('>', QString::SkipEmptyParts);
- Q_ASSERT(subQueries.count() == 2);
- auto parentQueryString = subQueries[0].trimmed();
- qCDebug(lcSocketApi) << "Find parent: " << parentQueryString;
- auto parent = findWidget(parentQueryString);
- if (!parent) {
- return nullptr;
- }
- auto childQueryString = subQueries[1].trimmed();
- auto child = findWidget(childQueryString, parent->findChildren<QWidget *>());
- qCDebug(lcSocketApi) << "found child: " << !!child;
- return child;
- } else if (queryString.startsWith('#')) {
- auto objectName = queryString.mid(1);
- qCDebug(lcSocketApi) << "find objectName: " << objectName;
- foundWidget = std::find_if(objects.constBegin(), objects.constEnd(), [&](QObject *widget) {
- return widget->objectName() == objectName;
- });
- } else {
- QList<QObject *> matches;
- std::copy_if(objects.constBegin(), objects.constEnd(), std::back_inserter(matches), [&](QObject *widget) {
- return widget->inherits(queryString.toLatin1());
- });
- std::for_each(matches.constBegin(), matches.constEnd(), [](QObject *w) {
- if (!w)
- return;
- qCDebug(lcSocketApi) << "WIDGET: " << w->objectName() << w->metaObject()->className();
- });
- if (matches.empty()) {
- return nullptr;
- }
- return matches[0];
- }
- if (foundWidget == objects.constEnd()) {
- return nullptr;
- }
- return *foundWidget;
- }
- #endif
- static inline QString removeTrailingSlash(QString path)
- {
- Q_ASSERT(path.endsWith(QLatin1Char('/')));
- path.truncate(path.length() - 1);
- return path;
- }
- static QString buildMessage(const QString &verb, const QString &path, const QString &status = QString())
- {
- QString msg(verb);
- if (!status.isEmpty()) {
- msg.append(QLatin1Char(':'));
- msg.append(status);
- }
- if (!path.isEmpty()) {
- msg.append(QLatin1Char(':'));
- QFileInfo fi(path);
- msg.append(QDir::toNativeSeparators(fi.absoluteFilePath()));
- }
- return msg;
- }
- }
- namespace OCC {
- Q_LOGGING_CATEGORY(lcSocketApi, "nextcloud.gui.socketapi", QtInfoMsg)
- Q_LOGGING_CATEGORY(lcPublicLink, "nextcloud.gui.socketapi.publiclink", QtInfoMsg)
- void SocketListener::sendMessage(const QString &message, bool doWait) const
- {
- if (!socket) {
- qCWarning(lcSocketApi) << "Not sending message to dead socket:" << message;
- return;
- }
- qCDebug(lcSocketApi) << "Sending SocketAPI message -->" << message << "to" << socket;
- QString localMessage = message;
- if (!localMessage.endsWith(QLatin1Char('\n'))) {
- localMessage.append(QLatin1Char('\n'));
- }
- QByteArray bytesToSend = localMessage.toUtf8();
- qint64 sent = socket->write(bytesToSend);
- if (doWait) {
- socket->waitForBytesWritten(1000);
- }
- if (sent != bytesToSend.length()) {
- qCWarning(lcSocketApi) << "Could not send all data on socket for " << localMessage;
- }
- }
- SocketApi::SocketApi(QObject *parent)
- : QObject(parent)
- {
- QString socketPath;
- qRegisterMetaType<SocketListener *>("SocketListener*");
- qRegisterMetaType<QSharedPointer<SocketApiJob>>("QSharedPointer<SocketApiJob>");
- qRegisterMetaType<QSharedPointer<SocketApiJobV2>>("QSharedPointer<SocketApiJobV2>");
- if (Utility::isWindows()) {
- socketPath = QLatin1String(R"(\\.\pipe\)")
- + QLatin1String(APPLICATION_EXECUTABLE)
- + QLatin1String("-")
- + QString::fromLocal8Bit(qgetenv("USERNAME"));
- // TODO: once the windows extension supports multiple
- // client connections, switch back to the theme name
- // See issue #2388
- // + Theme::instance()->appName();
- } else if (Utility::isMac()) {
- #ifdef Q_OS_MACOS
- socketPath = socketApiSocketPath();
- CFURLRef url = (CFURLRef)CFAutorelease((CFURLRef)CFBundleCopyBundleURL(CFBundleGetMainBundle()));
- QString bundlePath = QUrl::fromCFURL(url).path();
- auto _system = [](const QString &cmd, const QStringList &args) {
- QProcess process;
- process.setProcessChannelMode(QProcess::MergedChannels);
- process.start(cmd, args);
- if (!process.waitForFinished()) {
- qCWarning(lcSocketApi) << "Failed to load shell extension:" << cmd << args.join(" ") << process.errorString();
- } else {
- qCInfo(lcSocketApi) << (process.exitCode() != 0 ? "Failed to load" : "Loaded") << "shell extension:" << cmd << args.join(" ") << process.readAll();
- }
- };
- // Add it again. This was needed for Mojave to trigger a load.
- _system(QStringLiteral("pluginkit"), { QStringLiteral("-a"), QStringLiteral("%1Contents/PlugIns/FinderSyncExt.appex/").arg(bundlePath) });
- // Tell Finder to use the Extension (checking it from System Preferences -> Extensions)
- _system(QStringLiteral("pluginkit"), { QStringLiteral("-e"), QStringLiteral("use"), QStringLiteral("-i"), QStringLiteral(APPLICATION_REV_DOMAIN ".FinderSyncExt") });
- #endif
- } else if (Utility::isLinux() || Utility::isBSD()) {
- QString runtimeDir;
- runtimeDir = QStandardPaths::writableLocation(QStandardPaths::RuntimeLocation);
- socketPath = runtimeDir + "/" + Theme::instance()->appName() + "/socket";
- } else {
- qCWarning(lcSocketApi) << "An unexpected system detected, this probably won't work.";
- }
- QLocalServer::removeServer(socketPath);
- // Create the socket path:
- if (!Utility::isMac()) {
- // Not on macOS: there the directory is there, and created for us by the sandboxing
- // environment, because we belong to an App Group.
- QFileInfo info(socketPath);
- if (!info.dir().exists()) {
- bool result = info.dir().mkpath(".");
- qCDebug(lcSocketApi) << "creating" << info.dir().path() << result;
- if (result) {
- QFile::setPermissions(socketPath,
- QFile::Permissions(QFile::ReadOwner + QFile::WriteOwner + QFile::ExeOwner));
- }
- }
- }
- if (!_localServer.listen(socketPath)) {
- qCWarning(lcSocketApi) << "can't start server" << socketPath;
- } else {
- qCInfo(lcSocketApi) << "server started, listening at " << socketPath;
- }
- connect(&_localServer, &QLocalServer::newConnection, this, &SocketApi::slotNewConnection);
- // folder watcher
- connect(FolderMan::instance(), &FolderMan::folderSyncStateChange, this, &SocketApi::slotUpdateFolderView);
- }
- SocketApi::~SocketApi()
- {
- qCDebug(lcSocketApi) << "dtor";
- _localServer.close();
- // All remaining sockets will be destroyed with _localServer, their parent
- ASSERT(_listeners.isEmpty() || _listeners.first()->socket->parent() == &_localServer)
- _listeners.clear();
- }
- void SocketApi::slotNewConnection()
- {
- QIODevice *socket = _localServer.nextPendingConnection();
- if (!socket) {
- return;
- }
- qCInfo(lcSocketApi) << "New connection" << socket;
- connect(socket, &QIODevice::readyRead, this, &SocketApi::slotReadSocket);
- connect(socket, SIGNAL(disconnected()), this, SLOT(onLostConnection()));
- connect(socket, &QObject::destroyed, this, &SocketApi::slotSocketDestroyed);
- ASSERT(socket->readAll().isEmpty());
- auto listener = QSharedPointer<SocketListener>::create(socket);
- _listeners.insert(socket, listener);
- for (Folder *f : FolderMan::instance()->map()) {
- if (f->canSync()) {
- QString message = buildRegisterPathMessage(removeTrailingSlash(f->path()));
- qCInfo(lcSocketApi) << "Trying to send SocketAPI Register Path Message -->" << message << "to" << listener->socket;
- listener->sendMessage(message);
- }
- }
- }
- void SocketApi::onLostConnection()
- {
- qCInfo(lcSocketApi) << "Lost connection " << sender();
- sender()->deleteLater();
- auto socket = qobject_cast<QIODevice *>(sender());
- ASSERT(socket);
- _listeners.remove(socket);
- }
- void SocketApi::slotSocketDestroyed(QObject *obj)
- {
- auto *socket = dynamic_cast<QIODevice *>(obj);
- _listeners.remove(socket);
- }
- void SocketApi::slotReadSocket()
- {
- auto *socket = qobject_cast<QIODevice *>(sender());
- ASSERT(socket);
- // Find the SocketListener
- //
- // It's possible for the disconnected() signal to be triggered before
- // the readyRead() signals are received - in that case there won't be a
- // valid listener. We execute the handler anyway, but it will work with
- // a SocketListener that doesn't send any messages.
- static auto invalidListener = QSharedPointer<SocketListener>::create(nullptr);
- const auto listener = _listeners.value(socket, invalidListener);
- while (socket->canReadLine()) {
- // Make sure to normalize the input from the socket to
- // make sure that the path will match, especially on OS X.
- const QString line = QString::fromUtf8(socket->readLine().trimmed()).normalized(QString::NormalizationForm_C);
- qCInfo(lcSocketApi) << "Received SocketAPI message <--" << line << "from" << socket;
- const int argPos = line.indexOf(QLatin1Char(':'));
- const QByteArray command = line.midRef(0, argPos).toUtf8().toUpper();
- const int indexOfMethod = [&] {
- QByteArray functionWithArguments = QByteArrayLiteral("command_");
- if (command.startsWith("ASYNC_")) {
- functionWithArguments += command + QByteArrayLiteral("(QSharedPointer<SocketApiJob>)");
- } else if (command.startsWith("V2/")) {
- functionWithArguments += QByteArrayLiteral("V2_") + command.mid(3) + QByteArrayLiteral("(QSharedPointer<SocketApiJobV2>)");
- } else {
- functionWithArguments += command + QByteArrayLiteral("(QString,SocketListener*)");
- }
- Q_ASSERT(staticQtMetaObject.normalizedSignature(functionWithArguments) == functionWithArguments);
- const auto out = staticMetaObject.indexOfMethod(functionWithArguments);
- if (out == -1) {
- listener->sendError(QStringLiteral("Function %1 not found").arg(QString::fromUtf8(functionWithArguments)));
- }
- ASSERT(out != -1)
- return out;
- }();
- const auto argument = argPos != -1 ? line.midRef(argPos + 1) : QStringRef();
- if (command.startsWith("ASYNC_")) {
- auto arguments = argument.split('|');
- if (arguments.size() != 2) {
- listener->sendError(QStringLiteral("argument count is wrong"));
- return;
- }
- auto json = QJsonDocument::fromJson(arguments[1].toUtf8()).object();
- auto jobId = arguments[0];
- auto socketApiJob = QSharedPointer<SocketApiJob>(
- new SocketApiJob(jobId.toString(), listener, json), &QObject::deleteLater);
- if (indexOfMethod != -1) {
- staticMetaObject.method(indexOfMethod)
- .invoke(this, Qt::QueuedConnection,
- Q_ARG(QSharedPointer<SocketApiJob>, socketApiJob));
- } else {
- qCWarning(lcSocketApi) << "The command is not supported by this version of the client:" << command
- << "with argument:" << argument;
- socketApiJob->reject(QStringLiteral("command not found"));
- }
- } else if (command.startsWith("V2/")) {
- QJsonParseError error{};
- const auto json = QJsonDocument::fromJson(argument.toUtf8(), &error).object();
- if (error.error != QJsonParseError::NoError) {
- qCWarning(lcSocketApi()) << "Invalid json" << argument.toString() << error.errorString();
- listener->sendError(error.errorString());
- return;
- }
- auto socketApiJob = QSharedPointer<SocketApiJobV2>::create(listener, command, json);
- if (indexOfMethod != -1) {
- staticMetaObject.method(indexOfMethod)
- .invoke(this, Qt::QueuedConnection,
- Q_ARG(QSharedPointer<SocketApiJobV2>, socketApiJob));
- } else {
- qCWarning(lcSocketApi) << "The command is not supported by this version of the client:" << command
- << "with argument:" << argument;
- socketApiJob->failure(QStringLiteral("command not found"));
- }
- } else {
- if (indexOfMethod != -1) {
- // to ensure that listener is still valid we need to call it with Qt::DirectConnection
- ASSERT(thread() == QThread::currentThread())
- staticMetaObject.method(indexOfMethod)
- .invoke(this, Qt::DirectConnection, Q_ARG(QString, argument.toString()),
- Q_ARG(SocketListener *, listener.data()));
- }
- }
- }
- }
- void SocketApi::slotRegisterPath(const QString &alias)
- {
- // Make sure not to register twice to each connected client
- if (_registeredAliases.contains(alias))
- return;
- Folder *f = FolderMan::instance()->folder(alias);
- if (f) {
- const QString message = buildRegisterPathMessage(removeTrailingSlash(f->path()));
- for (const auto &listener : qAsConst(_listeners)) {
- qCInfo(lcSocketApi) << "Trying to send SocketAPI Register Path Message -->" << message << "to" << listener->socket;
- listener->sendMessage(message);
- }
- }
- _registeredAliases.insert(alias);
- }
- void SocketApi::slotUnregisterPath(const QString &alias)
- {
- if (!_registeredAliases.contains(alias))
- return;
- Folder *f = FolderMan::instance()->folder(alias);
- if (f)
- broadcastMessage(buildMessage(QLatin1String("UNREGISTER_PATH"), removeTrailingSlash(f->path()), QString()), true);
- _registeredAliases.remove(alias);
- }
- void SocketApi::slotUpdateFolderView(Folder *f)
- {
- if (_listeners.isEmpty()) {
- return;
- }
- if (f) {
- // do only send UPDATE_VIEW for a couple of status
- if (f->syncResult().status() == SyncResult::SyncPrepare
- || f->syncResult().status() == SyncResult::Success
- || f->syncResult().status() == SyncResult::Paused
- || f->syncResult().status() == SyncResult::Problem
- || f->syncResult().status() == SyncResult::Error
- || f->syncResult().status() == SyncResult::SetupError) {
- QString rootPath = removeTrailingSlash(f->path());
- broadcastStatusPushMessage(rootPath, f->syncEngine().syncFileStatusTracker().fileStatus(""));
- broadcastMessage(buildMessage(QLatin1String("UPDATE_VIEW"), rootPath));
- } else {
- qCDebug(lcSocketApi) << "Not sending UPDATE_VIEW for" << f->alias() << "because status() is" << f->syncResult().status();
- }
- }
- }
- void SocketApi::broadcastMessage(const QString &msg, bool doWait)
- {
- for (const auto &listener : qAsConst(_listeners)) {
- listener->sendMessage(msg, doWait);
- }
- }
- void SocketApi::processFileActivityRequest(const QString &localFile)
- {
- const auto fileData = FileData::get(localFile);
- emit fileActivityCommandReceived(fileData.localPath);
- }
- void SocketApi::processEncryptRequest(const QString &localFile)
- {
- Q_ASSERT(QFileInfo(localFile).isDir());
- const auto fileData = FileData::get(localFile);
- const auto folder = fileData.folder;
- Q_ASSERT(folder);
- const auto account = folder->accountState()->account();
- Q_ASSERT(account);
- const auto rec = fileData.journalRecord();
- Q_ASSERT(rec.isValid());
- if (!account->e2e() || account->e2e()->_mnemonic.isEmpty()) {
- const int ret = QMessageBox::critical(nullptr,
- tr("Failed to encrypt folder at \"%1\"").arg(fileData.folderRelativePath),
- tr("The account %1 does not have end-to-end encryption configured. "
- "Please configure this in your account settings to enable folder encryption.").arg(account->prettyName()));
- Q_UNUSED(ret)
- return;
- }
- auto path = rec._path;
- // Folder records have directory paths in Foo/Bar/ convention...
- // But EncryptFolderJob expects directory path Foo/Bar convention
- auto choppedPath = path;
- if (choppedPath.endsWith('/') && choppedPath != QStringLiteral("/")) {
- choppedPath.chop(1);
- }
- if (choppedPath.startsWith('/') && choppedPath != QStringLiteral("/")) {
- choppedPath = choppedPath.mid(1);
- }
- auto job = new OCC::EncryptFolderJob(account, folder->journalDb(), choppedPath, rec.numericFileId(), this);
- connect(job, &OCC::EncryptFolderJob::finished, this, [fileData, job](const int status) {
- if (status == OCC::EncryptFolderJob::Error) {
- const int ret = QMessageBox::critical(nullptr,
- tr("Failed to encrypt folder"),
- tr("Could not encrypt the following folder: \"%1\".\n\n"
- "Server replied with error: %2").arg(fileData.folderRelativePath, job->errorString()));
- Q_UNUSED(ret)
- } else {
- const int ret = QMessageBox::information(nullptr,
- tr("Folder encrypted successfully").arg(fileData.folderRelativePath),
- tr("The following folder was encrypted successfully: \"%1\"").arg(fileData.folderRelativePath));
- Q_UNUSED(ret)
- }
- });
- job->setProperty(encryptJobPropertyFolder, QVariant::fromValue(folder));
- job->setProperty(encryptJobPropertyPath, QVariant::fromValue(path));
- job->start();
- }
- void SocketApi::processShareRequest(const QString &localFile, SocketListener *listener)
- {
- auto theme = Theme::instance();
- auto fileData = FileData::get(localFile);
- auto shareFolder = fileData.folder;
- if (!shareFolder) {
- const QString message = QLatin1String("SHARE:NOP:") + QDir::toNativeSeparators(localFile);
- // files that are not within a sync folder are not synced.
- listener->sendMessage(message);
- } else if (!shareFolder->accountState()->isConnected()) {
- const QString message = QLatin1String("SHARE:NOTCONNECTED:") + QDir::toNativeSeparators(localFile);
- // if the folder isn't connected, don't open the share dialog
- listener->sendMessage(message);
- } else if (!theme->linkSharing() && (!theme->userGroupSharing() || shareFolder->accountState()->account()->serverVersionInt() < Account::makeServerVersion(8, 2, 0))) {
- const QString message = QLatin1String("SHARE:NOP:") + QDir::toNativeSeparators(localFile);
- listener->sendMessage(message);
- } else {
- // If the file doesn't have a journal record, it might not be uploaded yet
- if (!fileData.journalRecord().isValid()) {
- const QString message = QLatin1String("SHARE:NOTSYNCED:") + QDir::toNativeSeparators(localFile);
- listener->sendMessage(message);
- return;
- }
- if (!fileData.journalRecord().e2eMangledName().isEmpty()) {
- // we can not share an encrypted file or a subfolder under encrypted root foolder
- const QString message = QLatin1String("SHARE:NOP:") + QDir::toNativeSeparators(localFile);
- listener->sendMessage(message);
- return;
- }
- auto &remotePath = fileData.serverRelativePath;
- // Can't share root folder
- if (remotePath == "/") {
- const QString message = QLatin1String("SHARE:CANNOTSHAREROOT:") + QDir::toNativeSeparators(localFile);
- listener->sendMessage(message);
- return;
- }
- const QString message = QLatin1String("SHARE:OK:") + QDir::toNativeSeparators(localFile);
- listener->sendMessage(message);
- emit shareCommandReceived(fileData.localPath);
- }
- }
- void SocketApi::processLeaveShareRequest(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener)
- FolderMan::instance()->leaveShare(QDir::fromNativeSeparators(localFile));
- }
- void SocketApi::broadcastStatusPushMessage(const QString &systemPath, SyncFileStatus fileStatus)
- {
- QString msg = buildMessage(QLatin1String("STATUS"), systemPath, fileStatus.toSocketAPIString());
- Q_ASSERT(!systemPath.endsWith('/'));
- uint directoryHash = qHash(systemPath.left(systemPath.lastIndexOf('/')));
- for (const auto &listener : qAsConst(_listeners)) {
- listener->sendMessageIfDirectoryMonitored(msg, directoryHash);
- }
- }
- void SocketApi::command_RETRIEVE_FOLDER_STATUS(const QString &argument, SocketListener *listener)
- {
- // This command is the same as RETRIEVE_FILE_STATUS
- command_RETRIEVE_FILE_STATUS(argument, listener);
- }
- void SocketApi::command_RETRIEVE_FILE_STATUS(const QString &argument, SocketListener *listener)
- {
- QString statusString;
- auto fileData = FileData::get(argument);
- if (!fileData.folder) {
- // this can happen in offline mode e.g.: nothing to worry about
- statusString = QLatin1String("NOP");
- } else {
- // The user probably visited this directory in the file shell.
- // Let the listener know that it should now send status pushes for sibblings of this file.
- QString directory = fileData.localPath.left(fileData.localPath.lastIndexOf('/'));
- listener->registerMonitoredDirectory(qHash(directory));
- SyncFileStatus fileStatus = fileData.syncFileStatus();
- statusString = fileStatus.toSocketAPIString();
- }
- const QString message = QLatin1String("STATUS:") % statusString % QLatin1Char(':') % QDir::toNativeSeparators(argument);
- listener->sendMessage(message);
- }
- void SocketApi::command_SHARE(const QString &localFile, SocketListener *listener)
- {
- processShareRequest(localFile, listener);
- }
- void SocketApi::command_LEAVESHARE(const QString &localFile, SocketListener *listener)
- {
- processLeaveShareRequest(localFile, listener);
- }
- void SocketApi::command_ACTIVITY(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener);
- processFileActivityRequest(localFile);
- }
- void SocketApi::command_ENCRYPT(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener);
- processEncryptRequest(localFile);
- }
- void SocketApi::command_MANAGE_PUBLIC_LINKS(const QString &localFile, SocketListener *listener)
- {
- processShareRequest(localFile, listener);
- }
- void SocketApi::command_VERSION(const QString &, SocketListener *listener)
- {
- listener->sendMessage(QLatin1String("VERSION:" MIRALL_VERSION_STRING ":" MIRALL_SOCKET_API_VERSION));
- }
- void SocketApi::command_SHARE_MENU_TITLE(const QString &, SocketListener *listener)
- {
- //listener->sendMessage(QLatin1String("SHARE_MENU_TITLE:") + tr("Share with %1", "parameter is Nextcloud").arg(Theme::instance()->appNameGUI()));
- listener->sendMessage(QLatin1String("SHARE_MENU_TITLE:") + Theme::instance()->appNameGUI());
- }
- void SocketApi::command_EDIT(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener)
- auto fileData = FileData::get(localFile);
- if (!fileData.folder) {
- qCWarning(lcSocketApi) << "Unknown path" << localFile;
- return;
- }
- auto record = fileData.journalRecord();
- if (!record.isValid())
- return;
- DirectEditor* editor = getDirectEditorForLocalFile(fileData.localPath);
- if (!editor)
- return;
- auto *job = new JsonApiJob(fileData.folder->accountState()->account(), QLatin1String("ocs/v2.php/apps/files/api/v1/directEditing/open"), this);
- QUrlQuery params;
- params.addQueryItem("path", fileData.serverRelativePath);
- params.addQueryItem("editorId", editor->id());
- job->addQueryParams(params);
- job->setVerb(JsonApiJob::Verb::Post);
- QObject::connect(job, &JsonApiJob::jsonReceived, [](const QJsonDocument &json){
- auto data = json.object().value("ocs").toObject().value("data").toObject();
- auto url = QUrl(data.value("url").toString());
- if(!url.isEmpty())
- Utility::openBrowser(url);
- });
- job->start();
- }
- // don't pull the share manager into socketapi unittests
- #ifndef OWNCLOUD_TEST
- class GetOrCreatePublicLinkShare : public QObject
- {
- Q_OBJECT
- public:
- GetOrCreatePublicLinkShare(const AccountPtr &account, const QString &localFile, const bool isSecureFileDropOnlyFolder,
- QObject *parent)
- : QObject(parent)
- , _account(account)
- , _shareManager(account)
- , _localFile(localFile)
- , _isSecureFileDropOnlyFolder(isSecureFileDropOnlyFolder)
- {
- connect(&_shareManager, &ShareManager::sharesFetched,
- this, &GetOrCreatePublicLinkShare::sharesFetched);
- connect(&_shareManager, &ShareManager::linkShareCreated,
- this, &GetOrCreatePublicLinkShare::linkShareCreated);
- connect(&_shareManager, &ShareManager::linkShareRequiresPassword,
- this, &GetOrCreatePublicLinkShare::linkShareRequiresPassword);
- connect(&_shareManager, &ShareManager::serverError,
- this, &GetOrCreatePublicLinkShare::serverError);
- }
- void run()
- {
- qCDebug(lcPublicLink) << "Fetching shares";
- _shareManager.fetchShares(_localFile);
- }
- private slots:
- void sharesFetched(const QList<OCC::SharePtr> &shares)
- {
- auto shareName = SocketApi::tr("Context menu share");
- // If there already is a context menu share, reuse it
- for (const auto &share : shares) {
- const auto linkShare = qSharedPointerDynamicCast<LinkShare>(share);
- if (!linkShare)
- continue;
- if (linkShare->getName() == shareName) {
- qCDebug(lcPublicLink) << "Found existing share, reusing";
- return success(linkShare->getLink().toString());
- }
- }
- // otherwise create a new one
- qCDebug(lcPublicLink) << "Creating new share";
- if (_isSecureFileDropOnlyFolder) {
- _shareManager.createSecureFileDropShare(_localFile, shareName, QString());
- } else {
- _shareManager.createLinkShare(_localFile, shareName, QString());
- }
- }
- void linkShareCreated(const QSharedPointer<OCC::LinkShare> &share)
- {
- qCDebug(lcPublicLink) << "New share created";
- success(share->getLink().toString());
- }
- void passwordRequired() {
- bool ok = false;
- QString password = QInputDialog::getText(nullptr,
- tr("Password for share required"),
- tr("Please enter a password for your link share:"),
- QLineEdit::Normal,
- QString(),
- &ok);
- if (!ok) {
- // The dialog was canceled so no need to do anything
- return;
- }
- // Try to create the link share again with the newly entered password
- _shareManager.createLinkShare(_localFile, QString(), password);
- }
- void linkShareRequiresPassword(const QString &message)
- {
- qCInfo(lcPublicLink) << "Could not create link share:" << message;
- emit error(message);
- deleteLater();
- }
- void serverError(int code, const QString &message)
- {
- qCWarning(lcPublicLink) << "Share fetch/create error" << code << message;
- QMessageBox::warning(
- nullptr,
- tr("Sharing error"),
- tr("Could not retrieve or create the public link share. Error:\n\n%1").arg(message),
- QMessageBox::Ok,
- QMessageBox::NoButton);
- emit error(message);
- deleteLater();
- }
- signals:
- void done(const QString &link);
- void error(const QString &message);
- private:
- void success(const QString &link)
- {
- emit done(link);
- deleteLater();
- }
- AccountPtr _account;
- ShareManager _shareManager;
- QString _localFile;
- bool _isSecureFileDropOnlyFolder = false;
- };
- #else
- class GetOrCreatePublicLinkShare : public QObject
- {
- Q_OBJECT
- public:
- GetOrCreatePublicLinkShare(const AccountPtr &, const QString &,
- std::function<void(const QString &link)>, QObject *)
- {
- }
- void run()
- {
- }
- };
- #endif
- void SocketApi::command_COPY_SECUREFILEDROP_LINK(const QString &localFile, SocketListener *)
- {
- const auto fileData = FileData::get(localFile);
- if (!fileData.folder) {
- return;
- }
- const auto account = fileData.folder->accountState()->account();
- const auto getOrCreatePublicLinkShareJob = new GetOrCreatePublicLinkShare(account, fileData.serverRelativePath, true, this);
- connect(getOrCreatePublicLinkShareJob, &GetOrCreatePublicLinkShare::done, this, [](const QString &url) { copyUrlToClipboard(url); });
- connect(getOrCreatePublicLinkShareJob, &GetOrCreatePublicLinkShare::error, this, [=]() { emit shareCommandReceived(fileData.localPath); });
- getOrCreatePublicLinkShareJob->run();
- }
- void SocketApi::command_COPY_PUBLIC_LINK(const QString &localFile, SocketListener *)
- {
- const auto fileData = FileData::get(localFile);
- if (!fileData.folder) {
- return;
- }
- const auto account = fileData.folder->accountState()->account();
- const auto getOrCreatePublicLinkShareJob = new GetOrCreatePublicLinkShare(account, fileData.serverRelativePath, false, this);
- connect(getOrCreatePublicLinkShareJob, &GetOrCreatePublicLinkShare::done, this, [](const QString &url) {
- copyUrlToClipboard(url);
- });
- connect(getOrCreatePublicLinkShareJob, &GetOrCreatePublicLinkShare::error, this, [=]() {
- emit shareCommandReceived(fileData.localPath);
- });
- getOrCreatePublicLinkShareJob->run();
- }
- // Windows Shell / Explorer pinning fallbacks, see issue: https://github.com/nextcloud/desktop/issues/1599
- #ifdef Q_OS_WIN
- void SocketApi::command_COPYASPATH(const QString &localFile, SocketListener *)
- {
- QApplication::clipboard()->setText(localFile);
- }
- void SocketApi::command_OPENNEWWINDOW(const QString &localFile, SocketListener *)
- {
- QDesktopServices::openUrl(QUrl::fromLocalFile(localFile));
- }
- void SocketApi::command_OPEN(const QString &localFile, SocketListener *socketListener)
- {
- command_OPENNEWWINDOW(localFile, socketListener);
- }
- #endif
- // Fetches the private link url asynchronously and then calls the target slot
- void SocketApi::fetchPrivateLinkUrlHelper(const QString &localFile, const std::function<void(const QString &url)> &targetFun)
- {
- auto fileData = FileData::get(localFile);
- if (!fileData.folder) {
- qCWarning(lcSocketApi) << "Unknown path" << localFile;
- return;
- }
- auto record = fileData.journalRecord();
- if (!record.isValid())
- return;
- fetchPrivateLinkUrl(
- fileData.folder->accountState()->account(),
- fileData.serverRelativePath,
- record.numericFileId(),
- this,
- targetFun);
- }
- void SocketApi::command_COPY_PRIVATE_LINK(const QString &localFile, SocketListener *)
- {
- fetchPrivateLinkUrlHelper(localFile, &SocketApi::copyUrlToClipboard);
- }
- void SocketApi::command_EMAIL_PRIVATE_LINK(const QString &localFile, SocketListener *)
- {
- fetchPrivateLinkUrlHelper(localFile, &SocketApi::emailPrivateLink);
- }
- void SocketApi::command_OPEN_PRIVATE_LINK(const QString &localFile, SocketListener *)
- {
- fetchPrivateLinkUrlHelper(localFile, &SocketApi::openPrivateLink);
- }
- void SocketApi::command_MAKE_AVAILABLE_LOCALLY(const QString &filesArg, SocketListener *)
- {
- const QStringList files = split(filesArg);
- for (const auto &file : files) {
- auto data = FileData::get(file);
- if (!data.folder)
- continue;
- // Update the pin state on all items
- if (!data.folder->vfs().setPinState(data.folderRelativePath, PinState::AlwaysLocal)) {
- qCWarning(lcSocketApi) << "Could not set pin state of" << data.folderRelativePath << "to always local";
- }
- // Trigger sync
- data.folder->schedulePathForLocalDiscovery(data.folderRelativePath);
- data.folder->scheduleThisFolderSoon();
- }
- }
- /* Go over all the files and replace them by a virtual file */
- void SocketApi::command_MAKE_ONLINE_ONLY(const QString &filesArg, SocketListener *)
- {
- const QStringList files = split(filesArg);
- for (const auto &file : files) {
- auto data = FileData::get(file);
- if (!data.folder)
- continue;
- // Update the pin state on all items
- if (!data.folder->vfs().setPinState(data.folderRelativePath, PinState::OnlineOnly)) {
- qCWarning(lcSocketApi) << "Could not set pin state of" << data.folderRelativePath << "to online only";
- }
- // Trigger sync
- data.folder->schedulePathForLocalDiscovery(data.folderRelativePath);
- data.folder->scheduleThisFolderSoon();
- }
- }
- void SocketApi::copyUrlToClipboard(const QString &link)
- {
- QApplication::clipboard()->setText(link);
- }
- void SocketApi::command_RESOLVE_CONFLICT(const QString &localFile, SocketListener *)
- {
- const auto fileData = FileData::get(localFile);
- if (!fileData.folder || !Utility::isConflictFile(fileData.folderRelativePath))
- return; // should not have shown menu item
- const auto conflictedRelativePath = fileData.folderRelativePath;
- const auto baseRelativePath = fileData.folder->journalDb()->conflictFileBaseName(fileData.folderRelativePath.toUtf8());
- const auto dir = QDir(fileData.folder->path());
- const auto conflictedPath = dir.filePath(conflictedRelativePath);
- const auto basePath = dir.filePath(baseRelativePath);
- const auto baseName = QFileInfo(basePath).fileName();
- #ifndef OWNCLOUD_TEST
- ConflictDialog dialog;
- dialog.setBaseFilename(baseName);
- dialog.setLocalVersionFilename(conflictedPath);
- dialog.setRemoteVersionFilename(basePath);
- if (dialog.exec() == ConflictDialog::Accepted) {
- fileData.folder->scheduleThisFolderSoon();
- }
- #endif
- }
- void SocketApi::command_DELETE_ITEM(const QString &localFile, SocketListener *)
- {
- ConflictSolver solver;
- solver.setLocalVersionFilename(localFile);
- solver.exec(ConflictSolver::KeepRemoteVersion);
- }
- void SocketApi::command_MOVE_ITEM(const QString &localFile, SocketListener *)
- {
- const auto fileData = FileData::get(localFile);
- const auto parentDir = fileData.parentFolder();
- if (!fileData.folder)
- return; // should not have shown menu item
- QString defaultDirAndName = fileData.folderRelativePath;
- // If it's a conflict, we want to save it under the base name by default
- if (Utility::isConflictFile(defaultDirAndName)) {
- defaultDirAndName = fileData.folder->journalDb()->conflictFileBaseName(fileData.folderRelativePath.toUtf8());
- }
- // If the parent doesn't accept new files, go to the root of the sync folder
- QFileInfo fileInfo(localFile);
- const auto parentRecord = parentDir.journalRecord();
- if ((fileInfo.isFile() && !parentRecord._remotePerm.hasPermission(RemotePermissions::CanAddFile))
- || (fileInfo.isDir() && !parentRecord._remotePerm.hasPermission(RemotePermissions::CanAddSubDirectories))) {
- defaultDirAndName = QFileInfo(defaultDirAndName).fileName();
- }
- // Add back the folder path
- defaultDirAndName = QDir(fileData.folder->path()).filePath(defaultDirAndName);
- const auto target = QFileDialog::getSaveFileName(
- nullptr,
- tr("Select new location …"),
- defaultDirAndName,
- QString(), nullptr, QFileDialog::HideNameFilterDetails);
- if (target.isEmpty())
- return;
- ConflictSolver solver;
- solver.setLocalVersionFilename(localFile);
- solver.setRemoteVersionFilename(target);
- }
- void SocketApi::command_LOCK_FILE(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener)
- setFileLock(localFile, SyncFileItem::LockStatus::LockedItem);
- }
- void SocketApi::command_UNLOCK_FILE(const QString &localFile, SocketListener *listener)
- {
- Q_UNUSED(listener)
- setFileLock(localFile, SyncFileItem::LockStatus::UnlockedItem);
- }
- void SocketApi::setFileLock(const QString &localFile, const SyncFileItem::LockStatus lockState) const
- {
- const auto fileData = FileData::get(localFile);
- const auto shareFolder = fileData.folder;
- if (!shareFolder || !shareFolder->accountState()->isConnected()) {
- return;
- }
- shareFolder->accountState()->account()->setLockFileState(fileData.serverRelativePath, shareFolder->journalDb(), lockState);
- shareFolder->journalDb()->schedulePathForRemoteDiscovery(fileData.serverRelativePath);
- shareFolder->scheduleThisFolderSoon();
- }
- void SocketApi::command_V2_LIST_ACCOUNTS(const QSharedPointer<SocketApiJobV2> &job) const
- {
- QJsonArray out;
- const auto accounts = AccountManager::instance()->accounts();
- for (auto acc : accounts) {
- // TODO: Use uuid once https://github.com/owncloud/client/pull/8397 is merged
- out << QJsonObject({ { "name", acc->account()->displayName() }, { "id", acc->account()->id() } });
- }
- job->success({ { "accounts", out } });
- }
- void SocketApi::command_V2_UPLOAD_FILES_FROM(const QSharedPointer<SocketApiJobV2> &job) const
- {
- auto uploadJob = new SocketUploadJob(job);
- uploadJob->start();
- }
- void SocketApi::emailPrivateLink(const QString &link)
- {
- Utility::openEmailComposer(
- tr("I shared something with you"),
- link,
- nullptr);
- }
- void OCC::SocketApi::openPrivateLink(const QString &link)
- {
- Utility::openBrowser(link);
- }
- void SocketApi::command_GET_STRINGS(const QString &argument, SocketListener *listener)
- {
- static std::array<std::pair<const char *, QString>, 6> strings { {
- { "SHARE_MENU_TITLE", tr("Share options") },
- { "FILE_ACTIVITY_MENU_TITLE", tr("Activity") },
- { "CONTEXT_MENU_TITLE", Theme::instance()->appNameGUI() },
- { "COPY_PRIVATE_LINK_MENU_TITLE", tr("Copy private link to clipboard") },
- { "EMAIL_PRIVATE_LINK_MENU_TITLE", tr("Send private link by email …") },
- { "CONTEXT_MENU_ICON", APPLICATION_ICON_NAME },
- } };
- listener->sendMessage(QString("GET_STRINGS:BEGIN"));
- for (const auto& key_value : strings) {
- if (argument.isEmpty() || argument == QLatin1String(key_value.first)) {
- listener->sendMessage(QString("STRING:%1:%2").arg(key_value.first, key_value.second));
- }
- }
- listener->sendMessage(QString("GET_STRINGS:END"));
- }
- void SocketApi::sendSharingContextMenuOptions(const FileData &fileData, SocketListener *listener, SharingContextItemEncryptedFlag itemEncryptionFlag, SharingContextItemRootEncryptedFolderFlag rootE2eeFolderFlag)
- {
- const auto record = fileData.journalRecord();
- const auto isOnTheServer = record.isValid();
- const auto isSecureFileDropSupported = rootE2eeFolderFlag == SharingContextItemRootEncryptedFolderFlag::RootEncryptedFolder && fileData.folder->accountState()->account()->secureFileDropSupported();
- const auto flagString = isOnTheServer && (itemEncryptionFlag == SharingContextItemEncryptedFlag::NotEncryptedItem || isSecureFileDropSupported) ? QLatin1String("::") : QLatin1String(":d:");
- auto capabilities = fileData.folder->accountState()->account()->capabilities();
- auto theme = Theme::instance();
- if (!capabilities.shareAPI() || !(theme->userGroupSharing() || (theme->linkSharing() && capabilities.sharePublicLink())))
- return;
- if (record._isShared && !record._sharedByMe) {
- listener->sendMessage(QLatin1String("MENU_ITEM:LEAVESHARE") + flagString + tr("Leave this share"));
- }
- // If sharing is globally disabled, do not show any sharing entries.
- // If there is no permission to share for this file, add a disabled entry saying so
- if (isOnTheServer && !record._remotePerm.isNull() && !record._remotePerm.hasPermission(RemotePermissions::CanReshare)) {
- listener->sendMessage(QLatin1String("MENU_ITEM:DISABLED:d:") + (!record.isDirectory() ? tr("Resharing this file is not allowed") : tr("Resharing this folder is not allowed")));
- } else {
- listener->sendMessage(QLatin1String("MENU_ITEM:SHARE") + flagString + tr("Share options"));
- // Do we have public links?
- bool publicLinksEnabled = theme->linkSharing() && capabilities.sharePublicLink();
- // Is is possible to create a public link without user choices?
- bool canCreateDefaultPublicLink = publicLinksEnabled
- && !capabilities.sharePublicLinkEnforceExpireDate()
- && !capabilities.sharePublicLinkAskOptionalPassword()
- && !capabilities.sharePublicLinkEnforcePassword();
- if (canCreateDefaultPublicLink) {
- if (isSecureFileDropSupported) {
- listener->sendMessage(QLatin1String("MENU_ITEM:COPY_SECUREFILEDROP_LINK") + QLatin1String("::") + tr("Copy secure filedrop link"));
- } else {
- listener->sendMessage(QLatin1String("MENU_ITEM:COPY_PUBLIC_LINK") + flagString + tr("Copy public link"));
- }
- } else if (publicLinksEnabled) {
- if (isSecureFileDropSupported) {
- listener->sendMessage(QLatin1String("MENU_ITEM:MANAGE_PUBLIC_LINKS") + QLatin1String("::") + tr("Copy secure filedrop link"));
- } else {
- listener->sendMessage(QLatin1String("MENU_ITEM:MANAGE_PUBLIC_LINKS") + flagString + tr("Copy public link"));
- }
- }
- }
- if (itemEncryptionFlag == SharingContextItemEncryptedFlag::NotEncryptedItem) {
- listener->sendMessage(QLatin1String("MENU_ITEM:COPY_PRIVATE_LINK") + flagString + tr("Copy internal link"));
- }
- // Disabled: only providing email option for private links would look odd,
- // and the copy option is more general.
- //listener->sendMessage(QLatin1String("MENU_ITEM:EMAIL_PRIVATE_LINK") + flagString + tr("Send private link by email …"));
- }
- void SocketApi::sendEncryptFolderCommandMenuEntries(const QFileInfo &fileInfo,
- const FileData &fileData,
- const bool isE2eEncryptedPath,
- const OCC::SocketListener* const listener) const
- {
- if (!listener ||
- !fileData.folder ||
- !fileData.folder->accountState() ||
- !fileData.folder->accountState()->account() ||
- !fileData.folder->accountState()->account()->capabilities().clientSideEncryptionAvailable() ||
- !fileInfo.isDir() ||
- isE2eEncryptedPath) {
- return;
- }
- bool anyAncestorEncrypted = false;
- auto ancestor = fileData.parentFolder();
- while (ancestor.journalRecord().isValid()) {
- if (ancestor.journalRecord()._isE2eEncrypted) {
- anyAncestorEncrypted = true;
- break;
- }
- ancestor = ancestor.parentFolder();
- }
- if (!anyAncestorEncrypted) {
- const auto isOnTheServer = fileData.journalRecord().isValid();
- const auto flagString = isOnTheServer ? QLatin1String("::") : QLatin1String(":d:");
- listener->sendMessage(QStringLiteral("MENU_ITEM:ENCRYPT") + flagString + tr("Encrypt"));
- }
- }
- void SocketApi::sendLockFileCommandMenuEntries(const QFileInfo &fileInfo,
- Folder* const syncFolder,
- const FileData &fileData,
- const OCC::SocketListener* const listener) const
- {
- if (!fileInfo.isDir() && syncFolder->accountState()->account()->capabilities().filesLockAvailable()) {
- if (syncFolder->accountState()->account()->fileLockStatus(syncFolder->journalDb(), fileData.folderRelativePath) == SyncFileItem::LockStatus::UnlockedItem) {
- listener->sendMessage(QLatin1String("MENU_ITEM:LOCK_FILE::") + tr("Lock file"));
- } else {
- if (syncFolder->accountState()->account()->fileCanBeUnlocked(syncFolder->journalDb(), fileData.folderRelativePath)) {
- listener->sendMessage(QLatin1String("MENU_ITEM:UNLOCK_FILE::") + tr("Unlock file"));
- }
- }
- }
- }
- void SocketApi::sendLockFileInfoMenuEntries(const QFileInfo &fileInfo,
- Folder * const syncFolder,
- const FileData &fileData,
- const SocketListener * const listener,
- const SyncJournalFileRecord &record) const
- {
- static constexpr auto SECONDS_PER_MINUTE = 60;
- if (!fileInfo.isDir() && syncFolder->accountState()->account()->capabilities().filesLockAvailable() &&
- syncFolder->accountState()->account()->fileLockStatus(syncFolder->journalDb(), fileData.folderRelativePath) == SyncFileItem::LockStatus::LockedItem) {
- listener->sendMessage(QLatin1String("MENU_ITEM:LOCKED_FILE_OWNER:d:") + tr("Locked by %1").arg(record._lockstate._lockOwnerDisplayName));
- const auto lockExpirationTime = record._lockstate._lockTime + record._lockstate._lockTimeout;
- const auto remainingTime = QDateTime::currentDateTime().secsTo(QDateTime::fromSecsSinceEpoch(lockExpirationTime));
- const auto remainingTimeInMinute = static_cast<int>(remainingTime > 0 ? remainingTime / SECONDS_PER_MINUTE : 0);
- listener->sendMessage(QLatin1String("MENU_ITEM:LOCKED_FILE_DATE:d:") + tr("Expires in %1 minutes", "remaining time before lock expires", remainingTimeInMinute).arg(remainingTimeInMinute));
- }
- }
- SocketApi::FileData SocketApi::FileData::get(const QString &localFile)
- {
- FileData data;
- data.localPath = QDir::cleanPath(localFile);
- if (data.localPath.endsWith(QLatin1Char('/')))
- data.localPath.chop(1);
- data.folder = FolderMan::instance()->folderForPath(data.localPath);
- if (!data.folder)
- return data;
- data.folderRelativePath = data.localPath.mid(data.folder->cleanPath().length() + 1);
- data.serverRelativePath = QDir(data.folder->remotePath()).filePath(data.folderRelativePath);
- QString virtualFileExt = QStringLiteral(APPLICATION_DOTVIRTUALFILE_SUFFIX);
- if (data.serverRelativePath.endsWith(virtualFileExt)) {
- data.serverRelativePath.chop(virtualFileExt.size());
- }
- return data;
- }
- QString SocketApi::FileData::folderRelativePathNoVfsSuffix() const
- {
- auto result = folderRelativePath;
- QString virtualFileExt = QStringLiteral(APPLICATION_DOTVIRTUALFILE_SUFFIX);
- if (result.endsWith(virtualFileExt)) {
- result.chop(virtualFileExt.size());
- }
- return result;
- }
- SyncFileStatus SocketApi::FileData::syncFileStatus() const
- {
- if (!folder)
- return SyncFileStatus::StatusNone;
- return folder->syncEngine().syncFileStatusTracker().fileStatus(folderRelativePath);
- }
- SyncJournalFileRecord SocketApi::FileData::journalRecord() const
- {
- SyncJournalFileRecord record;
- if (!folder)
- return record;
- if (!folder->journalDb()->getFileRecord(folderRelativePath, &record)) {
- qCWarning(lcSocketApi) << "Failed to get journal record for path" << folderRelativePath;
- }
- return record;
- }
- SocketApi::FileData SocketApi::FileData::parentFolder() const
- {
- return FileData::get(QFileInfo(localPath).dir().path().toUtf8());
- }
- void SocketApi::command_GET_MENU_ITEMS(const QString &argument, OCC::SocketListener *listener)
- {
- listener->sendMessage(QString("GET_MENU_ITEMS:BEGIN"));
- const QStringList files = split(argument);
- // Find the common sync folder.
- // syncFolder will be null if files are in different folders.
- Folder *syncFolder = nullptr;
- for (const auto &file : files) {
- auto folder = FolderMan::instance()->folderForPath(file);
- if (folder != syncFolder) {
- if (!syncFolder) {
- syncFolder = folder;
- } else {
- syncFolder = nullptr;
- break;
- }
- }
- }
- // Sharing actions show for single files only
- if (syncFolder && files.size() == 1 && syncFolder->accountState()->isConnected()) {
- QString systemPath = QDir::cleanPath(argument);
- if (systemPath.endsWith(QLatin1Char('/'))) {
- systemPath.truncate(systemPath.length() - 1);
- }
- FileData fileData = FileData::get(argument);
- const auto record = fileData.journalRecord();
- const bool isOnTheServer = record.isValid();
- const auto isE2eEncryptedPath = fileData.journalRecord()._isE2eEncrypted || !fileData.journalRecord()._e2eMangledName.isEmpty();
- const auto isE2eEncryptedRootFolder = fileData.journalRecord()._isE2eEncrypted && fileData.journalRecord()._e2eMangledName.isEmpty();
- auto flagString = isOnTheServer && !isE2eEncryptedPath ? QLatin1String("::") : QLatin1String(":d:");
- const QFileInfo fileInfo(fileData.localPath);
- sendLockFileInfoMenuEntries(fileInfo, syncFolder, fileData, listener, record);
- if (!fileInfo.isDir()) {
- listener->sendMessage(QLatin1String("MENU_ITEM:ACTIVITY") + flagString + tr("Activity"));
- }
- DirectEditor* editor = getDirectEditorForLocalFile(fileData.localPath);
- if (editor) {
- //listener->sendMessage(QLatin1String("MENU_ITEM:EDIT") + flagString + tr("Edit via ") + editor->name());
- listener->sendMessage(QLatin1String("MENU_ITEM:EDIT") + flagString + tr("Edit"));
- } else {
- listener->sendMessage(QLatin1String("MENU_ITEM:OPEN_PRIVATE_LINK") + flagString + tr("Open in browser"));
- }
- sendEncryptFolderCommandMenuEntries(fileInfo, fileData, isE2eEncryptedPath, listener);
- sendLockFileCommandMenuEntries(fileInfo, syncFolder, fileData, listener);
- const auto itemEncryptionFlag = isE2eEncryptedPath ? SharingContextItemEncryptedFlag::EncryptedItem : SharingContextItemEncryptedFlag::NotEncryptedItem;
- const auto rootE2eeFolderFlag = isE2eEncryptedRootFolder ? SharingContextItemRootEncryptedFolderFlag::RootEncryptedFolder : SharingContextItemRootEncryptedFolderFlag::NonRootEncryptedFolder;
- sendSharingContextMenuOptions(fileData, listener, itemEncryptionFlag, rootE2eeFolderFlag);
- // Conflict files get conflict resolution actions
- bool isConflict = Utility::isConflictFile(fileData.folderRelativePath);
- if (isConflict || !isOnTheServer) {
- // Check whether this new file is in a read-only directory
- const auto parentDir = fileData.parentFolder();
- const auto parentRecord = parentDir.journalRecord();
- const bool canAddToDir =
- !parentRecord.isValid() // We're likely at the root of the sync folder, got to assume we can add there
- || (fileInfo.isFile() && parentRecord._remotePerm.hasPermission(RemotePermissions::CanAddFile))
- || (fileInfo.isDir() && parentRecord._remotePerm.hasPermission(RemotePermissions::CanAddSubDirectories));
- const bool canChangeFile =
- !isOnTheServer
- || (record._remotePerm.hasPermission(RemotePermissions::CanDelete)
- && record._remotePerm.hasPermission(RemotePermissions::CanMove)
- && record._remotePerm.hasPermission(RemotePermissions::CanRename));
- if (isConflict && canChangeFile) {
- if (canAddToDir) {
- listener->sendMessage(QLatin1String("MENU_ITEM:RESOLVE_CONFLICT::") + tr("Resolve conflict …"));
- } else {
- if (isOnTheServer) {
- // Uploaded conflict file in read-only directory
- listener->sendMessage(QLatin1String("MENU_ITEM:MOVE_ITEM::") + tr("Move and rename …"));
- } else {
- // Local-only conflict file in a read-only dir
- listener->sendMessage(QLatin1String("MENU_ITEM:MOVE_ITEM::") + tr("Move, rename and upload …"));
- }
- listener->sendMessage(QLatin1String("MENU_ITEM:DELETE_ITEM::") + tr("Delete local changes"));
- }
- }
- // File in a read-only directory?
- if (!isConflict && !isOnTheServer && !canAddToDir) {
- listener->sendMessage(QLatin1String("MENU_ITEM:MOVE_ITEM::") + tr("Move and upload …"));
- listener->sendMessage(QLatin1String("MENU_ITEM:DELETE_ITEM::") + tr("Delete"));
- }
- }
- }
- // File availability actions
- if (syncFolder
- && syncFolder->virtualFilesEnabled()
- && syncFolder->vfs().socketApiPinStateActionsShown()) {
- ENFORCE(!files.isEmpty());
- // Determine the combined availability status of the files
- auto combined = Optional<VfsItemAvailability>();
- auto merge = [](VfsItemAvailability lhs, VfsItemAvailability rhs) {
- if (lhs == rhs)
- return lhs;
- if (int(lhs) > int(rhs))
- std::swap(lhs, rhs); // reduce cases ensuring lhs < rhs
- if (lhs == VfsItemAvailability::AlwaysLocal && rhs == VfsItemAvailability::AllHydrated)
- return VfsItemAvailability::AllHydrated;
- if (lhs == VfsItemAvailability::AllDehydrated && rhs == VfsItemAvailability::OnlineOnly)
- return VfsItemAvailability::AllDehydrated;
- return VfsItemAvailability::Mixed;
- };
- for (const auto &file : files) {
- auto fileData = FileData::get(file);
- auto availability = syncFolder->vfs().availability(fileData.folderRelativePath);
- if (!availability) {
- if (availability.error() == Vfs::AvailabilityError::DbError)
- availability = VfsItemAvailability::Mixed;
- if (availability.error() == Vfs::AvailabilityError::NoSuchItem)
- continue;
- }
- if (!combined) {
- combined = *availability;
- } else {
- combined = merge(*combined, *availability);
- }
- }
- // TODO: Should be a submenu, should use icons
- auto makePinContextMenu = [&](bool makeAvailableLocally, bool freeSpace) {
- listener->sendMessage(QLatin1String("MENU_ITEM:CURRENT_PIN:d:")
- + Utility::vfsCurrentAvailabilityText(*combined));
- if (!Theme::instance()->enforceVirtualFilesSyncFolder()) {
- listener->sendMessage(QLatin1String("MENU_ITEM:MAKE_AVAILABLE_LOCALLY:")
- + (makeAvailableLocally ? QLatin1String(":") : QLatin1String("d:")) + Utility::vfsPinActionText());
- }
-
- listener->sendMessage(QLatin1String("MENU_ITEM:MAKE_ONLINE_ONLY:")
- + (freeSpace ? QLatin1String(":") : QLatin1String("d:"))
- + Utility::vfsFreeSpaceActionText());
- };
- if (combined) {
- switch (*combined) {
- case VfsItemAvailability::AlwaysLocal:
- makePinContextMenu(false, true);
- break;
- case VfsItemAvailability::AllHydrated:
- case VfsItemAvailability::Mixed:
- makePinContextMenu(true, true);
- break;
- case VfsItemAvailability::AllDehydrated:
- case VfsItemAvailability::OnlineOnly:
- makePinContextMenu(true, false);
- break;
- }
- }
- }
- listener->sendMessage(QString("GET_MENU_ITEMS:END"));
- }
- DirectEditor* SocketApi::getDirectEditorForLocalFile(const QString &localFile)
- {
- FileData fileData = FileData::get(localFile);
- auto capabilities = fileData.folder->accountState()->account()->capabilities();
- if (fileData.folder && fileData.folder->accountState()->isConnected()) {
- const auto record = fileData.journalRecord();
- const auto mimeMatchMode = record.isVirtualFile() ? QMimeDatabase::MatchExtension : QMimeDatabase::MatchDefault;
- QMimeDatabase db;
- QMimeType type = db.mimeTypeForFile(localFile, mimeMatchMode);
- DirectEditor* editor = capabilities.getDirectEditorForMimetype(type);
- if (!editor) {
- editor = capabilities.getDirectEditorForOptionalMimetype(type);
- }
- return editor;
- }
- return nullptr;
- }
- #if GUI_TESTING
- void SocketApi::command_ASYNC_LIST_WIDGETS(const QSharedPointer<SocketApiJob> &job)
- {
- QString response;
- for (auto &widget : allObjects(QApplication::allWidgets())) {
- auto objectName = widget->objectName();
- if (!objectName.isEmpty()) {
- response += objectName + ":" + widget->property("text").toString() + ", ";
- }
- }
- job->resolve(response);
- }
- void SocketApi::command_ASYNC_INVOKE_WIDGET_METHOD(const QSharedPointer<SocketApiJob> &job)
- {
- auto &arguments = job->arguments();
- auto widget = findWidget(arguments["objectName"].toString());
- if (!widget) {
- job->reject(QLatin1String("widget not found"));
- return;
- }
- QMetaObject::invokeMethod(widget, arguments["method"].toString().toUtf8().constData());
- job->resolve();
- }
- void SocketApi::command_ASYNC_GET_WIDGET_PROPERTY(const QSharedPointer<SocketApiJob> &job)
- {
- QString widgetName = job->arguments()[QLatin1String("objectName")].toString();
- auto widget = findWidget(widgetName);
- if (!widget) {
- QString message = QString(QLatin1String("Widget not found: 2: %1")).arg(widgetName);
- job->reject(message);
- return;
- }
- auto propertyName = job->arguments()[QLatin1String("property")].toString();
- auto segments = propertyName.split('.');
- QObject *currentObject = widget;
- QString value;
- for (int i = 0; i < segments.count(); i++) {
- auto segment = segments.at(i);
- auto var = currentObject->property(segment.toUtf8().constData());
- if (var.canConvert<QString>()) {
- var.convert(QMetaType::QString);
- value = var.value<QString>();
- break;
- }
- auto tmpObject = var.value<QObject *>();
- if (tmpObject) {
- currentObject = tmpObject;
- } else {
- QString message = QString(QLatin1String("Widget not found: 3: %1")).arg(widgetName);
- job->reject(message);
- return;
- }
- }
- job->resolve(value);
- }
- void SocketApi::command_ASYNC_SET_WIDGET_PROPERTY(const QSharedPointer<SocketApiJob> &job)
- {
- auto &arguments = job->arguments();
- QString widgetName = arguments["objectName"].toString();
- auto widget = findWidget(widgetName);
- if (!widget) {
- QString message = QString(QLatin1String("Widget not found: 4: %1")).arg(widgetName);
- job->reject(message);
- return;
- }
- widget->setProperty(arguments["property"].toString().toUtf8().constData(),
- arguments["value"]);
- job->resolve();
- }
- void SocketApi::command_ASYNC_WAIT_FOR_WIDGET_SIGNAL(const QSharedPointer<SocketApiJob> &job)
- {
- auto &arguments = job->arguments();
- QString widgetName = arguments["objectName"].toString();
- auto widget = findWidget(arguments["objectName"].toString());
- if (!widget) {
- QString message = QString(QLatin1String("Widget not found: 5: %1")).arg(widgetName);
- job->reject(message);
- return;
- }
- auto closure = new ListenerClosure([job]() { job->resolve("signal emitted"); });
- auto signalSignature = arguments["signalSignature"].toString();
- signalSignature.prepend("2");
- auto utf8 = signalSignature.toUtf8();
- auto signalSignatureFinal = utf8.constData();
- connect(widget, signalSignatureFinal, closure, SLOT(closureSlot()), Qt::QueuedConnection);
- }
- void SocketApi::command_ASYNC_TRIGGER_MENU_ACTION(const QSharedPointer<SocketApiJob> &job)
- {
- auto &arguments = job->arguments();
- auto objectName = arguments["objectName"].toString();
- auto widget = findWidget(objectName);
- if (!widget) {
- QString message = QString(QLatin1String("Object not found: 1: %1")).arg(objectName);
- job->reject(message);
- return;
- }
- auto children = widget->findChildren<QWidget *>();
- for (auto childWidget : children) {
- // foo is the popupwidget!
- auto actions = childWidget->actions();
- for (auto action : actions) {
- if (action->objectName() == arguments["actionName"].toString()) {
- action->trigger();
- job->resolve("action found");
- return;
- }
- }
- }
- QString message = QString(QLatin1String("Action not found: 1: %1")).arg(arguments["actionName"].toString());
- job->reject(message);
- }
- void SocketApi::command_ASYNC_ASSERT_ICON_IS_EQUAL(const QSharedPointer<SocketApiJob> &job)
- {
- auto widget = findWidget(job->arguments()[QLatin1String("queryString")].toString());
- if (!widget) {
- QString message = QString(QLatin1String("Object not found: 6: %1")).arg(job->arguments()["queryString"].toString());
- job->reject(message);
- return;
- }
- auto propertyName = job->arguments()[QLatin1String("propertyPath")].toString();
- auto segments = propertyName.split('.');
- QObject *currentObject = widget;
- QIcon value;
- for (int i = 0; i < segments.count(); i++) {
- auto segment = segments.at(i);
- auto var = currentObject->property(segment.toUtf8().constData());
- if (var.canConvert<QIcon>()) {
- var.convert(QMetaType::QIcon);
- value = var.value<QIcon>();
- break;
- }
- auto tmpObject = var.value<QObject *>();
- if (tmpObject) {
- currentObject = tmpObject;
- } else {
- job->reject(QString(QLatin1String("Icon not found: %1")).arg(propertyName));
- }
- }
- auto iconName = job->arguments()[QLatin1String("iconName")].toString();
- if (value.name() == iconName) {
- job->resolve();
- } else {
- job->reject("iconName " + iconName + " does not match: " + value.name());
- }
- }
- #endif
- QString SocketApi::buildRegisterPathMessage(const QString &path)
- {
- QFileInfo fi(path);
- QString message = QLatin1String("REGISTER_PATH:");
- message.append(QDir::toNativeSeparators(fi.absoluteFilePath()));
- return message;
- }
- void SocketApiJob::resolve(const QString &response)
- {
- _socketListener->sendMessage(QStringLiteral("RESOLVE|") + _jobId + QLatin1Char('|') + response);
- }
- void SocketApiJob::resolve(const QJsonObject &response)
- {
- resolve(QJsonDocument { response }.toJson());
- }
- void SocketApiJob::reject(const QString &response)
- {
- _socketListener->sendMessage(QStringLiteral("REJECT|") + _jobId + QLatin1Char('|') + response);
- }
- SocketApiJobV2::SocketApiJobV2(const QSharedPointer<SocketListener> &socketListener, const QByteArray &command, const QJsonObject &arguments)
- : _socketListener(socketListener)
- , _command(command)
- , _jobId(arguments[QStringLiteral("id")].toString())
- , _arguments(arguments[QStringLiteral("arguments")].toObject())
- {
- ASSERT(!_jobId.isEmpty())
- }
- void SocketApiJobV2::success(const QJsonObject &response) const
- {
- doFinish(response);
- }
- void SocketApiJobV2::failure(const QString &error) const
- {
- doFinish({ { QStringLiteral("error"), error } });
- }
- void SocketApiJobV2::doFinish(const QJsonObject &obj) const
- {
- _socketListener->sendMessage(_command + QStringLiteral("_RESULT:") + QJsonDocument({ { QStringLiteral("id"), _jobId }, { QStringLiteral("arguments"), obj } }).toJson(QJsonDocument::Compact));
- Q_EMIT finished();
- }
- } // namespace OCC
- #include "socketapi.moc"
|