mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-08 08:50:43 +00:00
* fix: address issue * fix: address PR review feedback * fix: address PR review feedback * docs: add changelog entry for PR merge
76 lines
2.4 KiB
TypeScript
76 lines
2.4 KiB
TypeScript
import { normalizeOptionalString } from "../shared/string-coerce.js";
|
|
import type { OpenClawConfig } from "./config-runtime.js";
|
|
|
|
type ApprovalKind = "exec" | "plugin";
|
|
type ApprovalAuthorizationResult = {
|
|
authorized: boolean;
|
|
reason?: string;
|
|
};
|
|
const IMPLICIT_SAME_CHAT_APPROVAL_AUTHORIZATION = Symbol(
|
|
"openclaw.implicitSameChatApprovalAuthorization",
|
|
);
|
|
|
|
function markImplicitSameChatApprovalAuthorization(
|
|
result: ApprovalAuthorizationResult,
|
|
): ApprovalAuthorizationResult {
|
|
// Keep this non-enumerable to avoid changing auth payload shape.
|
|
// Consumers must pass the same object reference to
|
|
// `isImplicitSameChatApprovalAuthorization`; spread/Object.assign/JSON clones
|
|
// drop this marker.
|
|
Object.defineProperty(result, IMPLICIT_SAME_CHAT_APPROVAL_AUTHORIZATION, {
|
|
value: true,
|
|
enumerable: false,
|
|
});
|
|
return result;
|
|
}
|
|
|
|
export function isImplicitSameChatApprovalAuthorization(
|
|
result: ApprovalAuthorizationResult | null | undefined,
|
|
): boolean {
|
|
return Boolean(
|
|
result &&
|
|
(
|
|
result as ApprovalAuthorizationResult & {
|
|
[IMPLICIT_SAME_CHAT_APPROVAL_AUTHORIZATION]?: true;
|
|
}
|
|
)[IMPLICIT_SAME_CHAT_APPROVAL_AUTHORIZATION],
|
|
);
|
|
}
|
|
|
|
export function createResolvedApproverActionAuthAdapter(params: {
|
|
channelLabel: string;
|
|
resolveApprovers: (params: { cfg: OpenClawConfig; accountId?: string | null }) => string[];
|
|
normalizeSenderId?: (value: string) => string | undefined;
|
|
}) {
|
|
const normalizeSenderId = params.normalizeSenderId ?? normalizeOptionalString;
|
|
|
|
return {
|
|
authorizeActorAction({
|
|
cfg,
|
|
accountId,
|
|
senderId,
|
|
approvalKind,
|
|
}: {
|
|
cfg: OpenClawConfig;
|
|
accountId?: string | null;
|
|
senderId?: string | null;
|
|
action: "approve";
|
|
approvalKind: ApprovalKind;
|
|
}) {
|
|
const approvers = params.resolveApprovers({ cfg, accountId });
|
|
if (approvers.length === 0) {
|
|
// Empty approver sets are implicit same-chat fallback, not explicit approver bypass.
|
|
return markImplicitSameChatApprovalAuthorization({ authorized: true });
|
|
}
|
|
const normalizedSenderId = senderId ? normalizeSenderId(senderId) : undefined;
|
|
if (normalizedSenderId && approvers.includes(normalizedSenderId)) {
|
|
return { authorized: true } as const;
|
|
}
|
|
return {
|
|
authorized: false,
|
|
reason: `❌ You are not authorized to approve ${approvalKind} requests on ${params.channelLabel}.`,
|
|
} as const;
|
|
},
|
|
};
|
|
}
|