mirror of
https://github.com/laurent22/joplin.git
synced 2025-04-01 21:24:45 +02:00
All: Improved sync locks so that they do not prevent upgrading a sync target
This commit is contained in:
parent
1efe3d3c6a
commit
06ed58b809
packages/lib
Synchronizer.ts
services/synchronizer
@ -443,7 +443,7 @@ export default class Synchronizer {
|
|||||||
const previousE2EE = localInfo.e2ee;
|
const previousE2EE = localInfo.e2ee;
|
||||||
logger.info('Sync target info differs between local and remote - merging infos: ', newInfo.toObject());
|
logger.info('Sync target info differs between local and remote - merging infos: ', newInfo.toObject());
|
||||||
|
|
||||||
await this.lockHandler().acquireLock(LockType.Exclusive, this.appType_, this.clientId_);
|
await this.lockHandler().acquireLock(LockType.Exclusive, this.appType_, this.clientId_, { clearExistingSyncLocksFromTheSameClient: true });
|
||||||
await uploadSyncInfo(this.api(), newInfo);
|
await uploadSyncInfo(this.api(), newInfo);
|
||||||
await saveLocalSyncInfo(newInfo);
|
await saveLocalSyncInfo(newInfo);
|
||||||
await this.lockHandler().releaseLock(LockType.Exclusive, this.appType_, this.clientId_);
|
await this.lockHandler().releaseLock(LockType.Exclusive, this.appType_, this.clientId_);
|
||||||
|
@ -18,6 +18,29 @@ export interface Lock {
|
|||||||
updatedTime?: number;
|
updatedTime?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface AcquireLockOptions {
|
||||||
|
// In theory, a client that tries to acquire an exclusive lock shouldn't
|
||||||
|
// also have a sync lock. It can however happen when the app is closed
|
||||||
|
// before the end of the sync process, and then the user tries to upgrade
|
||||||
|
// the sync target.
|
||||||
|
//
|
||||||
|
// So maybe we could always automatically clear the sync locks (that belongs
|
||||||
|
// to the current client) when acquiring an exclusive lock, but to be safe
|
||||||
|
// we make the behaviour explicit via this option. It is used for example
|
||||||
|
// when migrating a sync target.
|
||||||
|
//
|
||||||
|
// https://discourse.joplinapp.org/t/error-upgrading-to-2-3-3/19549/4?u=laurent
|
||||||
|
clearExistingSyncLocksFromTheSameClient?: boolean;
|
||||||
|
timeoutMs?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
function defaultAcquireLockOptions(): AcquireLockOptions {
|
||||||
|
return {
|
||||||
|
clearExistingSyncLocksFromTheSameClient: false,
|
||||||
|
timeoutMs: 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
interface RefreshTimer {
|
interface RefreshTimer {
|
||||||
id: any;
|
id: any;
|
||||||
inProgress: boolean;
|
inProgress: boolean;
|
||||||
@ -198,7 +221,7 @@ export default class LockHandler {
|
|||||||
return `(${lock.clientType} #${lock.clientId})`;
|
return `(${lock.clientType} #${lock.clientId})`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async acquireExclusiveLock(clientType: string, clientId: string, timeoutMs: number = 0): Promise<Lock> {
|
private async acquireExclusiveLock(clientType: string, clientId: string, options: AcquireLockOptions = null): Promise<Lock> {
|
||||||
// The logic to acquire an exclusive lock, while avoiding race conditions is as follow:
|
// The logic to acquire an exclusive lock, while avoiding race conditions is as follow:
|
||||||
//
|
//
|
||||||
// - Check if there is a lock file present
|
// - Check if there is a lock file present
|
||||||
@ -209,13 +232,18 @@ export default class LockHandler {
|
|||||||
//
|
//
|
||||||
// -If there is no lock file, create one with my identifier and try the whole cycle again to avoid race condition (re-check that the lock file is really mine)-.
|
// -If there is no lock file, create one with my identifier and try the whole cycle again to avoid race condition (re-check that the lock file is really mine)-.
|
||||||
|
|
||||||
|
options = {
|
||||||
|
...defaultAcquireLockOptions(),
|
||||||
|
...options,
|
||||||
|
};
|
||||||
|
|
||||||
const startTime = Date.now();
|
const startTime = Date.now();
|
||||||
|
|
||||||
async function waitForTimeout() {
|
async function waitForTimeout() {
|
||||||
if (!timeoutMs) return false;
|
if (!options.timeoutMs) return false;
|
||||||
|
|
||||||
const elapsed = Date.now() - startTime;
|
const elapsed = Date.now() - startTime;
|
||||||
if (timeoutMs && elapsed < timeoutMs) {
|
if (options.timeoutMs && elapsed < options.timeoutMs) {
|
||||||
await time.sleep(2);
|
await time.sleep(2);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -230,8 +258,12 @@ export default class LockHandler {
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
if (activeSyncLock) {
|
if (activeSyncLock) {
|
||||||
if (await waitForTimeout()) continue;
|
if (options.clearExistingSyncLocksFromTheSameClient && activeSyncLock.clientId === clientId && activeSyncLock.clientType === clientType) {
|
||||||
throw new JoplinError(`Cannot acquire exclusive lock because the following clients have a sync lock on the target: ${this.lockToClientString(activeSyncLock)}`, 'hasSyncLock');
|
await this.releaseLock(LockType.Sync, clientType, clientId);
|
||||||
|
} else {
|
||||||
|
if (await waitForTimeout()) continue;
|
||||||
|
throw new JoplinError(`Cannot acquire exclusive lock because the following clients have a sync lock on the target: ${this.lockToClientString(activeSyncLock)}`, 'hasSyncLock');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (activeExclusiveLock) {
|
if (activeExclusiveLock) {
|
||||||
@ -336,17 +368,22 @@ export default class LockHandler {
|
|||||||
delete this.refreshTimers_[handle];
|
delete this.refreshTimers_[handle];
|
||||||
}
|
}
|
||||||
|
|
||||||
async acquireLock(lockType: LockType, clientType: string, clientId: string, timeoutMs: number = 0): Promise<Lock> {
|
public async acquireLock(lockType: LockType, clientType: string, clientId: string, options: AcquireLockOptions = null): Promise<Lock> {
|
||||||
|
options = {
|
||||||
|
...defaultAcquireLockOptions(),
|
||||||
|
...options,
|
||||||
|
};
|
||||||
|
|
||||||
if (lockType === LockType.Sync) {
|
if (lockType === LockType.Sync) {
|
||||||
return this.acquireSyncLock(clientType, clientId);
|
return this.acquireSyncLock(clientType, clientId);
|
||||||
} else if (lockType === LockType.Exclusive) {
|
} else if (lockType === LockType.Exclusive) {
|
||||||
return this.acquireExclusiveLock(clientType, clientId, timeoutMs);
|
return this.acquireExclusiveLock(clientType, clientId, options);
|
||||||
} else {
|
} else {
|
||||||
throw new Error(`Invalid lock type: ${lockType}`);
|
throw new Error(`Invalid lock type: ${lockType}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async releaseLock(lockType: LockType, clientType: string, clientId: string) {
|
public async releaseLock(lockType: LockType, clientType: string, clientId: string) {
|
||||||
await this.api_.delete(this.lockFilePath({
|
await this.api_.delete(this.lockFilePath({
|
||||||
type: lockType,
|
type: lockType,
|
||||||
clientType: clientType,
|
clientType: clientType,
|
||||||
|
@ -106,7 +106,11 @@ export default class MigrationHandler extends BaseService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.logger().info('MigrationHandler: Acquiring exclusive lock');
|
this.logger().info('MigrationHandler: Acquiring exclusive lock');
|
||||||
const exclusiveLock = await this.lockHandler_.acquireLock(LockType.Exclusive, this.clientType_, this.clientId_, 1000 * 30);
|
const exclusiveLock = await this.lockHandler_.acquireLock(LockType.Exclusive, this.clientType_, this.clientId_, {
|
||||||
|
clearExistingSyncLocksFromTheSameClient: true,
|
||||||
|
timeoutMs: 1000 * 30,
|
||||||
|
});
|
||||||
|
|
||||||
let autoLockError = null;
|
let autoLockError = null;
|
||||||
this.lockHandler_.startAutoLockRefresh(exclusiveLock, (error: any) => {
|
this.lockHandler_.startAutoLockRefresh(exclusiveLock, (error: any) => {
|
||||||
autoLockError = error;
|
autoLockError = error;
|
||||||
|
@ -149,6 +149,23 @@ describe('synchronizer_LockHandler', function() {
|
|||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
it('should ignore locks by same client when trying to acquire exclusive lock', (async () => {
|
||||||
|
const lockHandler = newLockHandler();
|
||||||
|
|
||||||
|
await lockHandler.acquireLock(LockType.Sync, 'desktop', '111');
|
||||||
|
|
||||||
|
await expectThrow(async () => {
|
||||||
|
await lockHandler.acquireLock(LockType.Exclusive, 'desktop', '111', { clearExistingSyncLocksFromTheSameClient: false });
|
||||||
|
}, 'hasSyncLock');
|
||||||
|
|
||||||
|
await expectNotThrow(async () => {
|
||||||
|
await lockHandler.acquireLock(LockType.Exclusive, 'desktop', '111', { clearExistingSyncLocksFromTheSameClient: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
const activeLock = await lockHandler.activeLock(LockType.Exclusive);
|
||||||
|
expect(activeLock.clientId).toBe('111');
|
||||||
|
}));
|
||||||
|
|
||||||
// it('should not have race conditions', (async () => {
|
// it('should not have race conditions', (async () => {
|
||||||
// const lockHandler = newLockHandler();
|
// const lockHandler = newLockHandler();
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user