Files
openclaw/src/plugins/services.ts
samzong 1d121c1f08 chore(gateway): add startup trace attribution (#81738)
Adds owner-level startup trace attribution for gateway auth, plugin loading, lookup counts, and plugin sidecar services.

Verification:
- node scripts/run-vitest.mjs src/plugins/startup-trace-segment.test.ts src/plugins/services.test.ts src/plugins/loader.test.ts src/gateway/server-startup-config.secrets.test.ts
- pnpm build
- pnpm check

CI override:
- Red checks are unrelated baseline noise. The failed CI shard is src/cli/plugins-install-persist.test.ts, which fails on origin/main 336ba2a2b3 with the same missing resolveIsNixMode mock export. PR #81738 touches gateway/plugin startup trace files and CHANGELOG.md, not the failing CLI plugin install test.

Thanks @samzong.

Co-authored-by: samzong <13782141+samzong@users.noreply.github.com>
2026-05-14 16:50:08 +08:00

120 lines
3.7 KiB
TypeScript

import { STATE_DIR } from "../config/paths.js";
import type { OpenClawConfig } from "../config/types.openclaw.js";
import {
emitTrustedDiagnosticEvent,
onInternalDiagnosticEvent,
} from "../infra/diagnostic-events.js";
import { createSubsystemLogger } from "../logging/subsystem.js";
import type { PluginServiceRegistration } from "./registry-types.js";
import type { PluginRegistry } from "./registry.js";
import { encodeStartupTraceSegment } from "./startup-trace-segment.js";
import type { OpenClawPluginServiceContext, PluginLogger } from "./types.js";
const log = createSubsystemLogger("plugins");
function createPluginLogger(): PluginLogger {
return {
info: (msg) => log.info(msg),
warn: (msg) => log.warn(msg),
error: (msg) => log.error(msg),
debug: (msg) => log.debug(msg),
};
}
function createServiceContext(params: {
config: OpenClawConfig;
workspaceDir?: string;
service?: PluginServiceRegistration;
}): OpenClawPluginServiceContext {
const isDiagnosticsExporter =
params.service?.pluginId === params.service?.service.id &&
(params.service?.service.id === "diagnostics-otel" ||
params.service?.service.id === "diagnostics-prometheus");
const grantsInternalDiagnostics =
isDiagnosticsExporter &&
(params.service?.origin === "bundled" || params.service?.trustedOfficialInstall === true);
return {
config: params.config,
workspaceDir: params.workspaceDir,
stateDir: STATE_DIR,
logger: createPluginLogger(),
...(grantsInternalDiagnostics
? {
internalDiagnostics: {
emit: emitTrustedDiagnosticEvent,
onEvent: onInternalDiagnosticEvent,
},
}
: {}),
};
}
export type PluginServicesHandle = {
stop: () => Promise<void>;
};
type PluginServiceStartupTrace = {
detail?: (name: string, metrics: ReadonlyArray<readonly [string, number | string]>) => void;
measure: <T>(name: string, run: () => T | Promise<T>) => Promise<T>;
};
export async function startPluginServices(params: {
registry: PluginRegistry;
config: OpenClawConfig;
workspaceDir?: string;
startupTrace?: PluginServiceStartupTrace;
}): Promise<PluginServicesHandle> {
const running: Array<{
id: string;
stop?: () => void | Promise<void>;
}> = [];
let failedCount = 0;
for (const entry of params.registry.services) {
const service = entry.service;
const serviceContext = createServiceContext({
config: params.config,
workspaceDir: params.workspaceDir,
service: entry,
});
try {
const startService = () => service.start(serviceContext);
const traceName = `sidecars.plugin-services.${encodeStartupTraceSegment(entry.pluginId)}.${encodeStartupTraceSegment(service.id)}`;
if (params.startupTrace) {
await params.startupTrace.measure(traceName, startService);
} else {
await startService();
}
running.push({
id: service.id,
stop: service.stop ? () => service.stop?.(serviceContext) : undefined,
});
} catch (err) {
failedCount += 1;
const error = err as Error;
log.error(
`plugin service failed (${service.id}, plugin=${entry.pluginId}, root=${entry.rootDir ?? "unknown"}): ${error?.message ?? String(err)}`,
);
}
}
params.startupTrace?.detail?.("sidecars.plugin-services.summary", [
["serviceCount", params.registry.services.length],
["startedCount", running.length],
["failedCount", failedCount],
]);
return {
stop: async () => {
for (const entry of running.toReversed()) {
if (!entry.stop) {
continue;
}
try {
await entry.stop();
} catch (err) {
log.warn(`plugin service stop failed (${entry.id}): ${String(err)}`);
}
}
},
};
}