mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-06 05:51:15 +08:00
160 lines
5.2 KiB
TypeScript
160 lines
5.2 KiB
TypeScript
// Tests typing mode persistence across session updates and reply turns.
|
|
import { describe, it, expect, vi, beforeEach, afterEach, type Mock } from "vitest";
|
|
import { MAX_TIMER_TIMEOUT_MS } from "../../shared/number-coercion.js";
|
|
import { createTypingController } from "./typing.js";
|
|
|
|
describe("typing persistence bug fix", () => {
|
|
let onReplyStartSpy: Mock;
|
|
let onCleanupSpy: Mock;
|
|
let controller: ReturnType<typeof createTypingController>;
|
|
|
|
beforeEach(() => {
|
|
vi.useFakeTimers();
|
|
onReplyStartSpy = vi.fn();
|
|
onCleanupSpy = vi.fn();
|
|
|
|
controller = createTypingController({
|
|
onReplyStart: onReplyStartSpy,
|
|
onCleanup: onCleanupSpy,
|
|
typingIntervalSeconds: 6,
|
|
log: vi.fn(),
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
vi.useRealTimers();
|
|
});
|
|
|
|
it("should NOT restart typing after markRunComplete is called", async () => {
|
|
// Start typing normally
|
|
await controller.startTypingLoop();
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
// Mark run as complete (but not yet dispatch idle)
|
|
controller.markRunComplete();
|
|
|
|
// Advance time to trigger the typing interval (6 seconds)
|
|
vi.advanceTimersByTime(6000);
|
|
|
|
// BUG: The typing loop should NOT call onReplyStart again
|
|
// because the run is already complete
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
expect(onReplyStartSpy).not.toHaveBeenCalledTimes(2);
|
|
});
|
|
|
|
it("allows visible-delivery typing after run and dispatcher idle", async () => {
|
|
const lateController = createTypingController({
|
|
onReplyStart: onReplyStartSpy,
|
|
onCleanup: onCleanupSpy,
|
|
typingIntervalSeconds: 6,
|
|
log: vi.fn(),
|
|
});
|
|
|
|
lateController.markRunComplete();
|
|
lateController.markDispatchIdle();
|
|
|
|
await lateController.startTypingForVisibleDelivery();
|
|
await lateController.startTypingForVisibleDelivery();
|
|
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
expect(onCleanupSpy).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("keeps typing alive while keepalive ticks continue during long runs", async () => {
|
|
const longRunCleanupSpy = vi.fn();
|
|
const longRunController = createTypingController({
|
|
onReplyStart: onReplyStartSpy,
|
|
onCleanup: longRunCleanupSpy,
|
|
typingIntervalSeconds: 6,
|
|
typingTtlMs: 10_000,
|
|
log: vi.fn(),
|
|
});
|
|
|
|
await longRunController.startTypingLoop();
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
await vi.advanceTimersByTimeAsync(6000);
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(2);
|
|
|
|
await vi.advanceTimersByTimeAsync(5000);
|
|
expect(longRunCleanupSpy).not.toHaveBeenCalled();
|
|
|
|
longRunController.cleanup();
|
|
expect(longRunCleanupSpy).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("should stop typing when both runComplete and dispatchIdle are true", async () => {
|
|
// Start typing
|
|
await controller.startTypingLoop();
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
// Mark run complete
|
|
controller.markRunComplete();
|
|
expect(onCleanupSpy).not.toHaveBeenCalled();
|
|
|
|
// Mark dispatch idle - should trigger cleanup
|
|
controller.markDispatchIdle();
|
|
expect(onCleanupSpy).toHaveBeenCalledTimes(1);
|
|
|
|
// After cleanup, typing interval should not restart typing
|
|
vi.advanceTimersByTime(6000);
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1); // Still only the initial call
|
|
});
|
|
|
|
it("should prevent typing restart even if cleanup is delayed", async () => {
|
|
// Start typing
|
|
await controller.startTypingLoop();
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
// Mark run complete (but dispatch not idle yet - simulating cleanup delay)
|
|
controller.markRunComplete();
|
|
|
|
// Multiple typing intervals should NOT restart typing
|
|
vi.advanceTimersByTime(6000); // First interval
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
vi.advanceTimersByTime(6000); // Second interval
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
vi.advanceTimersByTime(6000); // Third interval
|
|
expect(onReplyStartSpy).toHaveBeenCalledTimes(1);
|
|
|
|
// Eventually dispatch becomes idle and triggers cleanup
|
|
controller.markDispatchIdle();
|
|
expect(onCleanupSpy).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("returns an inert controller when typing callbacks are absent", async () => {
|
|
const inert = createTypingController({});
|
|
|
|
await inert.onReplyStart();
|
|
await inert.startTypingLoop();
|
|
await inert.startTypingOnText("hello");
|
|
inert.refreshTypingTtl();
|
|
inert.markRunComplete();
|
|
inert.markDispatchIdle();
|
|
inert.cleanup();
|
|
|
|
expect(inert.isActive()).toBe(false);
|
|
expect(vi.getTimerCount()).toBe(0);
|
|
});
|
|
|
|
it("clamps oversized typing interval and TTL timers", async () => {
|
|
const setIntervalSpy = vi.spyOn(globalThis, "setInterval");
|
|
const setTimeoutSpy = vi.spyOn(globalThis, "setTimeout");
|
|
const boundedController = createTypingController({
|
|
onReplyStart: onReplyStartSpy,
|
|
onCleanup: onCleanupSpy,
|
|
typingIntervalSeconds: Number.MAX_SAFE_INTEGER,
|
|
typingTtlMs: Number.MAX_SAFE_INTEGER,
|
|
log: vi.fn(),
|
|
});
|
|
|
|
await boundedController.startTypingLoop();
|
|
|
|
expect(setIntervalSpy).toHaveBeenCalledWith(expect.any(Function), MAX_TIMER_TIMEOUT_MS);
|
|
expect(setTimeoutSpy).toHaveBeenCalledWith(expect.any(Function), MAX_TIMER_TIMEOUT_MS);
|
|
boundedController.cleanup();
|
|
});
|
|
});
|