150 lines
4.3 KiB
TypeScript
150 lines
4.3 KiB
TypeScript
import type { TelegramGroupConfig } from "../config/types.js";
|
|
import { isRecord } from "../utils.js";
|
|
|
|
const TELEGRAM_API_BASE = "https://api.telegram.org";
|
|
|
|
export type TelegramGroupMembershipAuditEntry = {
|
|
chatId: string;
|
|
ok: boolean;
|
|
status?: string | null;
|
|
error?: string | null;
|
|
matchKey?: string;
|
|
matchSource?: "id";
|
|
};
|
|
|
|
export type TelegramGroupMembershipAudit = {
|
|
ok: boolean;
|
|
checkedGroups: number;
|
|
unresolvedGroups: number;
|
|
hasWildcardUnmentionedGroups: boolean;
|
|
groups: TelegramGroupMembershipAuditEntry[];
|
|
elapsedMs: number;
|
|
};
|
|
|
|
type TelegramApiOk<T> = { ok: true; result: T };
|
|
type TelegramApiErr = { ok: false; description?: string };
|
|
|
|
export function collectTelegramUnmentionedGroupIds(
|
|
groups: Record<string, TelegramGroupConfig> | undefined,
|
|
) {
|
|
if (!groups || typeof groups !== "object") {
|
|
return {
|
|
groupIds: [] as string[],
|
|
unresolvedGroups: 0,
|
|
hasWildcardUnmentionedGroups: false,
|
|
};
|
|
}
|
|
const hasWildcardUnmentionedGroups =
|
|
Boolean(groups["*"]?.requireMention === false) && groups["*"]?.enabled !== false;
|
|
const groupIds: string[] = [];
|
|
let unresolvedGroups = 0;
|
|
for (const [key, value] of Object.entries(groups)) {
|
|
if (key === "*") {
|
|
continue;
|
|
}
|
|
if (!value || typeof value !== "object") {
|
|
continue;
|
|
}
|
|
if (value.enabled === false) {
|
|
continue;
|
|
}
|
|
if (value.requireMention !== false) {
|
|
continue;
|
|
}
|
|
const id = String(key).trim();
|
|
if (!id) {
|
|
continue;
|
|
}
|
|
if (/^-?\d+$/.test(id)) {
|
|
groupIds.push(id);
|
|
} else {
|
|
unresolvedGroups += 1;
|
|
}
|
|
}
|
|
groupIds.sort((a, b) => a.localeCompare(b));
|
|
return { groupIds, unresolvedGroups, hasWildcardUnmentionedGroups };
|
|
}
|
|
|
|
export async function auditTelegramGroupMembership(params: {
|
|
token: string;
|
|
botId: number;
|
|
groupIds: string[];
|
|
proxyUrl?: string;
|
|
timeoutMs: number;
|
|
}): Promise<TelegramGroupMembershipAudit> {
|
|
const started = Date.now();
|
|
const token = params.token?.trim() ?? "";
|
|
if (!token || params.groupIds.length === 0) {
|
|
return {
|
|
ok: true,
|
|
checkedGroups: 0,
|
|
unresolvedGroups: 0,
|
|
hasWildcardUnmentionedGroups: false,
|
|
groups: [],
|
|
elapsedMs: Date.now() - started,
|
|
};
|
|
}
|
|
|
|
// Lazy import to avoid pulling `undici` (ProxyAgent) into cold-path callers that only need
|
|
// `collectTelegramUnmentionedGroupIds` (e.g. config audits).
|
|
const fetcher = params.proxyUrl
|
|
? (await import("./proxy.js")).makeProxyFetch(params.proxyUrl)
|
|
: fetch;
|
|
const { fetchWithTimeout } = await import("../utils/fetch-timeout.js");
|
|
const base = `${TELEGRAM_API_BASE}/bot${token}`;
|
|
const groups: TelegramGroupMembershipAuditEntry[] = [];
|
|
|
|
for (const chatId of params.groupIds) {
|
|
try {
|
|
const url = `${base}/getChatMember?chat_id=${encodeURIComponent(chatId)}&user_id=${encodeURIComponent(String(params.botId))}`;
|
|
const res = await fetchWithTimeout(url, {}, params.timeoutMs, fetcher);
|
|
const json = (await res.json()) as TelegramApiOk<{ status?: string }> | TelegramApiErr;
|
|
if (!res.ok || !isRecord(json) || !json.ok) {
|
|
const desc =
|
|
isRecord(json) && !json.ok && typeof json.description === "string"
|
|
? json.description
|
|
: `getChatMember failed (${res.status})`;
|
|
groups.push({
|
|
chatId,
|
|
ok: false,
|
|
status: null,
|
|
error: desc,
|
|
matchKey: chatId,
|
|
matchSource: "id",
|
|
});
|
|
continue;
|
|
}
|
|
const status = isRecord((json as TelegramApiOk<unknown>).result)
|
|
? ((json as TelegramApiOk<{ status?: string }>).result.status ?? null)
|
|
: null;
|
|
const ok = status === "creator" || status === "administrator" || status === "member";
|
|
groups.push({
|
|
chatId,
|
|
ok,
|
|
status,
|
|
error: ok ? null : "bot not in group",
|
|
matchKey: chatId,
|
|
matchSource: "id",
|
|
});
|
|
} catch (err) {
|
|
groups.push({
|
|
chatId,
|
|
ok: false,
|
|
status: null,
|
|
error: err instanceof Error ? err.message : String(err),
|
|
matchKey: chatId,
|
|
matchSource: "id",
|
|
});
|
|
}
|
|
}
|
|
|
|
return {
|
|
ok: groups.every((g) => g.ok),
|
|
checkedGroups: groups.length,
|
|
unresolvedGroups: 0,
|
|
hasWildcardUnmentionedGroups: false,
|
|
groups,
|
|
elapsedMs: Date.now() - started,
|
|
};
|
|
}
|