@rawdash/mcp 0.12.0 → 0.15.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  import { McpServer } from '@modelcontextprotocol/sdk/server/mcp';
2
- import { Connector, DashboardConfig, ServerStorage, ConfiguredConnector } from '@rawdash/core';
2
+ import { Connector, DashboardConfig, ServerStorage, ConnectorRegistry, SecretsResolver, ConfiguredConnector } from '@rawdash/core';
3
3
  import { ZodObject, ZodRawShape } from 'zod';
4
4
 
5
5
  interface McpErrorPayload {
@@ -18,9 +18,11 @@ interface McpServerOptions {
18
18
  version?: string;
19
19
  config: DashboardConfig;
20
20
  storage: ServerStorage;
21
+ connectorRegistry?: ConnectorRegistry;
22
+ secretsResolver?: SecretsResolver;
21
23
  connectorFactories?: ConnectorFactory[];
22
24
  onAddConnector?: (entry: ConfiguredConnector) => void | Promise<void>;
23
- onRemoveConnector?: (connectorId: string) => void | Promise<void>;
25
+ onRemoveConnector?: (name: string) => void | Promise<void>;
24
26
  onSetSecret?: (name: string, value: string) => void | Promise<void>;
25
27
  listSecrets?: () => string[] | Promise<string[]>;
26
28
  }
package/dist/index.js CHANGED
@@ -16,16 +16,14 @@ var McpRuntime = class {
16
16
  return this.dashboards;
17
17
  }
18
18
  addConnector(entry) {
19
- if (this.connectors.some((e) => e.connector.id === entry.connector.id)) {
20
- throw new Error(`Connector "${entry.connector.id}" already exists`);
19
+ if (this.connectors.some((e) => e.name === entry.name)) {
20
+ throw new Error(`Connector "${entry.name}" already exists`);
21
21
  }
22
22
  this.connectors = [...this.connectors, entry];
23
23
  }
24
- removeConnector(connectorId) {
24
+ removeConnector(name) {
25
25
  const before = this.connectors.length;
26
- this.connectors = this.connectors.filter(
27
- (e) => e.connector.id !== connectorId
28
- );
26
+ this.connectors = this.connectors.filter((e) => e.name !== name);
29
27
  return this.connectors.length < before;
30
28
  }
31
29
  };
@@ -72,11 +70,14 @@ function registerAddConnector(server, runtime, options) {
72
70
  connector_type: z.string().describe(
73
71
  'The connector type ID (e.g. "github-actions"). Use list_connectors to see available types.'
74
72
  ),
73
+ name: z.string().describe(
74
+ "Instance name for this connector (must be unique within the deployment)."
75
+ ),
75
76
  settings: z.record(z.string(), z.unknown()).describe(
76
77
  "Connector settings as a JSON object. Schema depends on the connector type."
77
78
  )
78
79
  },
79
- async ({ connector_type, settings }) => {
80
+ async ({ connector_type, name, settings }) => {
80
81
  const factories = options.connectorFactories ?? [];
81
82
  const factory = factories.find((f) => f.id === connector_type);
82
83
  if (!factory) {
@@ -91,15 +92,11 @@ function registerAddConnector(server, runtime, options) {
91
92
  const issues = parsed.error.issues.map((i) => `${i.path.join(".")}: ${i.message}`).join("; ");
92
93
  return err("INVALID_SETTINGS", `Invalid settings: ${issues}`);
93
94
  }
94
- let entry;
95
- try {
96
- entry = { connector: factory.create(parsed.data) };
97
- } catch (e) {
98
- return err(
99
- "CREATE_FAILED",
100
- `Failed to create connector: ${e instanceof Error ? e.message : String(e)}`
101
- );
102
- }
95
+ const entry = {
96
+ name,
97
+ connectorId: connector_type,
98
+ config: parsed.data
99
+ };
103
100
  try {
104
101
  runtime.addConnector(entry);
105
102
  } catch (e) {
@@ -116,7 +113,7 @@ function registerAddConnector(server, runtime, options) {
116
113
  `Connector added in-memory but post-add callback failed: ${e instanceof Error ? e.message : String(e)}`
117
114
  );
118
115
  }
119
- return text({ added: entry.connector.id, idempotent: false });
116
+ return text({ added: entry.name, idempotent: false });
120
117
  }
121
118
  );
122
119
  }
@@ -131,12 +128,12 @@ function registerListConnectors(server, runtime, storage) {
131
128
  try {
132
129
  const syncState = await storage.getSyncState();
133
130
  const connectors = runtime.getConnectors().map((entry) => ({
134
- id: entry.connector.id,
131
+ id: entry.name,
132
+ connectorId: entry.connectorId,
135
133
  syncStatus: syncState.status,
136
134
  lastSyncAt: syncState.lastSyncAt,
137
135
  lastError: syncState.lastError,
138
- hasCredentials: Object.keys(entry.connector.credentials ?? {}).length > 0,
139
- credentialKeys: Object.keys(entry.connector.credentials ?? {})
136
+ configKeys: Object.keys(entry.config)
140
137
  }));
141
138
  return text({ connectors, overallSyncStatus: syncState.status });
142
139
  } catch (e) {
@@ -247,7 +244,7 @@ function registerReadWidget(server, runtime, storage) {
247
244
  });
248
245
  }
249
246
  const { connectorId } = widget.metric;
250
- const connectorEntry = runtime.getConnectors().find((e) => e.connector.id === connectorId);
247
+ const connectorEntry = runtime.getConnectors().find((e) => e.name === connectorId);
251
248
  if (!connectorEntry) {
252
249
  return err(
253
250
  "CONNECTOR_NOT_FOUND",
@@ -334,7 +331,7 @@ ${JSON.stringify(value, null, 2)}
334
331
  `;
335
332
  }
336
333
  function renderStatus(title, source, syncState) {
337
- const statusEmoji = syncState.status === "idle" ? "\u2705" : syncState.status === "syncing" ? "\u{1F504}" : "\u274C";
334
+ const statusEmoji = syncState.status === "succeeded" || syncState.status === "idle" ? "\u2705" : syncState.status === "queued" || syncState.status === "running" ? "\u{1F504}" : "\u274C";
338
335
  const lastSync = syncState.lastSyncAt ? new Date(syncState.lastSyncAt).toUTCString() : "never";
339
336
  return `## ${title}
340
337
 
@@ -375,7 +372,7 @@ function registerRenderWidget(server, runtime, storage) {
375
372
  };
376
373
  }
377
374
  const { connectorId } = widget.metric;
378
- const connectorEntry = runtime.getConnectors().find((e) => e.connector.id === connectorId);
375
+ const connectorEntry = runtime.getConnectors().find((e) => e.name === connectorId);
379
376
  if (!connectorEntry) {
380
377
  return err(
381
378
  "CONNECTOR_NOT_FOUND",
@@ -458,24 +455,37 @@ function registerSetSecret(server, trackedSecrets, options) {
458
455
  }
459
456
 
460
457
  // src/tools/trigger-sync.ts
458
+ import { instantiateConnector } from "@rawdash/core";
461
459
  import { z as z7 } from "zod";
462
- function registerTriggerSync(server, runtime, storage) {
460
+ function registerTriggerSync(server, runtime, storage, options) {
463
461
  server.tool(
464
462
  "trigger_sync",
465
463
  "Kick off a sync for one or all connectors. Runs in the background and returns immediately once started.",
466
464
  {
467
- connector_id: z7.string().optional().describe("Connector ID to sync. Omit to sync all connectors."),
465
+ connector_id: z7.string().optional().describe(
466
+ "Connector instance name to sync. Omit to sync all connectors."
467
+ ),
468
468
  mode: z7.enum(["full", "latest"]).optional().describe(
469
469
  '"latest" (default) fetches only recent data. "full" re-fetches all historical data.'
470
470
  )
471
471
  },
472
472
  async ({ connector_id, mode = "latest" }) => {
473
+ if (!options.connectorRegistry) {
474
+ return err(
475
+ "NO_REGISTRY",
476
+ "trigger_sync requires connectorRegistry on McpServerOptions"
477
+ );
478
+ }
473
479
  const connectors = runtime.getConnectors();
474
- const targets = connector_id ? connectors.filter((e) => e.connector.id === connector_id) : connectors;
480
+ const targets = connector_id ? connectors.filter((e) => e.name === connector_id) : connectors;
475
481
  if (connector_id && targets.length === 0) {
476
482
  return err("NOT_FOUND", `Connector "${connector_id}" not found`);
477
483
  }
478
- const acquired = await storage.setSyncing();
484
+ const queued = await storage.markSyncQueued();
485
+ if (!queued) {
486
+ return err("ALREADY_SYNCING", "A sync is already in progress");
487
+ }
488
+ const acquired = await storage.markSyncRunning();
479
489
  if (!acquired) {
480
490
  return err("ALREADY_SYNCING", "A sync is already in progress");
481
491
  }
@@ -483,27 +493,34 @@ function registerTriggerSync(server, runtime, storage) {
483
493
  const run = async () => {
484
494
  try {
485
495
  for (const entry of targets) {
486
- const handle = storage.getStorageHandle(entry.connector.id);
487
- await entry.connector.sync(
496
+ const connector = instantiateConnector(
497
+ entry,
498
+ options.connectorRegistry,
499
+ options.secretsResolver
500
+ );
501
+ const handle = storage.getStorageHandle(entry.name);
502
+ await connector.sync(
488
503
  { mode: mode ?? "latest" },
489
504
  handle,
490
505
  controller.signal
491
506
  );
492
507
  }
493
- await storage.setSyncSuccess();
508
+ await storage.markSyncSucceeded();
494
509
  } catch (e) {
495
510
  const message = e instanceof Error ? e.message : String(e);
496
511
  try {
497
- await storage.setSyncError(message);
498
- } catch {
512
+ await storage.markSyncFailed(message);
513
+ } catch (markErr) {
514
+ console.error("Failed to record sync failure in storage:", markErr);
499
515
  }
500
516
  }
501
517
  };
502
- void run().catch(() => {
518
+ void run().catch((rejection) => {
519
+ console.error("Unexpected background sync rejection:", rejection);
503
520
  });
504
521
  return text({
505
522
  triggered: true,
506
- connectors: targets.map((e) => e.connector.id),
523
+ connectors: targets.map((e) => e.name),
507
524
  mode
508
525
  });
509
526
  }
@@ -525,7 +542,7 @@ function createMcpServer(options) {
525
542
  registerRemoveConnector(server, runtime, options);
526
543
  registerSetSecret(server, trackedSecrets, options);
527
544
  registerListSecrets(server, trackedSecrets, options);
528
- registerTriggerSync(server, runtime, storage);
545
+ registerTriggerSync(server, runtime, storage, options);
529
546
  return server;
530
547
  }
531
548
  export {
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/server.ts","../src/runtime-config.ts","../src/tools/add-connector.ts","../src/tools/shared.ts","../src/tools/list-connectors.ts","../src/tools/list-dashboards.ts","../src/tools/list-secrets.ts","../src/tools/list-widgets.ts","../src/tools/read-widget.ts","../src/tools/remove-connector.ts","../src/tools/render-widget.ts","../src/tools/set-secret.ts","../src/tools/trigger-sync.ts"],"sourcesContent":["import { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport { McpRuntime } from './runtime-config';\nimport { registerAddConnector } from './tools/add-connector';\nimport { registerListConnectors } from './tools/list-connectors';\nimport { registerListDashboards } from './tools/list-dashboards';\nimport { registerListSecrets } from './tools/list-secrets';\nimport { registerListWidgets } from './tools/list-widgets';\nimport { registerReadWidget } from './tools/read-widget';\nimport { registerRemoveConnector } from './tools/remove-connector';\nimport { registerRenderWidget } from './tools/render-widget';\nimport { registerSetSecret } from './tools/set-secret';\nimport { registerTriggerSync } from './tools/trigger-sync';\nimport type { McpServerOptions } from './types';\n\nexport function createMcpServer(options: McpServerOptions): McpServer {\n const { name = 'rawdash', version = '1.0.0', config, storage } = options;\n\n const server = new McpServer({ name, version });\n const runtime = new McpRuntime(config);\n const trackedSecrets = new Set<string>();\n\n registerListDashboards(server, runtime);\n registerListWidgets(server, runtime);\n registerReadWidget(server, runtime, storage);\n registerRenderWidget(server, runtime, storage);\n registerListConnectors(server, runtime, storage);\n registerAddConnector(server, runtime, options);\n registerRemoveConnector(server, runtime, options);\n registerSetSecret(server, trackedSecrets, options);\n registerListSecrets(server, trackedSecrets, options);\n registerTriggerSync(server, runtime, storage);\n\n return server;\n}\n","import type {\n ConfiguredConnector,\n Dashboard,\n DashboardConfig,\n} from '@rawdash/core';\n\nexport class McpRuntime {\n private connectors: ConfiguredConnector[];\n private readonly dashboards: Record<string, Dashboard>;\n\n constructor(config: DashboardConfig) {\n this.connectors = [...config.connectors];\n this.dashboards = config.dashboards;\n }\n\n getConnectors(): ConfiguredConnector[] {\n return [...this.connectors];\n }\n\n getDashboards(): Record<string, Dashboard> {\n return this.dashboards;\n }\n\n addConnector(entry: ConfiguredConnector): void {\n if (this.connectors.some((e) => e.connector.id === entry.connector.id)) {\n throw new Error(`Connector \"${entry.connector.id}\" already exists`);\n }\n this.connectors = [...this.connectors, entry];\n }\n\n removeConnector(connectorId: string): boolean {\n const before = this.connectors.length;\n this.connectors = this.connectors.filter(\n (e) => e.connector.id !== connectorId,\n );\n return this.connectors.length < before;\n }\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerAddConnector(\n server: McpServer,\n runtime: McpRuntime,\n options: Pick<McpServerOptions, 'connectorFactories' | 'onAddConnector'>,\n): void {\n server.tool(\n 'add_connector',\n \"Add a connector instance to this Rawdash instance. Settings are validated against the connector's configFields schema. For OSS, this is a runtime-only change (not persisted to disk).\",\n {\n connector_type: z\n .string()\n .describe(\n 'The connector type ID (e.g. \"github-actions\"). Use list_connectors to see available types.',\n ),\n settings: z\n .record(z.string(), z.unknown())\n .describe(\n 'Connector settings as a JSON object. Schema depends on the connector type.',\n ),\n },\n async ({ connector_type, settings }) => {\n const factories = options.connectorFactories ?? [];\n const factory = factories.find((f) => f.id === connector_type);\n if (!factory) {\n const available = factories.map((f) => f.id);\n return err(\n 'UNKNOWN_CONNECTOR_TYPE',\n available.length > 0\n ? `Unknown connector type \"${connector_type}\". Available: ${available.join(', ')}`\n : `Unknown connector type \"${connector_type}\". No connector factories are registered.`,\n );\n }\n\n const parsed = factory.configFields.safeParse(settings);\n if (!parsed.success) {\n const issues = parsed.error.issues\n .map((i) => `${i.path.join('.')}: ${i.message}`)\n .join('; ');\n return err('INVALID_SETTINGS', `Invalid settings: ${issues}`);\n }\n\n let entry;\n try {\n entry = { connector: factory.create(parsed.data) };\n } catch (e) {\n return err(\n 'CREATE_FAILED',\n `Failed to create connector: ${e instanceof Error ? e.message : String(e)}`,\n );\n }\n\n try {\n runtime.addConnector(entry);\n } catch (e) {\n return err(\n 'ALREADY_EXISTS',\n e instanceof Error ? e.message : String(e),\n );\n }\n\n try {\n await options.onAddConnector?.(entry);\n } catch (e) {\n return err(\n 'ON_ADD_CONNECTOR_FAILED',\n `Connector added in-memory but post-add callback failed: ${e instanceof Error ? e.message : String(e)}`,\n );\n }\n\n return text({ added: entry.connector.id, idempotent: false });\n },\n );\n}\n","function safeStringify(data: unknown): string {\n try {\n return JSON.stringify(\n data,\n (_key, value) => (typeof value === 'bigint' ? value.toString() : value),\n 2,\n );\n } catch {\n return JSON.stringify(\n {\n error: {\n code: 'SERIALIZATION_ERROR',\n message: 'Failed to serialize tool response',\n },\n },\n null,\n 2,\n );\n }\n}\n\nexport function text(data: unknown) {\n return {\n content: [{ type: 'text' as const, text: safeStringify(data) }],\n };\n}\n\nexport function err(code: string, message: string) {\n return text({ error: { code, message } });\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerListConnectors(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'list_connectors',\n 'List all configured connectors and their sync status.',\n {},\n async () => {\n try {\n const syncState = await storage.getSyncState();\n const connectors = runtime.getConnectors().map((entry) => ({\n id: entry.connector.id,\n syncStatus: syncState.status,\n lastSyncAt: syncState.lastSyncAt,\n lastError: syncState.lastError,\n hasCredentials:\n Object.keys(entry.connector.credentials ?? {}).length > 0,\n credentialKeys: Object.keys(entry.connector.credentials ?? {}),\n }));\n return text({ connectors, overallSyncStatus: syncState.status });\n } catch (e) {\n return err(\n 'LIST_CONNECTORS_FAILED',\n e instanceof Error ? e.message : String(e),\n );\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpRuntime } from '../runtime-config';\nimport { text } from './shared';\n\nexport function registerListDashboards(\n server: McpServer,\n runtime: McpRuntime,\n): void {\n server.tool(\n 'list_dashboards',\n 'List all dashboards available on this Rawdash instance.',\n {},\n () => {\n const dashboards = Object.entries(runtime.getDashboards()).map(\n ([id, dashboard]) => ({\n id,\n widgetCount: Object.keys(dashboard.widgets).length,\n widgetIds: Object.keys(dashboard.widgets),\n }),\n );\n return Promise.resolve(text({ dashboards }));\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerListSecrets(\n server: McpServer,\n trackedSecrets: Set<string>,\n options: Pick<McpServerOptions, 'listSecrets'>,\n): void {\n server.tool(\n 'list_secrets',\n 'List secret names known to this Rawdash instance. Values are never returned.',\n {},\n async () => {\n try {\n const names = options.listSecrets\n ? await options.listSecrets()\n : [...trackedSecrets];\n return text({ secrets: [...names].sort() });\n } catch (e) {\n return err(\n 'LIST_SECRETS_FAILED',\n e instanceof Error ? e.message : 'Failed to list secrets.',\n );\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport { err, text } from './shared';\n\nexport function registerListWidgets(\n server: McpServer,\n runtime: McpRuntime,\n): void {\n server.tool(\n 'list_widgets',\n 'List all widgets on a dashboard.',\n {\n dashboard_id: z\n .string()\n .describe('The dashboard ID to list widgets for.'),\n },\n ({ dashboard_id }) => {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return Promise.resolve(\n err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`),\n );\n }\n const widgets = Object.entries(dashboard.widgets).map(([id, widget]) => ({\n id,\n kind: widget.kind,\n title: widget.title,\n ...(widget.kind !== 'status'\n ? { connectorId: widget.metric.connectorId }\n : { source: widget.source }),\n }));\n return Promise.resolve(text({ dashboard_id, widgets }));\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { computeMetric } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerReadWidget(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'read_widget',\n 'Fetch resolved widget data. Returns the same shape as the HTTP read API.',\n {\n dashboard_id: z.string().describe('The dashboard ID.'),\n widget_id: z.string().describe('The widget ID.'),\n },\n async ({ dashboard_id, widget_id }) => {\n try {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`);\n }\n const widget = dashboard.widgets[widget_id];\n if (!widget) {\n return err('NOT_FOUND', `Widget \"${widget_id}\" not found`);\n }\n\n if (widget.kind === 'status') {\n const syncState = await storage.getSyncState();\n return text({\n id: widget_id,\n widgetId: widget_id,\n connectorId: widget.source,\n data: null,\n cachedAt: syncState.lastSyncAt,\n });\n }\n\n const { connectorId } = widget.metric;\n const connectorEntry = runtime\n .getConnectors()\n .find((e) => e.connector.id === connectorId);\n if (!connectorEntry) {\n return err(\n 'CONNECTOR_NOT_FOUND',\n `Connector \"${connectorId}\" not found`,\n );\n }\n\n const handle = storage.getStorageHandle(connectorId);\n const data = await computeMetric(handle, widget.metric);\n const syncState = await storage.getSyncState();\n\n return text({\n id: widget_id,\n widgetId: widget_id,\n connectorId,\n data,\n cachedAt: syncState.lastSyncAt,\n });\n } catch (e) {\n return err('READ_FAILED', e instanceof Error ? e.message : String(e));\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerRemoveConnector(\n server: McpServer,\n runtime: McpRuntime,\n options: Pick<McpServerOptions, 'onRemoveConnector'>,\n): void {\n server.tool(\n 'remove_connector',\n 'Remove a connector from this Rawdash instance. For OSS, this is a runtime-only change. Idempotent — removing a non-existent connector returns success.',\n {\n connector_id: z.string().describe('The connector instance ID to remove.'),\n },\n async ({ connector_id }) => {\n const existed = runtime.removeConnector(connector_id);\n if (!existed) {\n return text({ removed: connector_id, existed: false });\n }\n\n try {\n await options.onRemoveConnector?.(connector_id);\n } catch (e) {\n return err(\n 'ON_REMOVE_CONNECTOR_FAILED',\n `Connector removed in-memory but post-remove callback failed: ${e instanceof Error ? e.message : String(e)}`,\n );\n }\n\n return text({ removed: connector_id, existed: true });\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { computeMetric } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err } from './shared';\n\nfunction renderStat(title: string, value: unknown): string {\n const display =\n typeof value === 'number'\n ? value.toLocaleString('en-US', { maximumFractionDigits: 2 })\n : String(value ?? '—');\n return `## ${title}\\n\\n**${display}**\\n`;\n}\n\nfunction renderTimeseries(\n title: string,\n points: Array<{ date: string; value: unknown }>,\n): string {\n if (points.length === 0) {\n return `## ${title}\\n\\n_No data_\\n`;\n }\n const rows = points\n .map(\n (p) =>\n `| ${p.date} | ${typeof p.value === 'number' ? p.value.toLocaleString('en-US', { maximumFractionDigits: 2 }) : String(p.value ?? '—')} |`,\n )\n .join('\\n');\n return `## ${title}\\n\\n| Date | Value |\\n|------|-------|\\n${rows}\\n`;\n}\n\nfunction renderDistribution(title: string, value: unknown): string {\n return `## ${title}\\n\\n\\`\\`\\`json\\n${JSON.stringify(value, null, 2)}\\n\\`\\`\\`\\n`;\n}\n\nfunction renderStatus(\n title: string,\n source: string,\n syncState: {\n status: string;\n lastSyncAt: string | null;\n lastError: string | null;\n },\n): string {\n const statusEmoji =\n syncState.status === 'idle'\n ? '✅'\n : syncState.status === 'syncing'\n ? '🔄'\n : '❌';\n const lastSync = syncState.lastSyncAt\n ? new Date(syncState.lastSyncAt).toUTCString()\n : 'never';\n return `## ${title}\\n\\n${statusEmoji} **${syncState.status}** (connector: \\`${source}\\`)\\n\\nLast sync: ${lastSync}${syncState.lastError ? `\\n\\nError: ${syncState.lastError}` : ''}\\n`;\n}\n\nexport function registerRenderWidget(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'render_widget',\n 'Render a widget as an inline artifact in the chat. Returns a formatted markdown representation of the widget data.',\n {\n dashboard_id: z.string().describe('The dashboard ID.'),\n widget_id: z.string().describe('The widget ID.'),\n },\n async ({ dashboard_id, widget_id }) => {\n try {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`);\n }\n const widget = dashboard.widgets[widget_id];\n if (!widget) {\n return err('NOT_FOUND', `Widget \"${widget_id}\" not found`);\n }\n\n const syncState = await storage.getSyncState();\n\n if (widget.kind === 'status') {\n return {\n content: [\n {\n type: 'text' as const,\n text: renderStatus(widget.title, widget.source, syncState),\n },\n ],\n };\n }\n\n const { connectorId } = widget.metric;\n const connectorEntry = runtime\n .getConnectors()\n .find((e) => e.connector.id === connectorId);\n if (!connectorEntry) {\n return err(\n 'CONNECTOR_NOT_FOUND',\n `Connector \"${connectorId}\" not found`,\n );\n }\n\n const handle = storage.getStorageHandle(connectorId);\n const data = await computeMetric(handle, widget.metric);\n\n let rendered: string;\n\n if (widget.kind === 'stat') {\n rendered = renderStat(widget.title, data);\n } else if (widget.kind === 'timeseries') {\n const points = Array.isArray(data)\n ? (data as Array<{ date: string; value: unknown }>)\n : [];\n rendered = renderTimeseries(widget.title, points);\n } else if (widget.kind === 'distribution') {\n rendered = renderDistribution(widget.title, data);\n } else {\n rendered = `## ${(widget as { title: string }).title}\\n\\n\\`\\`\\`json\\n${JSON.stringify(data, null, 2)}\\n\\`\\`\\`\\n`;\n }\n\n const cachedNote = syncState.lastSyncAt\n ? `\\n_Cached at ${new Date(syncState.lastSyncAt).toUTCString()}_`\n : '\\n_Not yet synced_';\n\n return {\n content: [{ type: 'text' as const, text: rendered + cachedNote }],\n };\n } catch (e) {\n return err('RENDER_FAILED', e instanceof Error ? e.message : String(e));\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nconst SECRET_NAME_RE = /^[A-Z][A-Z0-9_]*$/;\n\nexport function registerSetSecret(\n server: McpServer,\n trackedSecrets: Set<string>,\n options: Pick<McpServerOptions, 'onSetSecret'>,\n): void {\n server.tool(\n 'set_secret',\n 'Set a secret for the running Rawdash instance. Secret names must be uppercase (e.g. GITHUB_TOKEN). For OSS, secrets are stored in process.env (runtime-only, not persisted).',\n {\n name: z\n .string()\n .describe(\n 'Secret name — uppercase letters, digits, and underscores; must start with a letter (e.g. GITHUB_TOKEN).',\n ),\n value: z.string().describe('The secret value.'),\n },\n async ({ name, value }) => {\n if (!SECRET_NAME_RE.test(name)) {\n return err(\n 'INVALID_NAME',\n `Secret name \"${name}\" is invalid. Must match /^[A-Z][A-Z0-9_]*$/.`,\n );\n }\n\n try {\n if (options.onSetSecret) {\n await options.onSetSecret(name, value);\n } else {\n const env = (\n globalThis as {\n process?: { env?: Record<string, string | undefined> };\n }\n ).process?.env;\n if (!env) {\n return err(\n 'NO_SECRET_BACKEND',\n 'No secret backend is configured for this runtime.',\n );\n }\n env[name] = value;\n }\n } catch (e) {\n return err(\n 'SET_SECRET_FAILED',\n e instanceof Error ? e.message : 'Failed to set secret.',\n );\n }\n\n trackedSecrets.add(name);\n\n return text({ set: name });\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerTriggerSync(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'trigger_sync',\n 'Kick off a sync for one or all connectors. Runs in the background and returns immediately once started.',\n {\n connector_id: z\n .string()\n .optional()\n .describe('Connector ID to sync. Omit to sync all connectors.'),\n mode: z\n .enum(['full', 'latest'])\n .optional()\n .describe(\n '\"latest\" (default) fetches only recent data. \"full\" re-fetches all historical data.',\n ),\n },\n async ({ connector_id, mode = 'latest' }) => {\n const connectors = runtime.getConnectors();\n const targets = connector_id\n ? connectors.filter((e) => e.connector.id === connector_id)\n : connectors;\n\n if (connector_id && targets.length === 0) {\n return err('NOT_FOUND', `Connector \"${connector_id}\" not found`);\n }\n\n const acquired = await storage.setSyncing();\n if (!acquired) {\n return err('ALREADY_SYNCING', 'A sync is already in progress');\n }\n\n const controller = new AbortController();\n\n const run = async () => {\n try {\n for (const entry of targets) {\n const handle = storage.getStorageHandle(entry.connector.id);\n await entry.connector.sync(\n { mode: mode ?? 'latest' },\n handle,\n controller.signal,\n );\n }\n await storage.setSyncSuccess();\n } catch (e) {\n const message = e instanceof Error ? e.message : String(e);\n try {\n await storage.setSyncError(message);\n } catch {\n // Intentionally swallow to avoid unhandled background rejection.\n }\n }\n };\n\n void run().catch(() => {\n // Defensive terminal catch for any unexpected rejection path.\n });\n\n return text({\n triggered: true,\n connectors: targets.map((e) => e.connector.id),\n mode,\n });\n },\n );\n}\n"],"mappings":"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","names":["z","z","z","z","syncState","z","z","computeMetric","z","z","computeMetric","z","z","z","z"]}
1
+ {"version":3,"sources":["../src/server.ts","../src/runtime-config.ts","../src/tools/add-connector.ts","../src/tools/shared.ts","../src/tools/list-connectors.ts","../src/tools/list-dashboards.ts","../src/tools/list-secrets.ts","../src/tools/list-widgets.ts","../src/tools/read-widget.ts","../src/tools/remove-connector.ts","../src/tools/render-widget.ts","../src/tools/set-secret.ts","../src/tools/trigger-sync.ts"],"sourcesContent":["import { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport { McpRuntime } from './runtime-config';\nimport { registerAddConnector } from './tools/add-connector';\nimport { registerListConnectors } from './tools/list-connectors';\nimport { registerListDashboards } from './tools/list-dashboards';\nimport { registerListSecrets } from './tools/list-secrets';\nimport { registerListWidgets } from './tools/list-widgets';\nimport { registerReadWidget } from './tools/read-widget';\nimport { registerRemoveConnector } from './tools/remove-connector';\nimport { registerRenderWidget } from './tools/render-widget';\nimport { registerSetSecret } from './tools/set-secret';\nimport { registerTriggerSync } from './tools/trigger-sync';\nimport type { McpServerOptions } from './types';\n\nexport function createMcpServer(options: McpServerOptions): McpServer {\n const { name = 'rawdash', version = '1.0.0', config, storage } = options;\n\n const server = new McpServer({ name, version });\n const runtime = new McpRuntime(config);\n const trackedSecrets = new Set<string>();\n\n registerListDashboards(server, runtime);\n registerListWidgets(server, runtime);\n registerReadWidget(server, runtime, storage);\n registerRenderWidget(server, runtime, storage);\n registerListConnectors(server, runtime, storage);\n registerAddConnector(server, runtime, options);\n registerRemoveConnector(server, runtime, options);\n registerSetSecret(server, trackedSecrets, options);\n registerListSecrets(server, trackedSecrets, options);\n registerTriggerSync(server, runtime, storage, options);\n\n return server;\n}\n","import type {\n ConfiguredConnector,\n Dashboard,\n DashboardConfig,\n} from '@rawdash/core';\n\nexport class McpRuntime {\n private connectors: ConfiguredConnector[];\n private readonly dashboards: Record<string, Dashboard>;\n\n constructor(config: DashboardConfig) {\n this.connectors = [...config.connectors];\n this.dashboards = config.dashboards;\n }\n\n getConnectors(): ConfiguredConnector[] {\n return [...this.connectors];\n }\n\n getDashboards(): Record<string, Dashboard> {\n return this.dashboards;\n }\n\n addConnector(entry: ConfiguredConnector): void {\n if (this.connectors.some((e) => e.name === entry.name)) {\n throw new Error(`Connector \"${entry.name}\" already exists`);\n }\n this.connectors = [...this.connectors, entry];\n }\n\n removeConnector(name: string): boolean {\n const before = this.connectors.length;\n this.connectors = this.connectors.filter((e) => e.name !== name);\n return this.connectors.length < before;\n }\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport type { ConfiguredConnector } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerAddConnector(\n server: McpServer,\n runtime: McpRuntime,\n options: Pick<McpServerOptions, 'connectorFactories' | 'onAddConnector'>,\n): void {\n server.tool(\n 'add_connector',\n \"Add a connector instance to this Rawdash instance. Settings are validated against the connector's configFields schema. For OSS, this is a runtime-only change (not persisted to disk).\",\n {\n connector_type: z\n .string()\n .describe(\n 'The connector type ID (e.g. \"github-actions\"). Use list_connectors to see available types.',\n ),\n name: z\n .string()\n .describe(\n 'Instance name for this connector (must be unique within the deployment).',\n ),\n settings: z\n .record(z.string(), z.unknown())\n .describe(\n 'Connector settings as a JSON object. Schema depends on the connector type.',\n ),\n },\n async ({ connector_type, name, settings }) => {\n const factories = options.connectorFactories ?? [];\n const factory = factories.find((f) => f.id === connector_type);\n if (!factory) {\n const available = factories.map((f) => f.id);\n return err(\n 'UNKNOWN_CONNECTOR_TYPE',\n available.length > 0\n ? `Unknown connector type \"${connector_type}\". Available: ${available.join(', ')}`\n : `Unknown connector type \"${connector_type}\". No connector factories are registered.`,\n );\n }\n\n const parsed = factory.configFields.safeParse(settings);\n if (!parsed.success) {\n const issues = parsed.error.issues\n .map((i) => `${i.path.join('.')}: ${i.message}`)\n .join('; ');\n return err('INVALID_SETTINGS', `Invalid settings: ${issues}`);\n }\n\n const entry: ConfiguredConnector = {\n name,\n connectorId: connector_type,\n config: parsed.data as Record<string, unknown>,\n };\n\n try {\n runtime.addConnector(entry);\n } catch (e) {\n return err(\n 'ALREADY_EXISTS',\n e instanceof Error ? e.message : String(e),\n );\n }\n\n try {\n await options.onAddConnector?.(entry);\n } catch (e) {\n return err(\n 'ON_ADD_CONNECTOR_FAILED',\n `Connector added in-memory but post-add callback failed: ${e instanceof Error ? e.message : String(e)}`,\n );\n }\n\n return text({ added: entry.name, idempotent: false });\n },\n );\n}\n","function safeStringify(data: unknown): string {\n try {\n return JSON.stringify(\n data,\n (_key, value) => (typeof value === 'bigint' ? value.toString() : value),\n 2,\n );\n } catch {\n return JSON.stringify(\n {\n error: {\n code: 'SERIALIZATION_ERROR',\n message: 'Failed to serialize tool response',\n },\n },\n null,\n 2,\n );\n }\n}\n\nexport function text(data: unknown) {\n return {\n content: [{ type: 'text' as const, text: safeStringify(data) }],\n };\n}\n\nexport function err(code: string, message: string) {\n return text({ error: { code, message } });\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerListConnectors(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'list_connectors',\n 'List all configured connectors and their sync status.',\n {},\n async () => {\n try {\n const syncState = await storage.getSyncState();\n const connectors = runtime.getConnectors().map((entry) => ({\n id: entry.name,\n connectorId: entry.connectorId,\n syncStatus: syncState.status,\n lastSyncAt: syncState.lastSyncAt,\n lastError: syncState.lastError,\n configKeys: Object.keys(entry.config),\n }));\n return text({ connectors, overallSyncStatus: syncState.status });\n } catch (e) {\n return err(\n 'LIST_CONNECTORS_FAILED',\n e instanceof Error ? e.message : String(e),\n );\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpRuntime } from '../runtime-config';\nimport { text } from './shared';\n\nexport function registerListDashboards(\n server: McpServer,\n runtime: McpRuntime,\n): void {\n server.tool(\n 'list_dashboards',\n 'List all dashboards available on this Rawdash instance.',\n {},\n () => {\n const dashboards = Object.entries(runtime.getDashboards()).map(\n ([id, dashboard]) => ({\n id,\n widgetCount: Object.keys(dashboard.widgets).length,\n widgetIds: Object.keys(dashboard.widgets),\n }),\n );\n return Promise.resolve(text({ dashboards }));\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\n\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerListSecrets(\n server: McpServer,\n trackedSecrets: Set<string>,\n options: Pick<McpServerOptions, 'listSecrets'>,\n): void {\n server.tool(\n 'list_secrets',\n 'List secret names known to this Rawdash instance. Values are never returned.',\n {},\n async () => {\n try {\n const names = options.listSecrets\n ? await options.listSecrets()\n : [...trackedSecrets];\n return text({ secrets: [...names].sort() });\n } catch (e) {\n return err(\n 'LIST_SECRETS_FAILED',\n e instanceof Error ? e.message : 'Failed to list secrets.',\n );\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport { err, text } from './shared';\n\nexport function registerListWidgets(\n server: McpServer,\n runtime: McpRuntime,\n): void {\n server.tool(\n 'list_widgets',\n 'List all widgets on a dashboard.',\n {\n dashboard_id: z\n .string()\n .describe('The dashboard ID to list widgets for.'),\n },\n ({ dashboard_id }) => {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return Promise.resolve(\n err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`),\n );\n }\n const widgets = Object.entries(dashboard.widgets).map(([id, widget]) => ({\n id,\n kind: widget.kind,\n title: widget.title,\n ...(widget.kind !== 'status'\n ? { connectorId: widget.metric.connectorId }\n : { source: widget.source }),\n }));\n return Promise.resolve(text({ dashboard_id, widgets }));\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { computeMetric } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerReadWidget(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'read_widget',\n 'Fetch resolved widget data. Returns the same shape as the HTTP read API.',\n {\n dashboard_id: z.string().describe('The dashboard ID.'),\n widget_id: z.string().describe('The widget ID.'),\n },\n async ({ dashboard_id, widget_id }) => {\n try {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`);\n }\n const widget = dashboard.widgets[widget_id];\n if (!widget) {\n return err('NOT_FOUND', `Widget \"${widget_id}\" not found`);\n }\n\n if (widget.kind === 'status') {\n const syncState = await storage.getSyncState();\n return text({\n id: widget_id,\n widgetId: widget_id,\n connectorId: widget.source,\n data: null,\n cachedAt: syncState.lastSyncAt,\n });\n }\n\n const { connectorId } = widget.metric;\n const connectorEntry = runtime\n .getConnectors()\n .find((e) => e.name === connectorId);\n if (!connectorEntry) {\n return err(\n 'CONNECTOR_NOT_FOUND',\n `Connector \"${connectorId}\" not found`,\n );\n }\n\n const handle = storage.getStorageHandle(connectorId);\n const data = await computeMetric(handle, widget.metric);\n const syncState = await storage.getSyncState();\n\n return text({\n id: widget_id,\n widgetId: widget_id,\n connectorId,\n data,\n cachedAt: syncState.lastSyncAt,\n });\n } catch (e) {\n return err('READ_FAILED', e instanceof Error ? e.message : String(e));\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerRemoveConnector(\n server: McpServer,\n runtime: McpRuntime,\n options: Pick<McpServerOptions, 'onRemoveConnector'>,\n): void {\n server.tool(\n 'remove_connector',\n 'Remove a connector from this Rawdash instance. For OSS, this is a runtime-only change. Idempotent — removing a non-existent connector returns success.',\n {\n connector_id: z.string().describe('The connector instance ID to remove.'),\n },\n async ({ connector_id }) => {\n const existed = runtime.removeConnector(connector_id);\n if (!existed) {\n return text({ removed: connector_id, existed: false });\n }\n\n try {\n await options.onRemoveConnector?.(connector_id);\n } catch (e) {\n return err(\n 'ON_REMOVE_CONNECTOR_FAILED',\n `Connector removed in-memory but post-remove callback failed: ${e instanceof Error ? e.message : String(e)}`,\n );\n }\n\n return text({ removed: connector_id, existed: true });\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { computeMetric } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err } from './shared';\n\nfunction renderStat(title: string, value: unknown): string {\n const display =\n typeof value === 'number'\n ? value.toLocaleString('en-US', { maximumFractionDigits: 2 })\n : String(value ?? '—');\n return `## ${title}\\n\\n**${display}**\\n`;\n}\n\nfunction renderTimeseries(\n title: string,\n points: Array<{ date: string; value: unknown }>,\n): string {\n if (points.length === 0) {\n return `## ${title}\\n\\n_No data_\\n`;\n }\n const rows = points\n .map(\n (p) =>\n `| ${p.date} | ${typeof p.value === 'number' ? p.value.toLocaleString('en-US', { maximumFractionDigits: 2 }) : String(p.value ?? '—')} |`,\n )\n .join('\\n');\n return `## ${title}\\n\\n| Date | Value |\\n|------|-------|\\n${rows}\\n`;\n}\n\nfunction renderDistribution(title: string, value: unknown): string {\n return `## ${title}\\n\\n\\`\\`\\`json\\n${JSON.stringify(value, null, 2)}\\n\\`\\`\\`\\n`;\n}\n\nfunction renderStatus(\n title: string,\n source: string,\n syncState: {\n status: string;\n lastSyncAt: string | null;\n lastError: string | null;\n },\n): string {\n const statusEmoji =\n syncState.status === 'succeeded' || syncState.status === 'idle'\n ? '✅'\n : syncState.status === 'queued' || syncState.status === 'running'\n ? '🔄'\n : '❌';\n const lastSync = syncState.lastSyncAt\n ? new Date(syncState.lastSyncAt).toUTCString()\n : 'never';\n return `## ${title}\\n\\n${statusEmoji} **${syncState.status}** (connector: \\`${source}\\`)\\n\\nLast sync: ${lastSync}${syncState.lastError ? `\\n\\nError: ${syncState.lastError}` : ''}\\n`;\n}\n\nexport function registerRenderWidget(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n): void {\n server.tool(\n 'render_widget',\n 'Render a widget as an inline artifact in the chat. Returns a formatted markdown representation of the widget data.',\n {\n dashboard_id: z.string().describe('The dashboard ID.'),\n widget_id: z.string().describe('The widget ID.'),\n },\n async ({ dashboard_id, widget_id }) => {\n try {\n const dashboard = runtime.getDashboards()[dashboard_id];\n if (!dashboard) {\n return err('NOT_FOUND', `Dashboard \"${dashboard_id}\" not found`);\n }\n const widget = dashboard.widgets[widget_id];\n if (!widget) {\n return err('NOT_FOUND', `Widget \"${widget_id}\" not found`);\n }\n\n const syncState = await storage.getSyncState();\n\n if (widget.kind === 'status') {\n return {\n content: [\n {\n type: 'text' as const,\n text: renderStatus(widget.title, widget.source, syncState),\n },\n ],\n };\n }\n\n const { connectorId } = widget.metric;\n const connectorEntry = runtime\n .getConnectors()\n .find((e) => e.name === connectorId);\n if (!connectorEntry) {\n return err(\n 'CONNECTOR_NOT_FOUND',\n `Connector \"${connectorId}\" not found`,\n );\n }\n\n const handle = storage.getStorageHandle(connectorId);\n const data = await computeMetric(handle, widget.metric);\n\n let rendered: string;\n\n if (widget.kind === 'stat') {\n rendered = renderStat(widget.title, data);\n } else if (widget.kind === 'timeseries') {\n const points = Array.isArray(data)\n ? (data as Array<{ date: string; value: unknown }>)\n : [];\n rendered = renderTimeseries(widget.title, points);\n } else if (widget.kind === 'distribution') {\n rendered = renderDistribution(widget.title, data);\n } else {\n rendered = `## ${(widget as { title: string }).title}\\n\\n\\`\\`\\`json\\n${JSON.stringify(data, null, 2)}\\n\\`\\`\\`\\n`;\n }\n\n const cachedNote = syncState.lastSyncAt\n ? `\\n_Cached at ${new Date(syncState.lastSyncAt).toUTCString()}_`\n : '\\n_Not yet synced_';\n\n return {\n content: [{ type: 'text' as const, text: rendered + cachedNote }],\n };\n } catch (e) {\n return err('RENDER_FAILED', e instanceof Error ? e.message : String(e));\n }\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { z } from 'zod';\n\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nconst SECRET_NAME_RE = /^[A-Z][A-Z0-9_]*$/;\n\nexport function registerSetSecret(\n server: McpServer,\n trackedSecrets: Set<string>,\n options: Pick<McpServerOptions, 'onSetSecret'>,\n): void {\n server.tool(\n 'set_secret',\n 'Set a secret for the running Rawdash instance. Secret names must be uppercase (e.g. GITHUB_TOKEN). For OSS, secrets are stored in process.env (runtime-only, not persisted).',\n {\n name: z\n .string()\n .describe(\n 'Secret name — uppercase letters, digits, and underscores; must start with a letter (e.g. GITHUB_TOKEN).',\n ),\n value: z.string().describe('The secret value.'),\n },\n async ({ name, value }) => {\n if (!SECRET_NAME_RE.test(name)) {\n return err(\n 'INVALID_NAME',\n `Secret name \"${name}\" is invalid. Must match /^[A-Z][A-Z0-9_]*$/.`,\n );\n }\n\n try {\n if (options.onSetSecret) {\n await options.onSetSecret(name, value);\n } else {\n const env = (\n globalThis as {\n process?: { env?: Record<string, string | undefined> };\n }\n ).process?.env;\n if (!env) {\n return err(\n 'NO_SECRET_BACKEND',\n 'No secret backend is configured for this runtime.',\n );\n }\n env[name] = value;\n }\n } catch (e) {\n return err(\n 'SET_SECRET_FAILED',\n e instanceof Error ? e.message : 'Failed to set secret.',\n );\n }\n\n trackedSecrets.add(name);\n\n return text({ set: name });\n },\n );\n}\n","import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp';\nimport { instantiateConnector } from '@rawdash/core';\nimport { z } from 'zod';\n\nimport type { McpRuntime } from '../runtime-config';\nimport type { McpServerOptions } from '../types';\nimport { err, text } from './shared';\n\nexport function registerTriggerSync(\n server: McpServer,\n runtime: McpRuntime,\n storage: McpServerOptions['storage'],\n options: Pick<McpServerOptions, 'connectorRegistry' | 'secretsResolver'>,\n): void {\n server.tool(\n 'trigger_sync',\n 'Kick off a sync for one or all connectors. Runs in the background and returns immediately once started.',\n {\n connector_id: z\n .string()\n .optional()\n .describe(\n 'Connector instance name to sync. Omit to sync all connectors.',\n ),\n mode: z\n .enum(['full', 'latest'])\n .optional()\n .describe(\n '\"latest\" (default) fetches only recent data. \"full\" re-fetches all historical data.',\n ),\n },\n async ({ connector_id, mode = 'latest' }) => {\n if (!options.connectorRegistry) {\n return err(\n 'NO_REGISTRY',\n 'trigger_sync requires connectorRegistry on McpServerOptions',\n );\n }\n const connectors = runtime.getConnectors();\n const targets = connector_id\n ? connectors.filter((e) => e.name === connector_id)\n : connectors;\n\n if (connector_id && targets.length === 0) {\n return err('NOT_FOUND', `Connector \"${connector_id}\" not found`);\n }\n\n const queued = await storage.markSyncQueued();\n if (!queued) {\n return err('ALREADY_SYNCING', 'A sync is already in progress');\n }\n const acquired = await storage.markSyncRunning();\n if (!acquired) {\n return err('ALREADY_SYNCING', 'A sync is already in progress');\n }\n\n const controller = new AbortController();\n\n const run = async () => {\n try {\n for (const entry of targets) {\n const connector = instantiateConnector(\n entry,\n options.connectorRegistry!,\n options.secretsResolver,\n );\n const handle = storage.getStorageHandle(entry.name);\n await connector.sync(\n { mode: mode ?? 'latest' },\n handle,\n controller.signal,\n );\n }\n await storage.markSyncSucceeded();\n } catch (e) {\n const message = e instanceof Error ? e.message : String(e);\n try {\n await storage.markSyncFailed(message);\n } catch (markErr) {\n console.error('Failed to record sync failure in storage:', markErr);\n }\n }\n };\n\n void run().catch((rejection) => {\n console.error('Unexpected background sync rejection:', rejection);\n });\n\n return text({\n triggered: true,\n connectors: targets.map((e) => e.name),\n mode,\n });\n },\n );\n}\n"],"mappings":"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","names":["z","z","z","z","syncState","z","z","computeMetric","z","z","computeMetric","z","z","z","z"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rawdash/mcp",
3
- "version": "0.12.0",
3
+ "version": "0.15.0",
4
4
  "description": "Rawdash MCP server — exposes dashboards, widgets, and connectors to AI agents",
5
5
  "license": "Apache-2.0",
6
6
  "type": "module",
@@ -24,8 +24,7 @@
24
24
  "dependencies": {
25
25
  "@modelcontextprotocol/sdk": "^1.12.0",
26
26
  "zod": "^4.4.3",
27
- "@rawdash/core": "0.12.0",
28
- "@rawdash/server": "0.12.0"
27
+ "@rawdash/core": "0.15.0"
29
28
  },
30
29
  "devDependencies": {
31
30
  "tsup": "^8.0.0",