mirror of
http://192.168.0.88:13333/lywsvip/openclaw-zero-token.git
synced 2026-05-15 06:20:41 +08:00
Major upgrade from e26988a38 to upstream v2026.3.28 (f9b107928).
Key changes:
- Upstream src/, ui/, extensions/ (89 bundled extensions)
- Zero-token web providers preserved in src/zero-token/
- AskOnce plugin restored and registered as CLI command
- Added missing packages: @anthropic-ai/vertex-sdk, @modelcontextprotocol/sdk
- Fixed tsconfig rootDir, skipLibCheck for plugin-sdk DTS build
- Added askonce to bundled plugin metadata and package.json exports
- Fixed AskOnce CLI command registration (missing commands metadata)
- Restored AskOnce adapter imports (correct 5-level relative paths)
- Removed stale migration artifacts from root directory
105 lines
2.9 KiB
TypeScript
105 lines
2.9 KiB
TypeScript
import { describe, expect, it, vi } from "vitest";
|
|
import { runTasksWithConcurrency } from "./run-with-concurrency.js";
|
|
|
|
describe("runTasksWithConcurrency", () => {
|
|
it("preserves task order with bounded worker count", async () => {
|
|
const flushMicrotasks = async () => {
|
|
await Promise.resolve();
|
|
await Promise.resolve();
|
|
};
|
|
let running = 0;
|
|
let peak = 0;
|
|
const resolvers: Array<(() => void) | undefined> = [];
|
|
const tasks = [0, 1, 2, 3].map((index) => async (): Promise<number> => {
|
|
running += 1;
|
|
peak = Math.max(peak, running);
|
|
await new Promise<void>((resolve) => {
|
|
resolvers[index] = resolve;
|
|
});
|
|
running -= 1;
|
|
return index + 1;
|
|
});
|
|
|
|
const resultPromise = runTasksWithConcurrency({ tasks, limit: 2 });
|
|
await flushMicrotasks();
|
|
expect(typeof resolvers[0]).toBe("function");
|
|
expect(typeof resolvers[1]).toBe("function");
|
|
|
|
resolvers[1]?.();
|
|
await flushMicrotasks();
|
|
expect(typeof resolvers[2]).toBe("function");
|
|
|
|
resolvers[0]?.();
|
|
await flushMicrotasks();
|
|
expect(typeof resolvers[3]).toBe("function");
|
|
|
|
resolvers[2]?.();
|
|
resolvers[3]?.();
|
|
|
|
const result = await resultPromise;
|
|
expect(result.hasError).toBe(false);
|
|
expect(result.firstError).toBeUndefined();
|
|
expect(result.results).toEqual([1, 2, 3, 4]);
|
|
expect(peak).toBeLessThanOrEqual(2);
|
|
});
|
|
|
|
it("stops scheduling after first failure in stop mode", async () => {
|
|
const err = new Error("boom");
|
|
const seen: number[] = [];
|
|
const tasks = [
|
|
async () => {
|
|
seen.push(0);
|
|
return 10;
|
|
},
|
|
async () => {
|
|
seen.push(1);
|
|
throw err;
|
|
},
|
|
async () => {
|
|
seen.push(2);
|
|
return 30;
|
|
},
|
|
];
|
|
|
|
const result = await runTasksWithConcurrency({
|
|
tasks,
|
|
limit: 1,
|
|
errorMode: "stop",
|
|
});
|
|
expect(result.hasError).toBe(true);
|
|
expect(result.firstError).toBe(err);
|
|
expect(result.results[0]).toBe(10);
|
|
expect(result.results[2]).toBeUndefined();
|
|
expect(seen).toEqual([0, 1]);
|
|
});
|
|
|
|
it("continues after failures and reports the first one", async () => {
|
|
const firstErr = new Error("first");
|
|
const onTaskError = vi.fn();
|
|
const tasks = [
|
|
async () => {
|
|
throw firstErr;
|
|
},
|
|
async () => 20,
|
|
async () => {
|
|
throw new Error("second");
|
|
},
|
|
async () => 40,
|
|
];
|
|
|
|
const result = await runTasksWithConcurrency({
|
|
tasks,
|
|
limit: 1,
|
|
errorMode: "continue",
|
|
onTaskError,
|
|
});
|
|
expect(result.hasError).toBe(true);
|
|
expect(result.firstError).toBe(firstErr);
|
|
expect(result.results[1]).toBe(20);
|
|
expect(result.results[3]).toBe(40);
|
|
expect(onTaskError).toHaveBeenCalledTimes(2);
|
|
expect(onTaskError).toHaveBeenNthCalledWith(1, firstErr, 0);
|
|
expect(onTaskError).toHaveBeenNthCalledWith(2, expect.any(Error), 2);
|
|
});
|
|
});
|