Skip to content

Commit

Permalink
test: update the mock to handle exclusive locks
Browse files Browse the repository at this point in the history
  • Loading branch information
erikian committed Sep 8, 2023
1 parent 03851f5 commit 666cada
Show file tree
Hide file tree
Showing 2 changed files with 149 additions and 14 deletions.
141 changes: 127 additions & 14 deletions tests/setup.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { mocked } from 'jest-mock';
import { configure as mobxConfigure } from 'mobx';

import { ElectronFiddleMock } from './mocks/mocks';
import { getOrCreateMapValue } from './utils';

enzymeConfigure({ adapter: new Adapter() });

Expand Down Expand Up @@ -76,47 +77,159 @@ delete (window as any).localStorage;
window.navigator = window.navigator ?? {};
(window.navigator.clipboard as any) = {};

type MockLock = Lock & {
abortController: AbortController;
};

class FakeNavigatorLocks implements LockManager {
locks = {
held: new Set<Lock>(),
pending: new Set<Lock>(),
held: new Map<string, Map<LockMode, Set<MockLock>>>(),
};

query = async () => {
const result = {
held: [...this.locks.held],
pending: [...this.locks.pending],
held: [...this.locks.held.values()].reduce((acc, item) => {
acc.push(...[...item.get('exclusive')!.values()]);
acc.push(...[...item.get('shared')!.values()]);

return acc;
}, [] as MockLock[]),
};

return result as LockManagerSnapshot;
};

/**
* WIP. Right now, this is a **very** naive mock that will just happily grant a shared lock when one is requested,
* but I'll add some bookkeeping and expand it to cover the exclusive lock case as well.
*
* @TODO remove this comment
*/
request = (async (...args: Parameters<LockManager['request']>) => {
const [
name,
options = {
ifAvailable: false,
mode: 'exclusive',
steal: false,
},
cb,
] = args;

const { mode } = options;
const { ifAvailable, mode, steal } = options;

const lock = {
name,
mode,
abortController: new AbortController(),
} as MockLock;

const heldLocksWithSameName = getOrCreateMapValue(
this.locks.held,
name,
new Map<LockMode, Set<MockLock>>(),
);

const exclusiveLocksWithSameName = getOrCreateMapValue(
heldLocksWithSameName,
'exclusive',
new Set<MockLock>(),
);

const lock = { name, mode, cb } as Lock;
const sharedLocksWithSameName = getOrCreateMapValue(
heldLocksWithSameName,
'shared',
new Set<MockLock>(),
);

if (mode === 'shared') {
this.locks.held.add(lock);
sharedLocksWithSameName.add(lock);

await cb(lock);
try {
await cb(lock);
} finally {
sharedLocksWithSameName.delete(lock);
}

return;
}

// exclusive lock

// no locks with this name -> grant an exclusive lock
if (
exclusiveLocksWithSameName.size === 0 &&
sharedLocksWithSameName.size === 0
) {
exclusiveLocksWithSameName.add(lock);

try {
await cb(lock);
} finally {
exclusiveLocksWithSameName.delete(lock);
}

return;
}

// steal any currently held locks
if (steal) {
for (const lock of sharedLocksWithSameName) {
lock.abortController.abort();
}

for (const lock of exclusiveLocksWithSameName) {
lock.abortController.abort();
}

sharedLocksWithSameName.clear();
exclusiveLocksWithSameName.clear();

exclusiveLocksWithSameName.add(lock);

try {
await cb(lock);
} finally {
exclusiveLocksWithSameName.delete(lock);
}

return;
}

// run the callback without waiting for the lock to be released
if (ifAvailable) {
// just run the callback without waiting for it
cb(null);

return;
}

// @TODO add the lock to the list of pending locks?

// it's an exclusive lock, so there's only one value
const currentLock = exclusiveLocksWithSameName.values().next()
.value as MockLock;

const { abortController: currentLockAbortController } = currentLock;

// wait for the current lock to be released
await new Promise<void>((resolve, reject) => {
currentLockAbortController.signal.onabort = () => resolve();

const { abortController: pendingLockAbortController } = lock;

// this allows the locking mechanism to release this lock
pendingLockAbortController.signal.onabort = () => reject();
});

// clear the exclusive locks
exclusiveLocksWithSameName.clear();

// grant our lock
exclusiveLocksWithSameName.add(lock);

try {
// run the callback
await cb(lock);
} finally {
exclusiveLocksWithSameName.delete(lock);
}

return;
}) as LockManager['request'];
}

Expand Down
22 changes: 22 additions & 0 deletions tests/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -147,3 +147,25 @@ export function emitEvent(type: FiddleEvent, ...args: any[]) {
},
);
}

export function getOrCreateMapValue<T extends Map<unknown, unknown>>(
map: T,
key: MapKey<T>,
fallbackValue: MapValue<T>,
): MapValue<T> {
if (!map.has(key)) {
map.set(key, fallbackValue);

return fallbackValue;
}

return map.get(key) as MapValue<T>;
}

type MapKey<T extends Map<unknown, unknown>> = T extends Map<infer I, unknown>
? I
: never;

type MapValue<T extends Map<unknown, unknown>> = T extends Map<unknown, infer I>
? I
: never;

0 comments on commit 666cada

Please sign in to comment.