Skip to content

Commit

Permalink
introduce new jobs to handle lock/unlock of files
Browse files Browse the repository at this point in the history
close #4382

Signed-off-by: Matthieu Gallien <[email protected]>
  • Loading branch information
mgallien committed Apr 26, 2022
1 parent 694cafe commit eed9af2
Show file tree
Hide file tree
Showing 3 changed files with 266 additions and 0 deletions.
2 changes: 2 additions & 0 deletions src/libsync/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,8 @@ set(libsync_SRCS
userstatusconnector.cpp
ocsprofileconnector.h
ocsprofileconnector.cpp
lockfilejobs.h
lockfilejobs.cpp
creds/dummycredentials.h
creds/dummycredentials.cpp
creds/abstractcredentials.h
Expand Down
204 changes: 204 additions & 0 deletions src/libsync/lockfilejobs.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,204 @@
/*
* Copyright (C) by Matthieu Gallien <[email protected]>
*
* 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 "lockfilejobs.h"

#include "account.h"
#include "common/syncjournaldb.h"

#include <QLoggingCategory>
#include <QXmlStreamReader>

namespace OCC {

Q_LOGGING_CATEGORY(lcLockFileJob, "nextcloud.sync.networkjob.lockfile", QtInfoMsg)

LockFileJob::LockFileJob(const AccountPtr account,
SyncJournalDb* const journal,
const QString &path,
const SyncFileItem::LockStatus requestedLockState,
QObject *parent)
: AbstractNetworkJob(account, path, parent)
, _journal(journal)
, _requestedLockState(requestedLockState)
{
}

void LockFileJob::start()
{
qCInfo(lcLockFileJob()) << "start" << path() << _requestedLockState;

QNetworkRequest request;
request.setRawHeader("X-User-Lock", "1");

QByteArray verb;
switch(_requestedLockState)
{
case SyncFileItem::LockStatus::LockedItem:
verb = "LOCK";
break;
case SyncFileItem::LockStatus::UnlockedItem:
verb = "UNLOCK";
break;
}
sendRequest(verb, makeDavUrl(path()), request);

AbstractNetworkJob::start();
}

bool LockFileJob::finished()
{
if (reply()->error() != QNetworkReply::NoError) {
qCInfo(lcLockFileJob()) << "finished with error" << reply()->error() << reply()->errorString();
const auto httpErrorCode = reply()->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
if (httpErrorCode == LOCKED_HTTP_ERROR_CODE) {
const auto record = handleReply();
if (static_cast<SyncFileItem::LockOwnerType>(record._lockOwnerType) == SyncFileItem::LockOwnerType::UserLock) {
Q_EMIT finishedWithError(httpErrorCode, {}, record._lockOwnerDisplayName);
} else {
Q_EMIT finishedWithError(httpErrorCode, {}, record._lockEditorApp);
}
} else {
Q_EMIT finishedWithError(httpErrorCode, reply()->errorString(), {});
}
} else {
qCInfo(lcLockFileJob()) << "success" << path();
handleReply();
Q_EMIT finishedWithoutError();
}
return true;
}

void LockFileJob::setFileRecordLocked(SyncJournalFileRecord &record) const
{
record._locked = (_lockStatus == SyncFileItem::LockStatus::LockedItem);
record._lockOwnerType = static_cast<int>(_lockOwnerType);
record._lockOwnerDisplayName = _userDisplayName;
record._lockOwnerId = _userId;
record._lockEditorApp = _editorName;
record._lockTime = _lockTime;
record._lockTimeout = _lockTimeout;
}

void LockFileJob::resetState()
{
_lockStatus = SyncFileItem::LockStatus::UnlockedItem;
_lockOwnerType = SyncFileItem::LockOwnerType::UserLock;
_userDisplayName.clear();
_editorName.clear();
_userId.clear();
_lockTime = 0;
_lockTimeout = 0;
}

SyncJournalFileRecord LockFileJob::handleReply()
{
const auto xml = reply()->readAll();

QXmlStreamReader reader(xml);

resetState();

while (!reader.atEnd()) {
const auto type = reader.readNext();
const auto name = reader.name().toString();

switch (type) {
case QXmlStreamReader::TokenType::NoToken:
case QXmlStreamReader::TokenType::Invalid:
case QXmlStreamReader::TokenType::DTD:
case QXmlStreamReader::TokenType::EntityReference:
case QXmlStreamReader::TokenType::ProcessingInstruction:
case QXmlStreamReader::TokenType::Comment:
case QXmlStreamReader::TokenType::StartDocument:
case QXmlStreamReader::TokenType::Characters:
case QXmlStreamReader::TokenType::EndDocument:
case QXmlStreamReader::TokenType::EndElement:
break;
case QXmlStreamReader::TokenType::StartElement:
decodeStartElement(name, reader);
break;
}
}

SyncJournalFileRecord record;

if (_lockStatus == SyncFileItem::LockStatus::LockedItem) {
if (_lockOwnerType == SyncFileItem::LockOwnerType::UserLock && _userDisplayName.isEmpty()) {
return record;
}

if (_lockOwnerType == SyncFileItem::LockOwnerType::AppLock && _editorName.isEmpty()) {
return record;
}

if (_userId.isEmpty()) {
return record;
}

if (_lockTime <= 0) {
return record;
}

if (_lockTimeout <= 0) {
return record;
}
}

const auto relativePath = path().mid(1);
if (_journal->getFileRecord(relativePath, &record) && record.isValid()) {
setFileRecordLocked(record);
_journal->setFileRecord(record);
_journal->commit("lock file job");
}

return record;
}

void LockFileJob::decodeStartElement(const QString &name,
QXmlStreamReader &reader)
{
if (name == QStringLiteral("lock")) {
const auto valueText = reader.readElementText();
if (valueText.isEmpty()) {
_lockStatus = SyncFileItem::LockStatus::UnlockedItem;
} else {
bool success = false;
_lockStatus = static_cast<SyncFileItem::LockStatus>(valueText.toInt(&success));
if (!success) {
_lockStatus = SyncFileItem::LockStatus::UnlockedItem;
}
}
} else if (name == QStringLiteral("lock-owner-type")) {
const auto valueText = reader.readElementText();
_lockOwnerType = static_cast<SyncFileItem::LockOwnerType>(valueText.toInt());
} else if (name == QStringLiteral("lock-owner-displayname")) {
const auto valueText = reader.readElementText();
_userDisplayName = valueText;
} else if (name == QStringLiteral("lock-owner")) {
const auto valueText = reader.readElementText();
_userId = valueText;
} else if (name == QStringLiteral("lock-time")) {
const auto valueText = reader.readElementText();
_lockTime = valueText.toLongLong();
} else if (name == QStringLiteral("lock-timeout")) {
const auto valueText = reader.readElementText();
_lockTimeout = valueText.toLongLong();
} else if (name == QStringLiteral("lock-owner-editor")) {
const auto valueText = reader.readElementText();
_editorName = valueText;
}
}

}
60 changes: 60 additions & 0 deletions src/libsync/lockfilejobs.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
#ifndef LOCKFILEJOBS_H
#define LOCKFILEJOBS_H

#include "abstractnetworkjob.h"

#include "syncfileitem.h"

class QXmlStreamReader;

namespace OCC {

class SyncJournalDb;

class OWNCLOUDSYNC_EXPORT LockFileJob : public AbstractNetworkJob
{
Q_OBJECT

public:
static constexpr auto LOCKED_HTTP_ERROR_CODE = 423;

explicit LockFileJob(const AccountPtr account,
SyncJournalDb* const journal,
const QString &path,
const SyncFileItem::LockStatus requestedLockState,
QObject *parent = nullptr);
void start() override;

signals:
void finishedWithError(int httpErrorCode,
const QString &errorString,
const QString &lockOwnerName);
void finishedWithoutError();

private:
bool finished() override;

void setFileRecordLocked(SyncJournalFileRecord &record) const;

SyncJournalFileRecord handleReply();

void resetState();

void decodeStartElement(const QString &name,
QXmlStreamReader &reader);

SyncJournalDb* _journal = nullptr;
SyncFileItem::LockStatus _requestedLockState = SyncFileItem::LockStatus::LockedItem;

SyncFileItem::LockStatus _lockStatus = SyncFileItem::LockStatus::UnlockedItem;
SyncFileItem::LockOwnerType _lockOwnerType = SyncFileItem::LockOwnerType::UserLock;
QString _userDisplayName;
QString _editorName;
QString _userId;
qint64 _lockTime = 0;
qint64 _lockTimeout = 0;
};

}

#endif // LOCKFILEJOBS_H

0 comments on commit eed9af2

Please sign in to comment.