-
Notifications
You must be signed in to change notification settings - Fork 941
Expand file tree
/
Copy pathe2eefoldermanager.cpp
More file actions
176 lines (141 loc) · 6.3 KB
/
e2eefoldermanager.cpp
File metadata and controls
176 lines (141 loc) · 6.3 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
// SPDX-FileCopyrightText: 2026 Nextcloud GmbH and Nextcloud contributors
// SPDX-License-Identifier: GPL-2.0-or-later
#include "e2eefoldermanager.h"
#include "accountmanager.h"
#include "clientsideencryption.h"
#include "folderman.h"
#include "folder.h"
#include <QLoggingCategory>
namespace OCC {
Q_LOGGING_CATEGORY(lcE2eFolderManager, "nextcloud.gui.e2efoldermanager", QtInfoMsg)
E2EFolderManager *E2EFolderManager::_instance = nullptr;
E2EFolderManager *E2EFolderManager::instance()
{
if (!_instance) {
_instance = new E2EFolderManager();
}
return _instance;
}
E2EFolderManager::E2EFolderManager(QObject *parent)
: QObject(parent)
{
const auto accounts = AccountManager::instance()->accounts();
for (const auto &accountState : accounts) {
if (accountState && accountState->account() && accountState->account()->e2e()) {
connectE2eSignals(accountState->account());
}
}
connect(AccountManager::instance(), &AccountManager::accountAdded,
this, &E2EFolderManager::slotAccountAdded);
}
E2EFolderManager::~E2EFolderManager()
{
_instance = nullptr;
}
void E2EFolderManager::slotAccountAdded(AccountState *accountState)
{
if (accountState && accountState->account() && accountState->account()->e2e()) {
connectE2eSignals(accountState->account());
}
}
void E2EFolderManager::connectE2eSignals(const AccountPtr &account)
{
if (!account || !account->e2e()) {
return;
}
connect(account->e2e(), &ClientSideEncryption::initializationFinished,
this, &E2EFolderManager::slotE2eInitializationFinished, Qt::UniqueConnection);
if (account->e2e()->isInitialized()) {
restoreE2eFoldersForAccount(account);
}
}
void E2EFolderManager::slotE2eInitializationFinished()
{
qCDebug(lcE2eFolderManager) << "E2E initialization finished, restoring blacklisted E2E folders";
auto *e2e = qobject_cast<ClientSideEncryption *>(sender());
if (!e2e) {
qCWarning(lcE2eFolderManager) << "slotE2eInitializationFinished called but sender is not ClientSideEncryption";
return;
}
const auto accounts = AccountManager::instance()->accounts();
for (const auto &accountState : accounts) {
if (accountState->account()->e2e() == e2e) {
restoreE2eFoldersForAccount(accountState->account());
break;
}
}
}
void E2EFolderManager::restoreE2eFoldersForAccount(const AccountPtr &account)
{
if (!account || !account->e2e() || !account->e2e()->isInitialized()) {
qCDebug(lcE2eFolderManager) << "Cannot restore folders - account or E2E not ready";
return;
}
qCDebug(lcE2eFolderManager) << "Restoring E2E folders for account:" << account->displayName();
auto *folderMan = FolderMan::instance();
const auto folders = folderMan->map();
int foldersProcessed = 0;
for (const auto &folder : folders) {
if (folder->accountState()->account() != account) {
continue;
}
bool ok = false;
const auto foldersToRemoveFromBlacklist = folder->journalDb()->getSelectiveSyncList(
SyncJournalDb::SelectiveSyncE2eFoldersToRemoveFromBlacklist, &ok);
if (foldersToRemoveFromBlacklist.isEmpty()) {
continue;
}
qCDebug(lcE2eFolderManager) << "Found E2E folders to restore for" << folder->alias()
<< ":" << foldersToRemoveFromBlacklist;
auto blackList = folder->journalDb()->getSelectiveSyncList(SyncJournalDb::SelectiveSyncBlackList, &ok);
qCDebug(lcE2eFolderManager) << "Current blacklist:" << blackList;
const auto blackListSize = blackList.size();
for (const auto &pathToRemoveFromBlackList : foldersToRemoveFromBlacklist) {
blackList.removeAll(pathToRemoveFromBlackList);
}
if (blackList.size() != blackListSize) {
if (folder->isSyncRunning()) {
qCDebug(lcE2eFolderManager) << "Folder is syncing, terminating to prevent E2E folder deletion";
folderTerminateSyncAndUpdateBlackList(blackList, folder, foldersToRemoveFromBlacklist);
++foldersProcessed;
continue;
}
folder->journalDb()->setSelectiveSyncList(SyncJournalDb::SelectiveSyncBlackList, blackList);
folder->journalDb()->setSelectiveSyncList(SyncJournalDb::SelectiveSyncE2eFoldersToRemoveFromBlacklist, {});
for (const auto &pathToRemoteDiscover : foldersToRemoveFromBlacklist) {
folder->journalDb()->schedulePathForRemoteDiscovery(pathToRemoteDiscover);
}
folderMan->scheduleFolder(folder);
++foldersProcessed;
}
}
if (foldersProcessed > 0) {
qCDebug(lcE2eFolderManager) << "Restored E2E folders for" << foldersProcessed << "sync folders";
} else {
qCDebug(lcE2eFolderManager) << "No E2E folders needed restoration";
}
}
void E2EFolderManager::folderTerminateSyncAndUpdateBlackList(const QStringList &blackList, OCC::Folder *folder, const QStringList &foldersToRemoveFromBlacklist)
{
if (_folderConnections.contains(folder->alias())) {
qCWarning(lcE2eFolderManager) << "Folder " << folder->alias() << "is already terminating the sync.";
return;
}
// in case sync is already running - terminate it and start a new one
const QMetaObject::Connection syncTerminatedConnection = connect(folder, &Folder::syncFinished, this, [this, blackList, folder, foldersToRemoveFromBlacklist]() {
const auto foundConnectionIt = _folderConnections.find(folder->alias());
if (foundConnectionIt != _folderConnections.end()) {
disconnect(*foundConnectionIt);
_folderConnections.erase(foundConnectionIt);
}
folder->journalDb()->setSelectiveSyncList(SyncJournalDb::SelectiveSyncBlackList, blackList);
folder->journalDb()->setSelectiveSyncList(SyncJournalDb::SelectiveSyncE2eFoldersToRemoveFromBlacklist, {});
for (const auto &pathToRemoteDiscover : foldersToRemoveFromBlacklist) {
folder->journalDb()->schedulePathForRemoteDiscovery(pathToRemoteDiscover);
}
FolderMan::instance()->scheduleFolder(folder);
});
_folderConnections.insert(folder->alias(), syncTerminatedConnection);
folder->slotTerminateSync();
}
} // namespace OCC