Files
openclaw/extensions/tlon/src/setup-surface.ts
2026-03-17 05:17:51 +00:00

108 lines
3.5 KiB
TypeScript

import { DEFAULT_ACCOUNT_ID } from "openclaw/plugin-sdk/setup";
import {
applyTlonSetupConfig,
createTlonSetupWizardBase,
resolveTlonSetupConfigured,
resolveTlonSetupStatusLines,
type TlonSetupInput,
tlonSetupAdapter,
} from "./setup-core.js";
import { normalizeShip } from "./targets.js";
import { listTlonAccountIds, resolveTlonAccount, type TlonResolvedAccount } from "./types.js";
import { isBlockedUrbitHostname, validateUrbitBaseUrl } from "./urbit/base-url.js";
const channel = "tlon" as const;
function isConfigured(account: TlonResolvedAccount): boolean {
return Boolean(account.ship && account.url && account.code);
}
function parseList(value: string): string[] {
return value
.split(/[\n,;]+/g)
.map((entry) => entry.trim())
.filter(Boolean);
}
export { tlonSetupAdapter } from "./setup-core.js";
export const tlonSetupWizard = createTlonSetupWizardBase({
resolveConfigured: async ({ cfg }) => await resolveTlonSetupConfigured(cfg),
resolveStatusLines: async ({ cfg }) => await resolveTlonSetupStatusLines(cfg),
finalize: async ({ cfg, accountId, prompter }) => {
let next = cfg;
const resolved = resolveTlonAccount(next, accountId);
const validatedUrl = validateUrbitBaseUrl(resolved.url ?? "");
if (!validatedUrl.ok) {
throw new Error(`Invalid URL: ${validatedUrl.error}`);
}
let allowPrivateNetwork = resolved.allowPrivateNetwork ?? false;
if (isBlockedUrbitHostname(validatedUrl.hostname)) {
allowPrivateNetwork = await prompter.confirm({
message:
"Ship URL looks like a private/internal host. Allow private network access? (SSRF risk)",
initialValue: allowPrivateNetwork,
});
if (!allowPrivateNetwork) {
throw new Error("Refusing private/internal Ship URL without explicit approval");
}
}
next = applyTlonSetupConfig({
cfg: next,
accountId,
input: { allowPrivateNetwork },
});
const currentGroups = resolved.groupChannels;
const wantsGroupChannels = await prompter.confirm({
message: "Add group channels manually? (optional)",
initialValue: currentGroups.length > 0,
});
if (wantsGroupChannels) {
const entry = await prompter.text({
message: "Group channels (comma-separated)",
placeholder: "chat/~host-ship/general, chat/~host-ship/support",
initialValue: currentGroups.join(", ") || undefined,
});
next = applyTlonSetupConfig({
cfg: next,
accountId,
input: { groupChannels: parseList(String(entry ?? "")) },
});
}
const currentAllowlist = resolved.dmAllowlist;
const wantsAllowlist = await prompter.confirm({
message: "Restrict DMs with an allowlist?",
initialValue: currentAllowlist.length > 0,
});
if (wantsAllowlist) {
const entry = await prompter.text({
message: "DM allowlist (comma-separated ship names)",
placeholder: "~zod, ~nec",
initialValue: currentAllowlist.join(", ") || undefined,
});
next = applyTlonSetupConfig({
cfg: next,
accountId,
input: {
dmAllowlist: parseList(String(entry ?? "")).map((ship) => normalizeShip(ship)),
},
});
}
const autoDiscoverChannels = await prompter.confirm({
message: "Enable auto-discovery of group channels?",
initialValue: resolved.autoDiscoverChannels ?? true,
});
next = applyTlonSetupConfig({
cfg: next,
accountId,
input: { autoDiscoverChannels },
});
return { cfg: next };
},
});