* feat(feishu): add ACP session support * fix(feishu): preserve sender-scoped ACP rebinding * fix(feishu): recover sender scope from bound ACP sessions * fix(feishu): support DM ACP binding placement * feat(feishu): add current-conversation session binding * fix(feishu): avoid DM parent binding fallback * fix(feishu): require canonical topic sender ids * fix(feishu): honor sender-scoped ACP bindings * fix(feishu): allow user-id ACP DM bindings * fix(feishu): recover user-id ACP DM bindings
128 lines
3.7 KiB
TypeScript
128 lines
3.7 KiB
TypeScript
import { z } from "zod";
|
|
import { AgentDefaultsSchema } from "./zod-schema.agent-defaults.js";
|
|
import { AgentEntrySchema } from "./zod-schema.agent-runtime.js";
|
|
import { TranscribeAudioSchema } from "./zod-schema.core.js";
|
|
|
|
export const AgentsSchema = z
|
|
.object({
|
|
defaults: z.lazy(() => AgentDefaultsSchema).optional(),
|
|
list: z.array(AgentEntrySchema).optional(),
|
|
})
|
|
.strict()
|
|
.optional();
|
|
|
|
const BindingMatchSchema = z
|
|
.object({
|
|
channel: z.string(),
|
|
accountId: z.string().optional(),
|
|
peer: z
|
|
.object({
|
|
kind: z.union([
|
|
z.literal("direct"),
|
|
z.literal("group"),
|
|
z.literal("channel"),
|
|
/** @deprecated Use `direct` instead. Kept for backward compatibility. */
|
|
z.literal("dm"),
|
|
]),
|
|
id: z.string(),
|
|
})
|
|
.strict()
|
|
.optional(),
|
|
guildId: z.string().optional(),
|
|
teamId: z.string().optional(),
|
|
roles: z.array(z.string()).optional(),
|
|
})
|
|
.strict();
|
|
|
|
const RouteBindingSchema = z
|
|
.object({
|
|
type: z.literal("route").optional(),
|
|
agentId: z.string(),
|
|
comment: z.string().optional(),
|
|
match: BindingMatchSchema,
|
|
})
|
|
.strict();
|
|
|
|
const AcpBindingSchema = z
|
|
.object({
|
|
type: z.literal("acp"),
|
|
agentId: z.string(),
|
|
comment: z.string().optional(),
|
|
match: BindingMatchSchema,
|
|
acp: z
|
|
.object({
|
|
mode: z.enum(["persistent", "oneshot"]).optional(),
|
|
label: z.string().optional(),
|
|
cwd: z.string().optional(),
|
|
backend: z.string().optional(),
|
|
})
|
|
.strict()
|
|
.optional(),
|
|
})
|
|
.strict()
|
|
.superRefine((value, ctx) => {
|
|
const peerId = value.match.peer?.id?.trim() ?? "";
|
|
if (!peerId) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["match", "peer"],
|
|
message: "ACP bindings require match.peer.id to target a concrete conversation.",
|
|
});
|
|
return;
|
|
}
|
|
const channel = value.match.channel.trim().toLowerCase();
|
|
if (channel !== "discord" && channel !== "telegram" && channel !== "feishu") {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["match", "channel"],
|
|
message:
|
|
'ACP bindings currently support only "discord", "telegram", and "feishu" channels.',
|
|
});
|
|
return;
|
|
}
|
|
if (channel === "telegram" && !/^-\d+:topic:\d+$/.test(peerId)) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["match", "peer", "id"],
|
|
message:
|
|
"Telegram ACP bindings require canonical topic IDs in the form -1001234567890:topic:42.",
|
|
});
|
|
}
|
|
if (channel === "feishu") {
|
|
const peerKind = value.match.peer?.kind;
|
|
const isDirectId =
|
|
(peerKind === "direct" || peerKind === "dm") &&
|
|
/^[^:]+$/.test(peerId) &&
|
|
!peerId.startsWith("oc_") &&
|
|
!peerId.startsWith("on_");
|
|
const isTopicId =
|
|
peerKind === "group" && /^oc_[^:]+:topic:[^:]+(?::sender:ou_[^:]+)?$/.test(peerId);
|
|
if (!isDirectId && !isTopicId) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ["match", "peer", "id"],
|
|
message:
|
|
"Feishu ACP bindings require direct peer IDs for DMs or topic IDs in the form oc_group:topic:om_root[:sender:ou_xxx].",
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
export const BindingsSchema = z.array(z.union([RouteBindingSchema, AcpBindingSchema])).optional();
|
|
|
|
export const BroadcastStrategySchema = z.enum(["parallel", "sequential"]);
|
|
|
|
export const BroadcastSchema = z
|
|
.object({
|
|
strategy: BroadcastStrategySchema.optional(),
|
|
})
|
|
.catchall(z.array(z.string()))
|
|
.optional();
|
|
|
|
export const AudioSchema = z
|
|
.object({
|
|
transcription: TranscribeAudioSchema,
|
|
})
|
|
.strict()
|
|
.optional();
|