227 lines
6.4 KiB
TypeScript
227 lines
6.4 KiB
TypeScript
import { describe, it, expect, spyOn, beforeEach, afterEach } from "bun:test"
|
|
import * as spawnWithWindowsHideModule from "../../shared/spawn-with-windows-hide"
|
|
import * as loggerModule from "../../shared/logger"
|
|
import { executeOnCompleteHook } from "./on-complete-hook"
|
|
|
|
describe("executeOnCompleteHook", () => {
|
|
function createStream(text: string): ReadableStream<Uint8Array> | undefined {
|
|
if (text.length === 0) {
|
|
return undefined
|
|
}
|
|
|
|
const encoder = new TextEncoder()
|
|
return new ReadableStream<Uint8Array>({
|
|
start(controller) {
|
|
controller.enqueue(encoder.encode(text))
|
|
controller.close()
|
|
},
|
|
})
|
|
}
|
|
|
|
function createProc(exitCode: number, output?: { stdout?: string; stderr?: string }) {
|
|
return {
|
|
exited: Promise.resolve(exitCode),
|
|
exitCode,
|
|
stdout: createStream(output?.stdout ?? ""),
|
|
stderr: createStream(output?.stderr ?? ""),
|
|
kill: () => {},
|
|
} satisfies ReturnType<typeof spawnWithWindowsHideModule.spawnWithWindowsHide>
|
|
}
|
|
|
|
let logSpy: ReturnType<typeof spyOn<typeof loggerModule, "log">>
|
|
|
|
beforeEach(() => {
|
|
logSpy = spyOn(loggerModule, "log").mockImplementation(() => {})
|
|
})
|
|
|
|
afterEach(() => {
|
|
logSpy.mockRestore()
|
|
})
|
|
|
|
it("executes command with correct env vars", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(createProc(0))
|
|
|
|
try {
|
|
// when
|
|
await executeOnCompleteHook({
|
|
command: "echo test",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
|
|
// then
|
|
expect(spawnSpy).toHaveBeenCalledTimes(1)
|
|
const [args, options] = spawnSpy.mock.calls[0] as Parameters<typeof spawnWithWindowsHideModule.spawnWithWindowsHide>
|
|
|
|
expect(args).toEqual(["sh", "-c", "echo test"])
|
|
expect(options?.env?.SESSION_ID).toBe("session-123")
|
|
expect(options?.env?.EXIT_CODE).toBe("0")
|
|
expect(options?.env?.DURATION_MS).toBe("5000")
|
|
expect(options?.env?.MESSAGE_COUNT).toBe("10")
|
|
expect(options?.stdout).toBe("pipe")
|
|
expect(options?.stderr).toBe("pipe")
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("env var values are strings", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(createProc(0))
|
|
|
|
try {
|
|
// when
|
|
await executeOnCompleteHook({
|
|
command: "echo test",
|
|
sessionId: "session-123",
|
|
exitCode: 1,
|
|
durationMs: 12345,
|
|
messageCount: 42,
|
|
})
|
|
|
|
// then
|
|
const [_, options] = spawnSpy.mock.calls[0] as Parameters<typeof spawnWithWindowsHideModule.spawnWithWindowsHide>
|
|
|
|
expect(options?.env?.EXIT_CODE).toBe("1")
|
|
expect(options?.env?.EXIT_CODE).toBeTypeOf("string")
|
|
expect(options?.env?.DURATION_MS).toBe("12345")
|
|
expect(options?.env?.DURATION_MS).toBeTypeOf("string")
|
|
expect(options?.env?.MESSAGE_COUNT).toBe("42")
|
|
expect(options?.env?.MESSAGE_COUNT).toBeTypeOf("string")
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("empty command string is no-op", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(createProc(0))
|
|
|
|
try {
|
|
// when
|
|
await executeOnCompleteHook({
|
|
command: "",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
|
|
// then
|
|
expect(spawnSpy).not.toHaveBeenCalled()
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("whitespace-only command is no-op", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(createProc(0))
|
|
|
|
try {
|
|
// when
|
|
await executeOnCompleteHook({
|
|
command: " ",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
|
|
// then
|
|
expect(spawnSpy).not.toHaveBeenCalled()
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("command failure logs warning but does not throw", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(createProc(1))
|
|
|
|
try {
|
|
// when
|
|
expect(
|
|
executeOnCompleteHook({
|
|
command: "false",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
).resolves.toBeUndefined()
|
|
|
|
// then
|
|
const warningCall = logSpy.mock.calls.find(
|
|
(call) => call[0] === "On-complete hook exited with non-zero code"
|
|
)
|
|
expect(warningCall).toBeDefined()
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("spawn error logs warning but does not throw", async () => {
|
|
// given
|
|
const spawnError = new Error("Command not found")
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockImplementation(() => {
|
|
throw spawnError
|
|
})
|
|
|
|
try {
|
|
// when
|
|
expect(
|
|
executeOnCompleteHook({
|
|
command: "nonexistent-command",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
).resolves.toBeUndefined()
|
|
|
|
// then
|
|
const errorCall = logSpy.mock.calls.find(
|
|
(call) => call[0] === "Failed to execute on-complete hook"
|
|
)
|
|
expect(errorCall).toBeDefined()
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
|
|
it("hook stdout and stderr are logged to file logger", async () => {
|
|
// given
|
|
const spawnSpy = spyOn(spawnWithWindowsHideModule, "spawnWithWindowsHide").mockReturnValue(
|
|
createProc(0, { stdout: "hook output\n", stderr: "hook warning\n" })
|
|
)
|
|
|
|
try {
|
|
// when
|
|
await executeOnCompleteHook({
|
|
command: "echo test",
|
|
sessionId: "session-123",
|
|
exitCode: 0,
|
|
durationMs: 5000,
|
|
messageCount: 10,
|
|
})
|
|
|
|
// then
|
|
const stdoutCall = logSpy.mock.calls.find(
|
|
(call) => call[0] === "On-complete hook stdout"
|
|
)
|
|
const stderrCall = logSpy.mock.calls.find(
|
|
(call) => call[0] === "On-complete hook stderr"
|
|
)
|
|
|
|
expect(stdoutCall?.[1]).toEqual({ command: "echo test", stdout: "hook output" })
|
|
expect(stderrCall?.[1]).toEqual({ command: "echo test", stderr: "hook warning" })
|
|
} finally {
|
|
spawnSpy.mockRestore()
|
|
}
|
|
})
|
|
})
|