mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-16 09:33:24 +00:00
161 lines
6.3 KiB
TypeScript
161 lines
6.3 KiB
TypeScript
import { Effect, Option, Ref, Scope, Stream } from "effect"
|
|
import type { Headers } from "effect/unstable/http"
|
|
import * as CassetteService from "./cassette"
|
|
import { canonicalizeJson, decodeJson, safeText } from "./matching"
|
|
import { makeReplayState, resolveAutoMode } from "./recorder"
|
|
import type { RecordReplayMode } from "./effect"
|
|
import { redactUrl } from "./redaction"
|
|
import { defaults, type Redactor } from "./redactor"
|
|
import { webSocketInteractions, type CassetteMetadata, type WebSocketFrame } from "./schema"
|
|
|
|
export interface WebSocketRequest {
|
|
readonly url: string
|
|
readonly headers: Headers.Headers
|
|
}
|
|
|
|
export interface WebSocketConnection<E> {
|
|
readonly sendText: (message: string) => Effect.Effect<void, E>
|
|
readonly messages: Stream.Stream<string | Uint8Array, E>
|
|
readonly close: Effect.Effect<void>
|
|
}
|
|
|
|
export interface WebSocketExecutor<E> {
|
|
readonly open: (request: WebSocketRequest) => Effect.Effect<WebSocketConnection<E>, E>
|
|
}
|
|
|
|
export interface WebSocketRecordReplayOptions<E> {
|
|
readonly name: string
|
|
readonly mode?: RecordReplayMode
|
|
readonly metadata?: CassetteMetadata
|
|
readonly cassette: CassetteService.Interface
|
|
readonly live: WebSocketExecutor<E>
|
|
readonly redactor?: Redactor
|
|
readonly compareClientMessagesAsJson?: boolean
|
|
}
|
|
|
|
const headersRecord = (headers: Headers.Headers): Record<string, string> =>
|
|
Object.fromEntries(
|
|
Object.entries(headers as Record<string, unknown>).filter(
|
|
(entry): entry is [string, string] => typeof entry[1] === "string",
|
|
),
|
|
)
|
|
|
|
const encodeFrame = (message: string | Uint8Array): WebSocketFrame =>
|
|
typeof message === "string"
|
|
? { kind: "text", body: message }
|
|
: { kind: "binary", body: Buffer.from(message).toString("base64"), bodyEncoding: "base64" }
|
|
|
|
const decodeFrameMessage = (frame: WebSocketFrame): string | Uint8Array =>
|
|
frame.kind === "text" ? frame.body : new Uint8Array(Buffer.from(frame.body, "base64"))
|
|
|
|
const decodeFrameText = (frame: WebSocketFrame) =>
|
|
frame.kind === "text" ? frame.body : new TextDecoder().decode(Buffer.from(frame.body, "base64"))
|
|
|
|
const assertEqual = (message: string, actual: unknown, expected: unknown) =>
|
|
Effect.sync(() => {
|
|
if (JSON.stringify(actual) === JSON.stringify(expected)) return
|
|
throw new Error(`${message}: expected ${safeText(expected)}, received ${safeText(actual)}`)
|
|
})
|
|
|
|
const jsonOrText = (value: string) => Option.match(decodeJson(value), { onNone: () => value, onSome: canonicalizeJson })
|
|
|
|
const compareClientMessage = (actual: string, expected: WebSocketFrame | undefined, index: number, asJson: boolean) => {
|
|
if (!expected)
|
|
return Effect.sync(() => {
|
|
throw new Error(`Unexpected WebSocket client frame ${index + 1}: ${safeText(actual)}`)
|
|
})
|
|
const expectedText = decodeFrameText(expected)
|
|
if (!asJson) return assertEqual(`WebSocket client frame ${index + 1}`, actual, expectedText)
|
|
return assertEqual(`WebSocket client JSON frame ${index + 1}`, jsonOrText(actual), jsonOrText(expectedText))
|
|
}
|
|
|
|
export const makeWebSocketExecutor = <E>(
|
|
options: WebSocketRecordReplayOptions<E>,
|
|
): Effect.Effect<WebSocketExecutor<E>, never, Scope.Scope> =>
|
|
Effect.gen(function* () {
|
|
const requested = options.mode ?? "auto"
|
|
const mode = requested === "auto" ? yield* resolveAutoMode(options.cassette, options.name) : requested
|
|
const redactor = options.redactor ?? defaults()
|
|
const openSnapshot = (request: WebSocketRequest) => {
|
|
const redacted = redactor.request({
|
|
method: "GET",
|
|
url: request.url,
|
|
headers: headersRecord(request.headers),
|
|
body: "",
|
|
})
|
|
return { url: redacted.url, headers: redacted.headers }
|
|
}
|
|
|
|
if (mode === "passthrough") return options.live
|
|
|
|
if (mode === "record") {
|
|
return {
|
|
open: (request) =>
|
|
Effect.gen(function* () {
|
|
const client: WebSocketFrame[] = []
|
|
const server: WebSocketFrame[] = []
|
|
const connection = yield* options.live.open(request)
|
|
const closed = yield* Ref.make(false)
|
|
const closeOnce = Effect.gen(function* () {
|
|
if (yield* Ref.getAndSet(closed, true)) return
|
|
yield* connection.close
|
|
yield* options.cassette
|
|
.append(
|
|
options.name,
|
|
{ transport: "websocket", open: openSnapshot(request), client, server },
|
|
options.metadata,
|
|
)
|
|
.pipe(Effect.orDie)
|
|
})
|
|
return {
|
|
sendText: (message) =>
|
|
connection
|
|
.sendText(message)
|
|
.pipe(Effect.tap(() => Effect.sync(() => client.push(encodeFrame(message))))),
|
|
messages: connection.messages.pipe(
|
|
Stream.tap((message) => Effect.sync(() => server.push(encodeFrame(message)))),
|
|
),
|
|
close: closeOnce,
|
|
}
|
|
}),
|
|
}
|
|
}
|
|
|
|
const replay = yield* makeReplayState(options.cassette, options.name, webSocketInteractions)
|
|
|
|
return {
|
|
open: (request) =>
|
|
Effect.gen(function* () {
|
|
const interactions = yield* replay.load.pipe(Effect.orDie)
|
|
const index = yield* replay.cursor
|
|
const interaction = interactions[index]
|
|
if (!interaction)
|
|
return yield* Effect.die(new Error(`No recorded WebSocket interaction for ${redactUrl(request.url)}`))
|
|
yield* replay.advance
|
|
yield* assertEqual(`WebSocket open frame ${index + 1}`, openSnapshot(request), interaction.open)
|
|
const messageIndex = yield* Ref.make(0)
|
|
return {
|
|
sendText: (message) =>
|
|
Effect.gen(function* () {
|
|
const current = yield* Ref.get(messageIndex)
|
|
yield* compareClientMessage(
|
|
message,
|
|
interaction.client[current],
|
|
current,
|
|
options.compareClientMessagesAsJson === true,
|
|
)
|
|
yield* Ref.update(messageIndex, (value) => value + 1)
|
|
}),
|
|
messages: Stream.fromIterable(interaction.server).pipe(Stream.map(decodeFrameMessage)),
|
|
close: Effect.gen(function* () {
|
|
yield* assertEqual(
|
|
`WebSocket client frame count for interaction ${index + 1}`,
|
|
yield* Ref.get(messageIndex),
|
|
interaction.client.length,
|
|
)
|
|
}),
|
|
}
|
|
}),
|
|
}
|
|
})
|