Strip every comment from the source (parsed with the TypeScript compiler so strings, template literals, regex literals and JSX are never touched), and drop internal/working artifacts that do not belong in the public repository (design mockups, internal analysis docs, a stray backup file and an old log). No functional change: build is green, the full test suite passes.
205 lines
5.6 KiB
TypeScript
205 lines
5.6 KiB
TypeScript
import fs from "node:fs";
|
|
import { logTimestamp } from "./log-timestamp";
|
|
import path from "node:path";
|
|
import { AsyncLocalStorage } from "node:async_hooks";
|
|
import type { RotationEvent } from "../shared/types";
|
|
|
|
export type RotationItemSink = (event: RotationEvent) => void;
|
|
const rotationItemContext = new AsyncLocalStorage<RotationItemSink>();
|
|
|
|
export function runWithRotationItemSink<T>(sink: RotationItemSink, fn: () => Promise<T>): Promise<T> {
|
|
return rotationItemContext.run(sink, fn);
|
|
}
|
|
|
|
type RotationLevel = "INFO" | "WARN" | "ERROR";
|
|
|
|
const ROTATION_EVENT_RING_MAX = 60;
|
|
const rotationEventRing: RotationEvent[] = [];
|
|
let rotationEventSeq = 0;
|
|
let rotationEventListener: ((event: RotationEvent) => void) | null = null;
|
|
|
|
export function setRotationEventListener(listener: ((event: RotationEvent) => void) | null): void {
|
|
rotationEventListener = listener;
|
|
}
|
|
|
|
export function getRecentRotationEvents(limit = ROTATION_EVENT_RING_MAX): RotationEvent[] {
|
|
const slice = rotationEventRing.slice(-limit);
|
|
slice.reverse();
|
|
return slice;
|
|
}
|
|
|
|
function isUiRelevantRotationEvent(event: string): boolean {
|
|
return event !== "TEST";
|
|
}
|
|
|
|
function pushRotationEvent(
|
|
level: RotationLevel,
|
|
provider: string,
|
|
accountLabel: string,
|
|
event: string,
|
|
fields?: Record<string, unknown>,
|
|
at = Date.now()
|
|
): void {
|
|
rotationEventSeq += 1;
|
|
const entry: RotationEvent = {
|
|
id: `rot_${at}_${rotationEventSeq}`,
|
|
at,
|
|
level,
|
|
provider,
|
|
accountLabel,
|
|
event,
|
|
reason: fields && fields.reason != null ? String(fields.reason) : undefined,
|
|
category: fields && fields.category != null ? String(fields.category) : undefined,
|
|
cooldownSec: fields && fields.cooldownSec != null ? Number(fields.cooldownSec) || 0 : undefined,
|
|
next: fields && fields.next != null ? String(fields.next) : undefined
|
|
};
|
|
|
|
const itemSink = rotationItemContext.getStore();
|
|
if (itemSink) {
|
|
try {
|
|
itemSink(entry);
|
|
} catch {
|
|
}
|
|
}
|
|
|
|
if (!isUiRelevantRotationEvent(event)) {
|
|
return;
|
|
}
|
|
rotationEventRing.push(entry);
|
|
if (rotationEventRing.length > ROTATION_EVENT_RING_MAX) {
|
|
rotationEventRing.splice(0, rotationEventRing.length - ROTATION_EVENT_RING_MAX);
|
|
}
|
|
if (rotationEventListener) {
|
|
try {
|
|
rotationEventListener(entry);
|
|
} catch {
|
|
}
|
|
}
|
|
}
|
|
|
|
const ROTATION_LOG_MAX_FILE_BYTES = Number(process.env.RD_ACCOUNT_ROTATION_LOG_MAX_BYTES || 5 * 1024 * 1024);
|
|
const ROTATION_LOG_RETENTION_DAYS = Number(process.env.RD_ACCOUNT_ROTATION_LOG_RETENTION_DAYS || 14);
|
|
|
|
let rotationLogPath: string | null = null;
|
|
|
|
function sanitizeFieldValue(value: unknown): string {
|
|
if (value === undefined || value === null) {
|
|
return "";
|
|
}
|
|
if (typeof value === "string") {
|
|
return value.replace(/\r?\n/g, "\\n");
|
|
}
|
|
if (typeof value === "number" || typeof value === "boolean") {
|
|
return String(value);
|
|
}
|
|
try {
|
|
return JSON.stringify(value).replace(/\r?\n/g, "\\n");
|
|
} catch {
|
|
return String(value);
|
|
}
|
|
}
|
|
|
|
function formatFields(fields?: Record<string, unknown>): string {
|
|
if (!fields) {
|
|
return "";
|
|
}
|
|
const parts = Object.entries(fields)
|
|
.filter(([, value]) => value !== undefined && value !== null && sanitizeFieldValue(value) !== "")
|
|
.map(([key, value]) => `${key}=${sanitizeFieldValue(value)}`);
|
|
return parts.length > 0 ? ` | ${parts.join(" | ")}` : "";
|
|
}
|
|
|
|
function rotateIfNeeded(filePath: string): void {
|
|
try {
|
|
const stat = fs.statSync(filePath);
|
|
if (stat.size < ROTATION_LOG_MAX_FILE_BYTES) {
|
|
return;
|
|
}
|
|
const backup = `${filePath}.old`;
|
|
try {
|
|
fs.rmSync(backup, { force: true });
|
|
} catch {
|
|
}
|
|
fs.renameSync(filePath, backup);
|
|
} catch {
|
|
}
|
|
}
|
|
|
|
function cleanupOldBackup(filePath: string): void {
|
|
const backup = `${filePath}.old`;
|
|
try {
|
|
const stat = fs.statSync(backup);
|
|
const cutoff = Date.now() - ROTATION_LOG_RETENTION_DAYS * 24 * 60 * 60 * 1000;
|
|
if (stat.mtimeMs < cutoff) {
|
|
fs.rmSync(backup, { force: true });
|
|
}
|
|
} catch {
|
|
}
|
|
}
|
|
|
|
export function initAccountRotationLog(baseDir: string): void {
|
|
rotationLogPath = path.join(baseDir, "account-rotation.log");
|
|
try {
|
|
fs.mkdirSync(path.dirname(rotationLogPath), { recursive: true });
|
|
cleanupOldBackup(rotationLogPath);
|
|
if (!fs.existsSync(rotationLogPath)) {
|
|
fs.writeFileSync(rotationLogPath, "", "utf8");
|
|
}
|
|
rotateIfNeeded(rotationLogPath);
|
|
if (!fs.existsSync(rotationLogPath)) {
|
|
fs.writeFileSync(rotationLogPath, "", "utf8");
|
|
}
|
|
fs.appendFileSync(
|
|
rotationLogPath,
|
|
`=== Account-Rotation Log Start: ${logTimestamp()} ===\n`,
|
|
"utf8"
|
|
);
|
|
} catch {
|
|
rotationLogPath = null;
|
|
}
|
|
}
|
|
|
|
export function logAccountRotation(
|
|
level: RotationLevel,
|
|
provider: string,
|
|
accountLabel: string,
|
|
event: string,
|
|
fields?: Record<string, unknown>
|
|
): void {
|
|
pushRotationEvent(level, provider, accountLabel, event, fields);
|
|
if (!rotationLogPath) {
|
|
return;
|
|
}
|
|
try {
|
|
rotateIfNeeded(rotationLogPath);
|
|
if (!fs.existsSync(rotationLogPath)) {
|
|
fs.writeFileSync(rotationLogPath, "", "utf8");
|
|
}
|
|
const head = `${logTimestamp()} [${level}] ${provider} | ${accountLabel} | ${event}`;
|
|
fs.appendFileSync(rotationLogPath, `${head}${formatFields(fields)}\n`, "utf8");
|
|
} catch {
|
|
}
|
|
}
|
|
|
|
export function getAccountRotationLogPath(): string | null {
|
|
if (!rotationLogPath) {
|
|
return null;
|
|
}
|
|
return fs.existsSync(rotationLogPath) ? rotationLogPath : null;
|
|
}
|
|
|
|
export function shutdownAccountRotationLog(): void {
|
|
if (!rotationLogPath) {
|
|
return;
|
|
}
|
|
try {
|
|
fs.appendFileSync(
|
|
rotationLogPath,
|
|
`=== Account-Rotation Log Ende: ${logTimestamp()} ===\n`,
|
|
"utf8"
|
|
);
|
|
} catch {
|
|
}
|
|
rotationLogPath = null;
|
|
}
|