210 lines
No EOL
6.3 KiB
TypeScript
210 lines
No EOL
6.3 KiB
TypeScript
import { Logger } from "../tracing/logger";
|
|
import type { RelativePath } from "../persistence/database";
|
|
import { Locks } from "./locks";
|
|
|
|
describe("withLock", () => {
|
|
const testPath: RelativePath = "test/document/path";
|
|
const testPath2: RelativePath = "test/document/path2";
|
|
const logger = new Logger();
|
|
|
|
// eslint-disable-next-line @typescript-eslint/init-declarations
|
|
let locks: Locks<RelativePath>;
|
|
|
|
beforeEach(() => {
|
|
locks = new Locks<RelativePath>(logger);
|
|
});
|
|
|
|
test("should execute function with single key lock", async () => {
|
|
let executionCount = 0;
|
|
const result = await locks.withLock(testPath, () => {
|
|
executionCount++;
|
|
return "success";
|
|
});
|
|
|
|
expect(result).toBe("success");
|
|
expect(executionCount).toBe(1);
|
|
});
|
|
|
|
test("should execute async function with single key lock", async () => {
|
|
let executionCount = 0;
|
|
const result = await locks.withLock(testPath, async () => {
|
|
executionCount++;
|
|
await new Promise(resolve => setTimeout(resolve, 10));
|
|
return "async-success";
|
|
});
|
|
|
|
expect(result).toBe("async-success");
|
|
expect(executionCount).toBe(1);
|
|
});
|
|
|
|
test("should execute function with multiple key locks", async () => {
|
|
let executionCount = 0;
|
|
const result = await locks.withLock([testPath, testPath2], () => {
|
|
executionCount++;
|
|
return "multi-success";
|
|
});
|
|
|
|
expect(result).toBe("multi-success");
|
|
expect(executionCount).toBe(1);
|
|
});
|
|
|
|
test("should sort multiple keys to prevent deadlocks", async () => {
|
|
const executionOrder: string[] = [];
|
|
|
|
// Start two concurrent operations with keys in different orders
|
|
const promise1 = locks.withLock([testPath2, testPath], async () => {
|
|
executionOrder.push("operation1-start");
|
|
await new Promise(resolve => setTimeout(resolve, 50));
|
|
executionOrder.push("operation1-end");
|
|
return "result1";
|
|
});
|
|
|
|
const promise2 = locks.withLock([testPath, testPath2], async () => {
|
|
executionOrder.push("operation2-start");
|
|
await new Promise(resolve => setTimeout(resolve, 50));
|
|
executionOrder.push("operation2-end");
|
|
return "result2";
|
|
});
|
|
|
|
const [result1, result2] = await Promise.all([promise1, promise2]);
|
|
|
|
expect(result1).toBe("result1");
|
|
expect(result2).toBe("result2");
|
|
// One operation should complete entirely before the other starts
|
|
expect(executionOrder).toEqual([
|
|
"operation1-start",
|
|
"operation1-end",
|
|
"operation2-start",
|
|
"operation2-end"
|
|
]);
|
|
});
|
|
|
|
test("should serialize access to same key", async () => {
|
|
const executionOrder: string[] = [];
|
|
|
|
const promise1 = locks.withLock(testPath, async () => {
|
|
executionOrder.push("operation1-start");
|
|
await new Promise(resolve => setTimeout(resolve, 50));
|
|
executionOrder.push("operation1-end");
|
|
return "result1";
|
|
});
|
|
|
|
const promise2 = locks.withLock(testPath, async () => {
|
|
executionOrder.push("operation2-start");
|
|
await new Promise(resolve => setTimeout(resolve, 30));
|
|
executionOrder.push("operation2-end");
|
|
return "result2";
|
|
});
|
|
|
|
const [result1, result2] = await Promise.all([promise1, promise2]);
|
|
|
|
expect(result1).toBe("result1");
|
|
expect(result2).toBe("result2");
|
|
expect(executionOrder).toEqual([
|
|
"operation1-start",
|
|
"operation1-end",
|
|
"operation2-start",
|
|
"operation2-end"
|
|
]);
|
|
});
|
|
|
|
test("should allow concurrent access to different keys", async () => {
|
|
const executionOrder: string[] = [];
|
|
|
|
const promise1 = locks.withLock(testPath, async () => {
|
|
executionOrder.push("operation1-start");
|
|
await new Promise(resolve => setTimeout(resolve, 50));
|
|
executionOrder.push("operation1-end");
|
|
return "result1";
|
|
});
|
|
|
|
const promise2 = locks.withLock(testPath2, async () => {
|
|
executionOrder.push("operation2-start");
|
|
await new Promise(resolve => setTimeout(resolve, 30));
|
|
executionOrder.push("operation2-end");
|
|
return "result2";
|
|
});
|
|
|
|
const [result1, result2] = await Promise.all([promise1, promise2]);
|
|
|
|
expect(result1).toBe("result1");
|
|
expect(result2).toBe("result2");
|
|
// Both operations should run concurrently
|
|
expect(executionOrder[0]).toBe("operation1-start");
|
|
expect(executionOrder[1]).toBe("operation2-start");
|
|
});
|
|
|
|
test("should release locks even if function throws", async () => {
|
|
const error = new Error("test error");
|
|
|
|
await expect(locks.withLock(testPath, () => {
|
|
throw error;
|
|
})).rejects.toThrow("test error");
|
|
|
|
// Lock should be released, allowing another operation
|
|
const result = await locks.withLock(testPath, () => "success-after-error");
|
|
expect(result).toBe("success-after-error");
|
|
});
|
|
|
|
test("should release locks even if async function throws", async () => {
|
|
const error = new Error("async test error");
|
|
|
|
await expect(locks.withLock(testPath, async () => {
|
|
await new Promise(resolve => setTimeout(resolve, 10));
|
|
throw error;
|
|
})).rejects.toThrow("async test error");
|
|
|
|
// Lock should be released, allowing another operation
|
|
const result = await locks.withLock(testPath, () => "success-after-async-error");
|
|
expect(result).toBe("success-after-async-error");
|
|
});
|
|
|
|
test("should handle empty array of keys", async () => {
|
|
const result = await locks.withLock([], () => "empty-keys");
|
|
expect(result).toBe("empty-keys");
|
|
});
|
|
|
|
test("should maintain FIFO order for multiple waiters", async () => {
|
|
const executionOrder: string[] = [];
|
|
|
|
// Start first operation that holds the lock
|
|
const firstPromise = locks.withLock(testPath, async () => {
|
|
executionOrder.push("first-start");
|
|
await new Promise(resolve => setTimeout(resolve, 100));
|
|
executionOrder.push("first-end");
|
|
return "first";
|
|
});
|
|
|
|
// Small delay to ensure first operation starts
|
|
await new Promise(resolve => setTimeout(resolve, 10));
|
|
|
|
// Queue second and third operations
|
|
const secondPromise = locks.withLock(testPath, async () => {
|
|
executionOrder.push("second-start");
|
|
await new Promise(resolve => setTimeout(resolve, 30));
|
|
executionOrder.push("second-end");
|
|
return "second";
|
|
});
|
|
|
|
const thirdPromise = locks.withLock(testPath, async () => {
|
|
executionOrder.push("third-start");
|
|
await new Promise(resolve => setTimeout(resolve, 20));
|
|
executionOrder.push("third-end");
|
|
return "third";
|
|
});
|
|
|
|
const [first, second, third] = await Promise.all([firstPromise, secondPromise, thirdPromise]);
|
|
|
|
expect(first).toBe("first");
|
|
expect(second).toBe("second");
|
|
expect(third).toBe("third");
|
|
expect(executionOrder).toEqual([
|
|
"first-start",
|
|
"first-end",
|
|
"second-start",
|
|
"second-end",
|
|
"third-start",
|
|
"third-end"
|
|
]);
|
|
});
|
|
}); |