Files
openclaw/extensions/telegram/src/update-offset-store.ts
Peter Steinberger 694ca50e97 Revert "refactor: move runtime state to SQLite"
This reverts commit f91de52f0d.
2026-05-13 13:33:38 +01:00

184 lines
5.3 KiB
TypeScript

import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
import { readJsonFileWithFallback, writeJsonFileAtomically } from "openclaw/plugin-sdk/json-store";
import { resolveStateDir } from "openclaw/plugin-sdk/state-paths";
import { fingerprintTelegramBotToken } from "./token-fingerprint.js";
const STORE_VERSION = 3;
type TelegramUpdateOffsetState = {
version: number;
lastUpdateId: number | null;
botId: string | null;
tokenFingerprint: string | null;
};
function isValidUpdateId(value: unknown): value is number {
return typeof value === "number" && Number.isSafeInteger(value) && value >= 0;
}
function normalizeAccountId(accountId?: string) {
const trimmed = accountId?.trim();
if (!trimmed) {
return "default";
}
return trimmed.replace(/[^a-z0-9._-]+/gi, "_");
}
function resolveTelegramUpdateOffsetPath(
accountId?: string,
env: NodeJS.ProcessEnv = process.env,
): string {
const stateDir = resolveStateDir(env, os.homedir);
const normalized = normalizeAccountId(accountId);
return path.join(stateDir, "telegram", `update-offset-${normalized}.json`);
}
function extractBotIdFromToken(token?: string): string | null {
const trimmed = token?.trim();
if (!trimmed) {
return null;
}
const [rawBotId] = trimmed.split(":", 1);
if (!rawBotId || !/^\d+$/.test(rawBotId)) {
return null;
}
return rawBotId;
}
function fingerprintFromToken(token?: string): string | null {
const trimmed = token?.trim();
if (!trimmed) {
return null;
}
return fingerprintTelegramBotToken(trimmed);
}
function safeParseState(parsed: unknown): TelegramUpdateOffsetState | null {
try {
const state = parsed as {
version?: number;
lastUpdateId?: number | null;
botId?: string | null;
tokenFingerprint?: string | null;
};
if (state?.version !== STORE_VERSION && state?.version !== 2 && state?.version !== 1) {
return null;
}
if (state.lastUpdateId !== null && !isValidUpdateId(state.lastUpdateId)) {
return null;
}
if (state.version >= 2 && state.botId !== null && typeof state.botId !== "string") {
return null;
}
if (
state.version === STORE_VERSION &&
state.tokenFingerprint !== null &&
typeof state.tokenFingerprint !== "string"
) {
return null;
}
return {
version: state.version,
lastUpdateId: state.lastUpdateId ?? null,
botId: state.version >= 2 ? (state.botId ?? null) : null,
tokenFingerprint: state.version === STORE_VERSION ? (state.tokenFingerprint ?? null) : null,
};
} catch {
return null;
}
}
export type TelegramOffsetRotationReason = "bot-id-changed" | "token-rotated" | "legacy-state";
export type TelegramUpdateOffsetRotationInfo = {
reason: TelegramOffsetRotationReason;
previousBotId: string | null;
currentBotId: string;
staleLastUpdateId: number;
};
function rotationForToken(
parsed: TelegramUpdateOffsetState,
botToken?: string,
): TelegramUpdateOffsetRotationInfo | null {
const currentBotId = extractBotIdFromToken(botToken);
if (!currentBotId || parsed.lastUpdateId === null) {
return null;
}
let reason: TelegramOffsetRotationReason | null = null;
if (parsed.botId === null) {
reason = "legacy-state";
} else if (parsed.botId !== currentBotId) {
reason = "bot-id-changed";
} else if (parsed.tokenFingerprint === null) {
reason = "legacy-state";
} else if (parsed.tokenFingerprint !== fingerprintFromToken(botToken)) {
reason = "token-rotated";
}
return reason
? {
reason,
previousBotId: parsed.botId,
currentBotId,
staleLastUpdateId: parsed.lastUpdateId,
}
: null;
}
export async function readTelegramUpdateOffset(params: {
accountId?: string;
botToken?: string;
env?: NodeJS.ProcessEnv;
onRotationDetected?: (info: TelegramUpdateOffsetRotationInfo) => void | Promise<void>;
}): Promise<number | null> {
const filePath = resolveTelegramUpdateOffsetPath(params.accountId, params.env);
const { value } = await readJsonFileWithFallback<unknown>(filePath, null);
const parsed = safeParseState(value);
if (!parsed) {
return null;
}
const rotation = rotationForToken(parsed, params.botToken);
if (rotation) {
await params.onRotationDetected?.(rotation);
return null;
}
return parsed.lastUpdateId;
}
export async function writeTelegramUpdateOffset(params: {
accountId?: string;
updateId: number;
botToken?: string;
env?: NodeJS.ProcessEnv;
}): Promise<void> {
if (!isValidUpdateId(params.updateId)) {
throw new Error("Telegram update offset must be a non-negative safe integer.");
}
const filePath = resolveTelegramUpdateOffsetPath(params.accountId, params.env);
const payload: TelegramUpdateOffsetState = {
version: STORE_VERSION,
lastUpdateId: params.updateId,
botId: extractBotIdFromToken(params.botToken),
tokenFingerprint: fingerprintFromToken(params.botToken),
};
await writeJsonFileAtomically(filePath, payload);
}
export async function deleteTelegramUpdateOffset(params: {
accountId?: string;
env?: NodeJS.ProcessEnv;
}): Promise<void> {
const filePath = resolveTelegramUpdateOffsetPath(params.accountId, params.env);
try {
await fs.unlink(filePath);
} catch (err) {
const code = (err as { code?: string }).code;
if (code === "ENOENT") {
return;
}
throw err;
}
}