@modelcontextprotocol/server-pdf 1.2.2 → 1.3.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/server.js CHANGED
@@ -28266,8 +28266,8 @@ class j {
28266
28266
  sessionId;
28267
28267
  setProtocolVersion;
28268
28268
  }
28269
- var b = exports_external.union([exports_external.literal("light"), exports_external.literal("dark")]).describe("Color theme preference for the host environment.");
28270
- var L = exports_external.union([exports_external.literal("inline"), exports_external.literal("fullscreen"), exports_external.literal("pip")]).describe("Display mode for UI presentation.");
28269
+ var g = exports_external.union([exports_external.literal("light"), exports_external.literal("dark")]).describe("Color theme preference for the host environment.");
28270
+ var G = exports_external.union([exports_external.literal("inline"), exports_external.literal("fullscreen"), exports_external.literal("pip")]).describe("Display mode for UI presentation.");
28271
28271
  var i = exports_external.union([exports_external.literal("--color-background-primary"), exports_external.literal("--color-background-secondary"), exports_external.literal("--color-background-tertiary"), exports_external.literal("--color-background-inverse"), exports_external.literal("--color-background-ghost"), exports_external.literal("--color-background-info"), exports_external.literal("--color-background-danger"), exports_external.literal("--color-background-success"), exports_external.literal("--color-background-warning"), exports_external.literal("--color-background-disabled"), exports_external.literal("--color-text-primary"), exports_external.literal("--color-text-secondary"), exports_external.literal("--color-text-tertiary"), exports_external.literal("--color-text-inverse"), exports_external.literal("--color-text-ghost"), exports_external.literal("--color-text-info"), exports_external.literal("--color-text-danger"), exports_external.literal("--color-text-success"), exports_external.literal("--color-text-warning"), exports_external.literal("--color-text-disabled"), exports_external.literal("--color-border-primary"), exports_external.literal("--color-border-secondary"), exports_external.literal("--color-border-tertiary"), exports_external.literal("--color-border-inverse"), exports_external.literal("--color-border-ghost"), exports_external.literal("--color-border-info"), exports_external.literal("--color-border-danger"), exports_external.literal("--color-border-success"), exports_external.literal("--color-border-warning"), exports_external.literal("--color-border-disabled"), exports_external.literal("--color-ring-primary"), exports_external.literal("--color-ring-secondary"), exports_external.literal("--color-ring-inverse"), exports_external.literal("--color-ring-info"), exports_external.literal("--color-ring-danger"), exports_external.literal("--color-ring-success"), exports_external.literal("--color-ring-warning"), exports_external.literal("--font-sans"), exports_external.literal("--font-mono"), exports_external.literal("--font-weight-normal"), exports_external.literal("--font-weight-medium"), exports_external.literal("--font-weight-semibold"), exports_external.literal("--font-weight-bold"), exports_external.literal("--font-text-xs-size"), exports_external.literal("--font-text-sm-size"), exports_external.literal("--font-text-md-size"), exports_external.literal("--font-text-lg-size"), exports_external.literal("--font-heading-xs-size"), exports_external.literal("--font-heading-sm-size"), exports_external.literal("--font-heading-md-size"), exports_external.literal("--font-heading-lg-size"), exports_external.literal("--font-heading-xl-size"), exports_external.literal("--font-heading-2xl-size"), exports_external.literal("--font-heading-3xl-size"), exports_external.literal("--font-text-xs-line-height"), exports_external.literal("--font-text-sm-line-height"), exports_external.literal("--font-text-md-line-height"), exports_external.literal("--font-text-lg-line-height"), exports_external.literal("--font-heading-xs-line-height"), exports_external.literal("--font-heading-sm-line-height"), exports_external.literal("--font-heading-md-line-height"), exports_external.literal("--font-heading-lg-line-height"), exports_external.literal("--font-heading-xl-line-height"), exports_external.literal("--font-heading-2xl-line-height"), exports_external.literal("--font-heading-3xl-line-height"), exports_external.literal("--border-radius-xs"), exports_external.literal("--border-radius-sm"), exports_external.literal("--border-radius-md"), exports_external.literal("--border-radius-lg"), exports_external.literal("--border-radius-xl"), exports_external.literal("--border-radius-full"), exports_external.literal("--border-width-regular"), exports_external.literal("--shadow-hairline"), exports_external.literal("--shadow-sm"), exports_external.literal("--shadow-md"), exports_external.literal("--shadow-lg")]).describe("CSS variable keys available to MCP apps for theming.");
28272
28272
  var o = exports_external.record(i.describe(`Style variables for theming MCP apps.
28273
28273
 
@@ -28302,15 +28302,16 @@ var t = exports_external.object({ method: exports_external.literal("ui/notificat
28302
28302
  var A = exports_external.object({ method: exports_external.literal("ui/notifications/tool-input"), params: exports_external.object({ arguments: exports_external.record(exports_external.string(), exports_external.unknown().describe("Complete tool call arguments as key-value pairs.")).optional().describe("Complete tool call arguments as key-value pairs.") }) });
28303
28303
  var P = exports_external.object({ method: exports_external.literal("ui/notifications/tool-input-partial"), params: exports_external.object({ arguments: exports_external.record(exports_external.string(), exports_external.unknown().describe("Partial tool call arguments (incomplete, may change).")).optional().describe("Partial tool call arguments (incomplete, may change).") }) });
28304
28304
  var H = exports_external.object({ method: exports_external.literal("ui/notifications/tool-cancelled"), params: exports_external.object({ reason: exports_external.string().optional().describe('Optional reason for the cancellation (e.g., "user action", "timeout").') }) });
28305
- var g = exports_external.object({ fonts: exports_external.string().optional() });
28306
- var C = exports_external.object({ variables: o.optional().describe("CSS variables for theming the app."), css: g.optional().describe("CSS blocks that apps can inject.") });
28305
+ var S = exports_external.object({ fonts: exports_external.string().optional() });
28306
+ var y = exports_external.object({ variables: o.optional().describe("CSS variables for theming the app."), css: S.optional().describe("CSS blocks that apps can inject.") });
28307
28307
  var _ = exports_external.object({ method: exports_external.literal("ui/resource-teardown"), params: exports_external.object({}) });
28308
28308
  var e = exports_external.record(exports_external.string(), exports_external.unknown());
28309
28309
  var q = exports_external.object({ text: exports_external.object({}).optional().describe("Host supports text content blocks."), image: exports_external.object({}).optional().describe("Host supports image content blocks."), audio: exports_external.object({}).optional().describe("Host supports audio content blocks."), resource: exports_external.object({}).optional().describe("Host supports resource content blocks."), resourceLink: exports_external.object({}).optional().describe("Host supports resource link content blocks."), structuredContent: exports_external.object({}).optional().describe("Host supports structured content.") });
28310
- var y = exports_external.object({ experimental: exports_external.object({}).optional().describe("Experimental features (structure TBD)."), openLinks: exports_external.object({}).optional().describe("Host supports opening external URLs."), downloadFile: exports_external.object({}).optional().describe("Host supports file downloads via ui/download-file."), serverTools: exports_external.object({ listChanged: exports_external.boolean().optional().describe("Host supports tools/list_changed notifications.") }).optional().describe("Host can proxy tool calls to the MCP server."), serverResources: exports_external.object({ listChanged: exports_external.boolean().optional().describe("Host supports resources/list_changed notifications.") }).optional().describe("Host can proxy resource reads to the MCP server."), logging: exports_external.object({}).optional().describe("Host accepts log messages."), sandbox: exports_external.object({ permissions: K.optional().describe("Permissions granted by the host (camera, microphone, geolocation)."), csp: B.optional().describe("CSP domains approved by the host.") }).optional().describe("Sandbox configuration applied by the host."), updateModelContext: q.optional().describe("Host accepts context updates (ui/update-model-context) to be included in the model's context for future turns."), message: q.optional().describe("Host supports receiving content messages (ui/message) from the view.") });
28311
- var u = exports_external.object({ experimental: exports_external.object({}).optional().describe("Experimental features (structure TBD)."), tools: exports_external.object({ listChanged: exports_external.boolean().optional().describe("App supports tools/list_changed notifications.") }).optional().describe("App exposes MCP-style tools that the host can call."), availableDisplayModes: exports_external.array(L).optional().describe("Display modes the app supports.") });
28312
- var QQ = exports_external.object({ method: exports_external.literal("ui/notifications/initialized"), params: exports_external.object({}).optional() });
28313
- var ZQ = exports_external.object({ csp: B.optional().describe("Content Security Policy configuration for UI resources."), permissions: K.optional().describe("Sandbox permissions requested by the UI resource."), domain: exports_external.string().optional().describe(`Dedicated origin for view sandbox.
28310
+ var QQ = exports_external.object({ method: exports_external.literal("ui/notifications/request-teardown"), params: exports_external.object({}).optional() });
28311
+ var C = exports_external.object({ experimental: exports_external.object({}).optional().describe("Experimental features (structure TBD)."), openLinks: exports_external.object({}).optional().describe("Host supports opening external URLs."), downloadFile: exports_external.object({}).optional().describe("Host supports file downloads via ui/download-file."), serverTools: exports_external.object({ listChanged: exports_external.boolean().optional().describe("Host supports tools/list_changed notifications.") }).optional().describe("Host can proxy tool calls to the MCP server."), serverResources: exports_external.object({ listChanged: exports_external.boolean().optional().describe("Host supports resources/list_changed notifications.") }).optional().describe("Host can proxy resource reads to the MCP server."), logging: exports_external.object({}).optional().describe("Host accepts log messages."), sandbox: exports_external.object({ permissions: K.optional().describe("Permissions granted by the host (camera, microphone, geolocation)."), csp: B.optional().describe("CSP domains approved by the host.") }).optional().describe("Sandbox configuration applied by the host."), updateModelContext: q.optional().describe("Host accepts context updates (ui/update-model-context) to be included in the model's context for future turns."), message: q.optional().describe("Host supports receiving content messages (ui/message) from the view.") });
28312
+ var f = exports_external.object({ experimental: exports_external.object({}).optional().describe("Experimental features (structure TBD)."), tools: exports_external.object({ listChanged: exports_external.boolean().optional().describe("App supports tools/list_changed notifications.") }).optional().describe("App exposes MCP-style tools that the host can call."), availableDisplayModes: exports_external.array(G).optional().describe("Display modes the app supports.") });
28313
+ var ZQ = exports_external.object({ method: exports_external.literal("ui/notifications/initialized"), params: exports_external.object({}).optional() });
28314
+ var $Q = exports_external.object({ csp: B.optional().describe("Content Security Policy configuration for UI resources."), permissions: K.optional().describe("Sandbox permissions requested by the UI resource."), domain: exports_external.string().optional().describe(`Dedicated origin for view sandbox.
28314
28315
 
28315
28316
  Useful when views need stable, dedicated origins for OAuth callbacks, CORS policies, or API key allowlists.
28316
28317
 
@@ -28325,27 +28326,27 @@ Boolean requesting whether a visible border and background is provided by the ho
28325
28326
  - \`true\`: request visible border + background
28326
28327
  - \`false\`: request no visible border + background
28327
28328
  - omitted: host decides border`) });
28328
- var $Q = exports_external.object({ method: exports_external.literal("ui/request-display-mode"), params: exports_external.object({ mode: L.describe("The display mode being requested.") }) });
28329
- var T = exports_external.object({ mode: L.describe("The display mode that was actually set. May differ from requested if not supported.") }).passthrough();
28330
- var f = exports_external.union([exports_external.literal("model"), exports_external.literal("app")]).describe("Tool visibility scope - who can access the tool.");
28331
- var JQ = exports_external.object({ resourceUri: exports_external.string().optional(), visibility: exports_external.array(f).optional().describe(`Who can access this tool. Default: ["model", "app"]
28329
+ var JQ = exports_external.object({ method: exports_external.literal("ui/request-display-mode"), params: exports_external.object({ mode: G.describe("The display mode being requested.") }) });
28330
+ var T = exports_external.object({ mode: G.describe("The display mode that was actually set. May differ from requested if not supported.") }).passthrough();
28331
+ var u = exports_external.union([exports_external.literal("model"), exports_external.literal("app")]).describe("Tool visibility scope - who can access the tool.");
28332
+ var XQ = exports_external.object({ resourceUri: exports_external.string().optional(), visibility: exports_external.array(u).optional().describe(`Who can access this tool. Default: ["model", "app"]
28332
28333
  - "model": Tool visible to and callable by the agent
28333
28334
  - "app": Tool callable by the app from this server only`) });
28334
- var kQ = exports_external.object({ mimeTypes: exports_external.array(exports_external.string()).optional().describe('Array of supported MIME types for UI resources.\nMust include `"text/html;profile=mcp-app"` for MCP Apps support.') });
28335
- var XQ = exports_external.object({ method: exports_external.literal("ui/download-file"), params: exports_external.object({ contents: exports_external.array(exports_external.union([EmbeddedResourceSchema, ResourceLinkSchema])).describe("Resource contents to download — embedded (inline data) or linked (host fetches). Uses standard MCP resource types.") }) });
28336
- var VQ = exports_external.object({ method: exports_external.literal("ui/message"), params: exports_external.object({ role: exports_external.literal("user").describe('Message role, currently only "user" is supported.'), content: exports_external.array(ContentBlockSchema).describe("Message content blocks (text, image, etc.).") }) });
28337
- var DQ = exports_external.object({ method: exports_external.literal("ui/notifications/sandbox-resource-ready"), params: exports_external.object({ html: exports_external.string().describe("HTML content to load into the inner iframe."), sandbox: exports_external.string().optional().describe("Optional override for the inner iframe's sandbox attribute."), csp: B.optional().describe("CSP configuration from resource metadata."), permissions: K.optional().describe("Sandbox permissions from resource metadata.") }) });
28338
- var U = exports_external.object({ method: exports_external.literal("ui/notifications/tool-result"), params: CallToolResultSchema.describe("Standard MCP tool execution result.") });
28339
- var k = exports_external.object({ toolInfo: exports_external.object({ id: RequestIdSchema.optional().describe("JSON-RPC id of the tools/call request."), tool: ToolSchema.describe("Tool definition including name, inputSchema, etc.") }).optional().describe("Metadata of the tool call that instantiated this App."), theme: b.optional().describe("Current color theme preference."), styles: C.optional().describe("Style configuration for theming the app."), displayMode: L.optional().describe("How the UI is currently displayed."), availableDisplayModes: exports_external.array(L).optional().describe("Display modes the host supports."), containerDimensions: exports_external.union([exports_external.object({ height: exports_external.number().describe("Fixed container height in pixels.") }), exports_external.object({ maxHeight: exports_external.union([exports_external.number(), exports_external.undefined()]).optional().describe("Maximum container height in pixels.") })]).and(exports_external.union([exports_external.object({ width: exports_external.number().describe("Fixed container width in pixels.") }), exports_external.object({ maxWidth: exports_external.union([exports_external.number(), exports_external.undefined()]).optional().describe("Maximum container width in pixels.") })])).optional().describe(`Container dimensions. Represents the dimensions of the iframe or other
28335
+ var RQ = exports_external.object({ mimeTypes: exports_external.array(exports_external.string()).optional().describe('Array of supported MIME types for UI resources.\nMust include `"text/html;profile=mcp-app"` for MCP Apps support.') });
28336
+ var VQ = exports_external.object({ method: exports_external.literal("ui/download-file"), params: exports_external.object({ contents: exports_external.array(exports_external.union([EmbeddedResourceSchema, ResourceLinkSchema])).describe("Resource contents to download — embedded (inline data) or linked (host fetches). Uses standard MCP resource types.") }) });
28337
+ var DQ = exports_external.object({ method: exports_external.literal("ui/message"), params: exports_external.object({ role: exports_external.literal("user").describe('Message role, currently only "user" is supported.'), content: exports_external.array(ContentBlockSchema).describe("Message content blocks (text, image, etc.).") }) });
28338
+ var GQ = exports_external.object({ method: exports_external.literal("ui/notifications/sandbox-resource-ready"), params: exports_external.object({ html: exports_external.string().describe("HTML content to load into the inner iframe."), sandbox: exports_external.string().optional().describe("Optional override for the inner iframe's sandbox attribute."), csp: B.optional().describe("CSP configuration from resource metadata."), permissions: K.optional().describe("Sandbox permissions from resource metadata.") }) });
28339
+ var E = exports_external.object({ method: exports_external.literal("ui/notifications/tool-result"), params: CallToolResultSchema.describe("Standard MCP tool execution result.") });
28340
+ var k = exports_external.object({ toolInfo: exports_external.object({ id: RequestIdSchema.optional().describe("JSON-RPC id of the tools/call request."), tool: ToolSchema.describe("Tool definition including name, inputSchema, etc.") }).optional().describe("Metadata of the tool call that instantiated this App."), theme: g.optional().describe("Current color theme preference."), styles: y.optional().describe("Style configuration for theming the app."), displayMode: G.optional().describe("How the UI is currently displayed."), availableDisplayModes: exports_external.array(G).optional().describe("Display modes the host supports."), containerDimensions: exports_external.union([exports_external.object({ height: exports_external.number().describe("Fixed container height in pixels.") }), exports_external.object({ maxHeight: exports_external.union([exports_external.number(), exports_external.undefined()]).optional().describe("Maximum container height in pixels.") })]).and(exports_external.union([exports_external.object({ width: exports_external.number().describe("Fixed container width in pixels.") }), exports_external.object({ maxWidth: exports_external.union([exports_external.number(), exports_external.undefined()]).optional().describe("Maximum container width in pixels.") })])).optional().describe(`Container dimensions. Represents the dimensions of the iframe or other
28340
28341
  container holding the app. Specify either width or maxWidth, and either height or maxHeight.`), locale: exports_external.string().optional().describe("User's language and region preference in BCP 47 format."), timeZone: exports_external.string().optional().describe("User's timezone in IANA format."), userAgent: exports_external.string().optional().describe("Host application identifier."), platform: exports_external.union([exports_external.literal("web"), exports_external.literal("desktop"), exports_external.literal("mobile")]).optional().describe("Platform type for responsive design decisions."), deviceCapabilities: exports_external.object({ touch: exports_external.boolean().optional().describe("Whether the device supports touch input."), hover: exports_external.boolean().optional().describe("Whether the device supports hover interactions.") }).optional().describe("Device input capabilities."), safeAreaInsets: exports_external.object({ top: exports_external.number().describe("Top safe area inset in pixels."), right: exports_external.number().describe("Right safe area inset in pixels."), bottom: exports_external.number().describe("Bottom safe area inset in pixels."), left: exports_external.number().describe("Left safe area inset in pixels.") }).optional().describe("Mobile safe area boundaries in pixels.") }).passthrough();
28341
- var E = exports_external.object({ method: exports_external.literal("ui/notifications/host-context-changed"), params: k.describe("Partial context update containing only changed fields.") });
28342
+ var R = exports_external.object({ method: exports_external.literal("ui/notifications/host-context-changed"), params: k.describe("Partial context update containing only changed fields.") });
28342
28343
  var LQ = exports_external.object({ method: exports_external.literal("ui/update-model-context"), params: exports_external.object({ content: exports_external.array(ContentBlockSchema).optional().describe("Context content blocks (text, image, etc.)."), structuredContent: exports_external.record(exports_external.string(), exports_external.unknown().describe("Structured content for machine-readable context data.")).optional().describe("Structured content for machine-readable context data.") }) });
28343
- var GQ = exports_external.object({ method: exports_external.literal("ui/initialize"), params: exports_external.object({ appInfo: ImplementationSchema.describe("App identification (name and version)."), appCapabilities: u.describe("Features and capabilities this app provides."), protocolVersion: exports_external.string().describe("Protocol version this app supports.") }) });
28344
- var R = exports_external.object({ protocolVersion: exports_external.string().describe('Negotiated protocol version string (e.g., "2025-11-21").'), hostInfo: ImplementationSchema.describe("Host application identification and version."), hostCapabilities: y.describe("Features and capabilities provided by the host."), hostContext: k.describe("Rich context about the host environment.") }).passthrough();
28344
+ var WQ = exports_external.object({ method: exports_external.literal("ui/initialize"), params: exports_external.object({ appInfo: ImplementationSchema.describe("App identification (name and version)."), appCapabilities: f.describe("Features and capabilities this app provides."), protocolVersion: exports_external.string().describe("Protocol version this app supports.") }) });
28345
+ var U = exports_external.object({ protocolVersion: exports_external.string().describe('Negotiated protocol version string (e.g., "2025-11-21").'), hostInfo: ImplementationSchema.describe("Host application identification and version."), hostCapabilities: C.describe("Features and capabilities provided by the host."), hostContext: k.describe("Rich context about the host environment.") }).passthrough();
28345
28346
  var v = "ui/resourceUri";
28346
28347
  var d = "text/html;profile=mcp-app";
28347
28348
 
28348
- class qQ extends Protocol {
28349
+ class OQ extends Protocol {
28349
28350
  _appInfo;
28350
28351
  _capabilities;
28351
28352
  options;
@@ -28377,13 +28378,13 @@ class qQ extends Protocol {
28377
28378
  this.setNotificationHandler(P, ($) => Z($.params));
28378
28379
  }
28379
28380
  set ontoolresult(Z) {
28380
- this.setNotificationHandler(U, ($) => Z($.params));
28381
+ this.setNotificationHandler(E, ($) => Z($.params));
28381
28382
  }
28382
28383
  set ontoolcancelled(Z) {
28383
28384
  this.setNotificationHandler(H, ($) => Z($.params));
28384
28385
  }
28385
28386
  set onhostcontextchanged(Z) {
28386
- this.setNotificationHandler(E, ($) => {
28387
+ this.setNotificationHandler(R, ($) => {
28387
28388
  this._hostContext = { ...this._hostContext, ...$.params }, Z($.params);
28388
28389
  });
28389
28390
  }
@@ -28445,6 +28446,9 @@ class qQ extends Protocol {
28445
28446
  downloadFile(Z, $) {
28446
28447
  return this.request({ method: "ui/download-file", params: Z }, I, $);
28447
28448
  }
28449
+ requestTeardown(Z = {}) {
28450
+ return this.notification({ method: "ui/notifications/request-teardown", params: Z });
28451
+ }
28448
28452
  requestDisplayMode(Z, $) {
28449
28453
  return this.request({ method: "ui/request-display-mode", params: Z }, T, $);
28450
28454
  }
@@ -28457,11 +28461,11 @@ class qQ extends Protocol {
28457
28461
  return;
28458
28462
  Z = true, requestAnimationFrame(() => {
28459
28463
  Z = false;
28460
- let V = document.documentElement, G = V.style.width, W = V.style.height;
28464
+ let V = document.documentElement, L = V.style.width, W = V.style.height;
28461
28465
  V.style.width = "fit-content", V.style.height = "max-content";
28462
- let M = V.getBoundingClientRect();
28463
- V.style.width = G, V.style.height = W;
28464
- let h = window.innerWidth - V.clientWidth, N = Math.ceil(M.width + h), Y = Math.ceil(M.height);
28466
+ let x = V.getBoundingClientRect();
28467
+ V.style.width = L, V.style.height = W;
28468
+ let h = window.innerWidth - V.clientWidth, N = Math.ceil(x.width + h), Y = Math.ceil(x.height);
28465
28469
  if (N !== $ || Y !== J)
28466
28470
  $ = N, J = Y, this.sendSizeChanged({ width: N, height: Y });
28467
28471
  });
@@ -28475,7 +28479,7 @@ class qQ extends Protocol {
28475
28479
  throw Error("App is already connected. Call close() before connecting again.");
28476
28480
  await super.connect(Z);
28477
28481
  try {
28478
- let J = await this.request({ method: "ui/initialize", params: { appCapabilities: this._capabilities, appInfo: this._appInfo, protocolVersion: F } }, R, $);
28482
+ let J = await this.request({ method: "ui/initialize", params: { appCapabilities: this._capabilities, appInfo: this._appInfo, protocolVersion: F } }, U, $);
28479
28483
  if (J === undefined)
28480
28484
  throw Error(`Server sent invalid initialize result: ${J}`);
28481
28485
  if (this._hostCapabilities = J.hostCapabilities, this._hostInfo = J.hostInfo, this._hostContext = J.hostContext, await this.notification({ method: "ui/notifications/initialized" }), this.options?.autoResize)
@@ -28485,19 +28489,42 @@ class qQ extends Protocol {
28485
28489
  }
28486
28490
  }
28487
28491
  }
28488
- function uZ(Z, $, J, X) {
28489
- let D = J._meta, V = D.ui, G = D[v], W = D;
28490
- if (V?.resourceUri && !G)
28492
+ function hZ(Z, $, J, X) {
28493
+ let D = J._meta, V = D.ui, L = D[v], W = D;
28494
+ if (V?.resourceUri && !L)
28491
28495
  W = { ...D, [v]: V.resourceUri };
28492
- else if (G && !V?.resourceUri)
28493
- W = { ...D, ui: { ...V, resourceUri: G } };
28496
+ else if (L && !V?.resourceUri)
28497
+ W = { ...D, ui: { ...V, resourceUri: L } };
28494
28498
  return Z.registerTool($, { ...J, _meta: W }, X);
28495
28499
  }
28496
- function fZ(Z, $, J, X, D) {
28500
+ function mZ(Z, $, J, X, D) {
28497
28501
  return Z.registerResource($, J, { mimeType: d, ...X }, D);
28498
28502
  }
28499
28503
 
28500
28504
  // server.ts
28505
+ import {
28506
+ getDocument,
28507
+ VerbosityLevel,
28508
+ version as PDFJS_VERSION
28509
+ } from "pdfjs-dist/legacy/build/pdf.mjs";
28510
+ var STANDARD_FONT_DATA_URL = `https://unpkg.com/pdfjs-dist@${PDFJS_VERSION}/standard_fonts/`;
28511
+ var STANDARD_FONT_ORIGIN = "https://unpkg.com";
28512
+
28513
+ class FetchStandardFontDataFactory {
28514
+ baseUrl;
28515
+ constructor({ baseUrl = null }) {
28516
+ this.baseUrl = baseUrl;
28517
+ }
28518
+ async fetch({ filename }) {
28519
+ if (!this.baseUrl)
28520
+ throw new Error("standardFontDataUrl not provided");
28521
+ const url2 = `${this.baseUrl}${filename}`;
28522
+ const res = await globalThis.fetch(url2);
28523
+ if (!res.ok)
28524
+ throw new Error(`Failed to fetch ${url2}: ${res.status}`);
28525
+ return new Uint8Array(await res.arrayBuffer());
28526
+ }
28527
+ }
28501
28528
  var DEFAULT_PDF = "https://arxiv.org/pdf/1706.03762";
28502
28529
  var MAX_CHUNK_BYTES = 512 * 1024;
28503
28530
  var RESOURCE_URI = "ui://pdf-viewer/mcp-app.html";
@@ -28506,7 +28533,158 @@ var CACHE_MAX_LIFETIME_MS = 60000;
28506
28533
  var CACHE_MAX_PDF_SIZE_BYTES = 50 * 1024 * 1024;
28507
28534
  var allowedLocalFiles = new Set;
28508
28535
  var allowedLocalDirs = new Set;
28536
+ var cliLocalFiles = new Set;
28537
+ var writeFlags = {
28538
+ allowUploadsRoot: false
28539
+ };
28540
+ function isWritablePath(resolved) {
28541
+ if (cliLocalFiles.has(resolved))
28542
+ return true;
28543
+ if (allowedLocalFiles.has(resolved))
28544
+ return false;
28545
+ return [...allowedLocalDirs].some((dir) => {
28546
+ if (!isAncestorDir(dir, resolved))
28547
+ return false;
28548
+ if (!writeFlags.allowUploadsRoot && path.basename(dir) === "uploads") {
28549
+ return false;
28550
+ }
28551
+ return true;
28552
+ });
28553
+ }
28509
28554
  var DIST_DIR = import.meta.filename.endsWith(".ts") ? path.join(import.meta.dirname, "dist") : import.meta.dirname;
28555
+ var COMMAND_TTL_MS = 60000;
28556
+ var SWEEP_INTERVAL_MS = 30000;
28557
+ var POLL_BATCH_WAIT_MS = 200;
28558
+ var LONG_POLL_TIMEOUT_MS = 30000;
28559
+ var FormField = exports_external.object({
28560
+ name: exports_external.string(),
28561
+ value: exports_external.union([exports_external.string(), exports_external.boolean()])
28562
+ });
28563
+ var PageInterval = exports_external.object({
28564
+ start: exports_external.number().min(1).optional(),
28565
+ end: exports_external.number().min(1).optional()
28566
+ });
28567
+ var GET_PAGES_TIMEOUT_MS = 45000;
28568
+ var pendingPageRequests = new Map;
28569
+ function waitForPageData(requestId, signal) {
28570
+ return new Promise((resolve, reject) => {
28571
+ const settle = (v2) => {
28572
+ clearTimeout(timer);
28573
+ signal?.removeEventListener("abort", onAbort);
28574
+ pendingPageRequests.delete(requestId);
28575
+ v2 instanceof Error ? reject(v2) : resolve(v2);
28576
+ };
28577
+ const onAbort = () => settle(new Error("interact request cancelled"));
28578
+ const timer = setTimeout(() => settle(new Error("Timeout waiting for page data from viewer")), GET_PAGES_TIMEOUT_MS);
28579
+ signal?.addEventListener("abort", onAbort);
28580
+ pendingPageRequests.set(requestId, settle);
28581
+ });
28582
+ }
28583
+ var commandQueues = new Map;
28584
+ var pollWaiters = new Map;
28585
+ var viewFieldNames = new Map;
28586
+ var viewFieldInfo = new Map;
28587
+ var viewFileWatches = new Map;
28588
+ var viewLastActivity = new Map;
28589
+ function touchView(uuid3) {
28590
+ viewLastActivity.set(uuid3, Date.now());
28591
+ }
28592
+ function pruneStaleQueues() {
28593
+ const now = Date.now();
28594
+ for (const [uuid3, lastActivity] of viewLastActivity) {
28595
+ if (now - lastActivity > COMMAND_TTL_MS) {
28596
+ viewLastActivity.delete(uuid3);
28597
+ commandQueues.delete(uuid3);
28598
+ viewFieldNames.delete(uuid3);
28599
+ viewFieldInfo.delete(uuid3);
28600
+ stopFileWatch(uuid3);
28601
+ }
28602
+ }
28603
+ }
28604
+ setInterval(pruneStaleQueues, SWEEP_INTERVAL_MS).unref();
28605
+ function enqueueCommand(viewUUID, command) {
28606
+ let entry = commandQueues.get(viewUUID);
28607
+ if (!entry) {
28608
+ entry = { commands: [], lastActivity: Date.now() };
28609
+ commandQueues.set(viewUUID, entry);
28610
+ }
28611
+ entry.commands.push(command);
28612
+ entry.lastActivity = Date.now();
28613
+ touchView(viewUUID);
28614
+ const waiter = pollWaiters.get(viewUUID);
28615
+ if (waiter) {
28616
+ pollWaiters.delete(viewUUID);
28617
+ waiter();
28618
+ }
28619
+ }
28620
+ function dequeueCommands(viewUUID) {
28621
+ touchView(viewUUID);
28622
+ const entry = commandQueues.get(viewUUID);
28623
+ if (!entry)
28624
+ return [];
28625
+ const commands = entry.commands;
28626
+ commandQueues.delete(viewUUID);
28627
+ return commands;
28628
+ }
28629
+ var FILE_WATCH_DEBOUNCE_MS = 150;
28630
+ function startFileWatch(viewUUID, filePath) {
28631
+ const resolved = path.resolve(filePath);
28632
+ let stat;
28633
+ try {
28634
+ stat = fs.statSync(resolved);
28635
+ } catch {
28636
+ return;
28637
+ }
28638
+ stopFileWatch(viewUUID);
28639
+ const entry = {
28640
+ filePath: resolved,
28641
+ watcher: null,
28642
+ lastMtimeMs: stat.mtimeMs,
28643
+ debounce: null
28644
+ };
28645
+ const onEvent = (eventType) => {
28646
+ if (entry.debounce)
28647
+ clearTimeout(entry.debounce);
28648
+ entry.debounce = setTimeout(() => {
28649
+ entry.debounce = null;
28650
+ let s2;
28651
+ try {
28652
+ s2 = fs.statSync(resolved);
28653
+ } catch {
28654
+ return;
28655
+ }
28656
+ if (s2.mtimeMs === entry.lastMtimeMs)
28657
+ return;
28658
+ entry.lastMtimeMs = s2.mtimeMs;
28659
+ enqueueCommand(viewUUID, { type: "file_changed", mtimeMs: s2.mtimeMs });
28660
+ }, FILE_WATCH_DEBOUNCE_MS);
28661
+ if (eventType === "rename") {
28662
+ try {
28663
+ entry.watcher.close();
28664
+ } catch {}
28665
+ try {
28666
+ entry.watcher = fs.watch(resolved, onEvent);
28667
+ } catch {}
28668
+ }
28669
+ };
28670
+ try {
28671
+ entry.watcher = fs.watch(resolved, onEvent);
28672
+ } catch {
28673
+ return;
28674
+ }
28675
+ viewFileWatches.set(viewUUID, entry);
28676
+ }
28677
+ function stopFileWatch(viewUUID) {
28678
+ const entry = viewFileWatches.get(viewUUID);
28679
+ if (!entry)
28680
+ return;
28681
+ if (entry.debounce)
28682
+ clearTimeout(entry.debounce);
28683
+ try {
28684
+ entry.watcher.close();
28685
+ } catch {}
28686
+ viewFileWatches.delete(viewUUID);
28687
+ }
28510
28688
  function isFileUrl(url2) {
28511
28689
  return url2.startsWith("file://") || url2.startsWith("computer://");
28512
28690
  }
@@ -28717,8 +28895,133 @@ async function refreshRoots(server) {
28717
28895
  console.error(`[pdf-server] Failed to list roots: ${err instanceof Error ? err.message : err}`);
28718
28896
  }
28719
28897
  }
28898
+ async function extractFormFieldInfo(url2, readRange) {
28899
+ const { totalBytes } = await readRange(url2, 0, 1);
28900
+ const { data } = await readRange(url2, 0, totalBytes);
28901
+ const loadingTask = getDocument({
28902
+ data,
28903
+ standardFontDataUrl: STANDARD_FONT_DATA_URL,
28904
+ StandardFontDataFactory: FetchStandardFontDataFactory,
28905
+ verbosity: VerbosityLevel.ERRORS
28906
+ });
28907
+ const pdfDoc = await loadingTask.promise;
28908
+ const fields = [];
28909
+ try {
28910
+ for (let i2 = 1;i2 <= pdfDoc.numPages; i2++) {
28911
+ const page = await pdfDoc.getPage(i2);
28912
+ const pageHeight = page.getViewport({ scale: 1 }).height;
28913
+ const annotations = await page.getAnnotations();
28914
+ for (const ann of annotations) {
28915
+ if (ann.annotationType !== 20)
28916
+ continue;
28917
+ if (!ann.rect)
28918
+ continue;
28919
+ const fieldName = ann.fieldName || "";
28920
+ const fieldType = ann.fieldType || "unknown";
28921
+ const x1 = Math.min(ann.rect[0], ann.rect[2]);
28922
+ const y1 = Math.min(ann.rect[1], ann.rect[3]);
28923
+ const x2 = Math.max(ann.rect[0], ann.rect[2]);
28924
+ const y2 = Math.max(ann.rect[1], ann.rect[3]);
28925
+ const width = x2 - x1;
28926
+ const height = y2 - y1;
28927
+ const modelY = pageHeight - y2;
28928
+ fields.push({
28929
+ name: fieldName,
28930
+ type: fieldType,
28931
+ page: i2,
28932
+ x: Math.round(x1),
28933
+ y: Math.round(modelY),
28934
+ width: Math.round(width),
28935
+ height: Math.round(height),
28936
+ ...ann.alternativeText ? { label: ann.alternativeText } : undefined
28937
+ });
28938
+ }
28939
+ }
28940
+ } finally {
28941
+ pdfDoc.destroy();
28942
+ }
28943
+ return fields;
28944
+ }
28945
+ async function extractFormSchema(url2, readRange) {
28946
+ const { totalBytes } = await readRange(url2, 0, 1);
28947
+ const { data } = await readRange(url2, 0, totalBytes);
28948
+ const loadingTask = getDocument({
28949
+ data,
28950
+ standardFontDataUrl: STANDARD_FONT_DATA_URL,
28951
+ StandardFontDataFactory: FetchStandardFontDataFactory,
28952
+ verbosity: VerbosityLevel.ERRORS
28953
+ });
28954
+ const pdfDoc = await loadingTask.promise;
28955
+ let fieldObjects;
28956
+ try {
28957
+ fieldObjects = await pdfDoc.getFieldObjects();
28958
+ } catch {
28959
+ pdfDoc.destroy();
28960
+ return null;
28961
+ }
28962
+ if (!fieldObjects || Object.keys(fieldObjects).length === 0) {
28963
+ pdfDoc.destroy();
28964
+ return null;
28965
+ }
28966
+ const properties = {};
28967
+ for (const [name, fields] of Object.entries(fieldObjects)) {
28968
+ const field = fields[0];
28969
+ if (!field.editable)
28970
+ continue;
28971
+ switch (field.type) {
28972
+ case "text":
28973
+ properties[name] = { type: "string", title: name };
28974
+ break;
28975
+ case "checkbox":
28976
+ properties[name] = { type: "boolean", title: name };
28977
+ break;
28978
+ case "radiobutton": {
28979
+ const options = fields.map((f2) => f2.exportValues).filter((v2) => !!v2 && v2 !== "Off");
28980
+ properties[name] = options.length > 0 ? { type: "string", title: name, enum: options } : { type: "string", title: name };
28981
+ break;
28982
+ }
28983
+ case "combobox":
28984
+ case "listbox": {
28985
+ const items = field.items?.map((i2) => i2.exportValue).filter(Boolean);
28986
+ properties[name] = items && items.length > 0 ? { type: "string", title: name, enum: items } : { type: "string", title: name };
28987
+ break;
28988
+ }
28989
+ }
28990
+ }
28991
+ const fieldLabels = new Map;
28992
+ try {
28993
+ for (let i2 = 1;i2 <= pdfDoc.numPages; i2++) {
28994
+ const page = await pdfDoc.getPage(i2);
28995
+ const annotations = await page.getAnnotations();
28996
+ for (const ann of annotations) {
28997
+ if (ann.fieldName && ann.alternativeText) {
28998
+ fieldLabels.set(ann.fieldName, ann.alternativeText);
28999
+ }
29000
+ }
29001
+ }
29002
+ } catch {}
29003
+ for (const [name, prop] of Object.entries(properties)) {
29004
+ const label = fieldLabels.get(name);
29005
+ if (label) {
29006
+ prop.title = label;
29007
+ }
29008
+ }
29009
+ const hasMechanicalNames = Object.keys(properties).some((name) => {
29010
+ if (fieldLabels.has(name))
29011
+ return false;
29012
+ return /[[\]().]/.test(name) || /^[A-Z0-9_]+$/.test(name);
29013
+ });
29014
+ pdfDoc.destroy();
29015
+ if (Object.keys(properties).length === 0)
29016
+ return null;
29017
+ if (hasMechanicalNames)
29018
+ return null;
29019
+ return { type: "object", properties };
29020
+ }
28720
29021
  function createServer(options = {}) {
28721
- const { useClientRoots = false } = options;
29022
+ const { enableInteract = false, useClientRoots = false } = options;
29023
+ const debug = options.debug ?? false;
29024
+ const disableInteract = !enableInteract;
28722
29025
  const server = new McpServer({ name: "PDF Server", version: "2.0.0" });
28723
29026
  if (useClientRoots) {
28724
29027
  server.server.oninitialized = () => {
@@ -28727,23 +29030,60 @@ function createServer(options = {}) {
28727
29030
  server.server.setNotificationHandler(RootsListChangedNotificationSchema, async () => {
28728
29031
  await refreshRoots(server.server);
28729
29032
  });
28730
- } else {
28731
- console.error("[pdf-server] Client roots are ignored (default for remote transports). " + "Pass --use-client-roots to allow the client to expose local directories.");
28732
29033
  }
28733
29034
  const { readPdfRange } = createPdfCache();
28734
29035
  server.tool("list_pdfs", "List available PDFs that can be displayed", {}, async () => {
28735
- const pdfs = [];
28736
- for (const filePath of allowedLocalFiles) {
28737
- pdfs.push({ url: pathToFileUrl(filePath), type: "local" });
28738
- }
29036
+ const seen = new Set;
29037
+ const localFiles = [];
29038
+ const addLocal = (filePath) => {
29039
+ const url2 = pathToFileUrl(filePath);
29040
+ if (seen.has(url2))
29041
+ return;
29042
+ seen.add(url2);
29043
+ localFiles.push(url2);
29044
+ };
29045
+ for (const filePath of allowedLocalFiles)
29046
+ addLocal(filePath);
29047
+ const WALK_MAX_DEPTH = 8;
29048
+ const WALK_MAX_FILES = 500;
29049
+ let truncated = false;
29050
+ const walk = async (dir, depth) => {
29051
+ if (depth > WALK_MAX_DEPTH || localFiles.length >= WALK_MAX_FILES) {
29052
+ truncated ||= localFiles.length >= WALK_MAX_FILES;
29053
+ return;
29054
+ }
29055
+ let entries;
29056
+ try {
29057
+ entries = await fs.promises.readdir(dir, { withFileTypes: true });
29058
+ } catch {
29059
+ return;
29060
+ }
29061
+ for (const e2 of entries) {
29062
+ if (localFiles.length >= WALK_MAX_FILES) {
29063
+ truncated = true;
29064
+ return;
29065
+ }
29066
+ if (e2.name.startsWith(".") || e2.name === "node_modules")
29067
+ continue;
29068
+ const full = path.join(dir, e2.name);
29069
+ if (e2.isDirectory()) {
29070
+ await walk(full, depth + 1);
29071
+ } else if (e2.isFile() && /\.pdf$/i.test(e2.name)) {
29072
+ addLocal(full);
29073
+ }
29074
+ }
29075
+ };
29076
+ for (const dir of allowedLocalDirs)
29077
+ await walk(dir, 0);
28739
29078
  const parts = [];
28740
- if (pdfs.length > 0) {
28741
- parts.push(`Available PDFs:
28742
- ${pdfs.map((p) => `- ${p.url} (${p.type})`).join(`
29079
+ if (localFiles.length > 0) {
29080
+ const header = truncated ? `Available PDFs (showing first ${WALK_MAX_FILES}):` : `Available PDFs:`;
29081
+ parts.push(`${header}
29082
+ ${localFiles.map((u2) => `- ${u2}`).join(`
28743
29083
  `)}`);
28744
29084
  }
28745
29085
  if (allowedLocalDirs.size > 0) {
28746
- parts.push(`Allowed local directories (from client roots):
29086
+ parts.push(`Allowed local directories:
28747
29087
  ${[...allowedLocalDirs].map((d2) => `- ${d2}`).join(`
28748
29088
  `)}
28749
29089
  Any PDF file under these directories can be displayed.`);
@@ -28754,12 +29094,13 @@ Any PDF file under these directories can be displayed.`);
28754
29094
 
28755
29095
  `) }],
28756
29096
  structuredContent: {
28757
- localFiles: pdfs.filter((p) => p.type === "local").map((p) => p.url),
28758
- allowedDirectories: [...allowedLocalDirs]
29097
+ localFiles,
29098
+ allowedDirectories: [...allowedLocalDirs],
29099
+ truncated
28759
29100
  }
28760
29101
  };
28761
29102
  });
28762
- uZ(server, "read_pdf_bytes", {
29103
+ hZ(server, "read_pdf_bytes", {
28763
29104
  title: "Read PDF Bytes",
28764
29105
  description: "Read a range of bytes from a PDF (max 512KB per request)",
28765
29106
  inputSchema: {
@@ -28817,25 +29158,50 @@ Any PDF file under these directories can be displayed.`);
28817
29158
  };
28818
29159
  }
28819
29160
  });
28820
- uZ(server, "display_pdf", {
29161
+ hZ(server, "display_pdf", {
28821
29162
  title: "Display PDF",
28822
- description: `Display an interactive PDF viewer.
29163
+ description: disableInteract ? `Show and render a PDF in a read-only viewer.
28823
29164
 
28824
- Accepts:
28825
- - Local files explicitly added to the server (use list_pdfs to see available files)
28826
- - Local files under directories provided by the client as MCP roots
28827
- - Any remote PDF accessible via HTTPS`,
29165
+ Use this tool when the user wants to view or read a PDF. The renderer displays the document for viewing.
29166
+
29167
+ Accepts local files (use list_pdfs), client MCP root directories, or any HTTPS URL.` : `Open a PDF in an interactive viewer. Call this ONCE per PDF.
29168
+
29169
+ **All follow-up actions go through the \`interact\` tool** with the returned viewUUID — annotating, signing, stamping, filling forms, navigating, searching, extracting text/screenshots. Calling display_pdf again creates a SEPARATE viewer with a different viewUUID — interact calls using the new UUID will not reach the viewer the user already sees.
29170
+
29171
+ Returns a viewUUID in structuredContent. Pass it to \`interact\`:
29172
+ - add_annotations, update_annotations, remove_annotations, highlight_text
29173
+ - fill_form (fill PDF form fields)
29174
+ - navigate, search, find, search_navigate, zoom
29175
+ - get_text, get_screenshot (extract content)
29176
+
29177
+ Accepts local files (use list_pdfs), client MCP root directories, or any HTTPS URL.
29178
+ Set \`elicit_form_inputs\` to true to prompt the user to fill form fields before display.`,
28828
29179
  inputSchema: {
28829
29180
  url: exports_external.string().default(DEFAULT_PDF).describe("PDF URL or local file path"),
28830
- page: exports_external.number().min(1).default(1).describe("Initial page")
29181
+ page: exports_external.number().min(1).default(1).describe("Initial page"),
29182
+ ...disableInteract ? {} : {
29183
+ elicit_form_inputs: exports_external.boolean().default(false).describe("If true and the PDF has form fields, prompt the user to fill them before displaying")
29184
+ }
28831
29185
  },
28832
29186
  outputSchema: exports_external.object({
29187
+ viewUUID: exports_external.string().describe("UUID for this viewer instance" + (disableInteract ? "" : " — pass to interact tool")),
28833
29188
  url: exports_external.string(),
28834
29189
  initialPage: exports_external.number(),
28835
- totalBytes: exports_external.number()
29190
+ totalBytes: exports_external.number(),
29191
+ formFieldValues: exports_external.record(exports_external.string(), exports_external.union([exports_external.string(), exports_external.boolean()])).optional().describe("Form field values filled by the user via elicitation"),
29192
+ formFields: exports_external.array(exports_external.object({
29193
+ name: exports_external.string(),
29194
+ type: exports_external.string(),
29195
+ page: exports_external.number(),
29196
+ label: exports_external.string().optional(),
29197
+ x: exports_external.number(),
29198
+ y: exports_external.number(),
29199
+ width: exports_external.number(),
29200
+ height: exports_external.number()
29201
+ })).optional().describe("Form fields with bounding boxes in model coordinates (top-left origin)")
28836
29202
  }),
28837
29203
  _meta: { ui: { resourceUri: RESOURCE_URI } }
28838
- }, async ({ url: url2, page }) => {
29204
+ }, async ({ url: url2, page, elicit_form_inputs }) => {
28839
29205
  const normalized = isArxivUrl(url2) ? normalizeArxivUrl(url2) : url2;
28840
29206
  const validation = validateUrl(normalized);
28841
29207
  if (!validation.valid) {
@@ -28845,40 +29211,834 @@ Accepts:
28845
29211
  };
28846
29212
  }
28847
29213
  const { totalBytes } = await readPdfRange(normalized, 0, 1);
29214
+ const uuid3 = randomUUID();
29215
+ if (!disableInteract)
29216
+ touchView(uuid3);
29217
+ let writable = false;
29218
+ let debugResolved;
29219
+ if (isFileUrl(normalized) || isLocalPath(normalized)) {
29220
+ const localPath = isFileUrl(normalized) ? fileUrlToPath(normalized) : decodeURIComponent(normalized);
29221
+ const resolved = path.resolve(localPath);
29222
+ debugResolved = resolved;
29223
+ if (isWritablePath(resolved)) {
29224
+ try {
29225
+ await fs.promises.access(resolved, fs.constants.W_OK);
29226
+ writable = true;
29227
+ } catch {}
29228
+ }
29229
+ if (!disableInteract) {
29230
+ startFileWatch(uuid3, localPath);
29231
+ }
29232
+ }
29233
+ let formSchema = null;
29234
+ try {
29235
+ formSchema = await extractFormSchema(normalized, readPdfRange);
29236
+ } catch {}
29237
+ if (formSchema) {
29238
+ viewFieldNames.set(uuid3, new Set(Object.keys(formSchema.properties)));
29239
+ }
29240
+ let fieldInfo = [];
29241
+ try {
29242
+ fieldInfo = await extractFormFieldInfo(normalized, readPdfRange);
29243
+ if (fieldInfo.length > 0) {
29244
+ viewFieldInfo.set(uuid3, fieldInfo);
29245
+ if (!viewFieldNames.has(uuid3)) {
29246
+ viewFieldNames.set(uuid3, new Set(fieldInfo.map((f2) => f2.name).filter(Boolean)));
29247
+ }
29248
+ }
29249
+ } catch {}
29250
+ let formFieldValues;
29251
+ let elicitResult;
29252
+ if (elicit_form_inputs && formSchema) {
29253
+ const clientCaps = server.server.getClientCapabilities();
29254
+ if (clientCaps?.elicitation?.form) {
29255
+ try {
29256
+ elicitResult = await server.server.elicitInput({
29257
+ message: `Please fill in the PDF form fields for "${normalized.split("/").pop() || normalized}":`,
29258
+ requestedSchema: formSchema
29259
+ });
29260
+ if (elicitResult.action === "accept" && elicitResult.content) {
29261
+ formFieldValues = {};
29262
+ for (const [k2, v2] of Object.entries(elicitResult.content)) {
29263
+ if (typeof v2 === "string" || typeof v2 === "boolean") {
29264
+ formFieldValues[k2] = v2;
29265
+ }
29266
+ }
29267
+ enqueueCommand(uuid3, {
29268
+ type: "fill_form",
29269
+ fields: Object.entries(formFieldValues).map(([name, value]) => ({ name, value }))
29270
+ });
29271
+ }
29272
+ } catch (err) {
29273
+ console.error("[pdf-server] Form elicitation failed:", err);
29274
+ }
29275
+ }
29276
+ }
29277
+ const contentParts = [
29278
+ {
29279
+ type: "text",
29280
+ text: disableInteract ? `Displaying PDF: ${normalized}` : `PDF opened. viewUUID: ${uuid3}
29281
+
29282
+ → To annotate, sign, stamp, fill forms, navigate, or extract: call \`interact\` with this viewUUID.
29283
+ → DO NOT call display_pdf again — that spawns a separate viewer with a different viewUUID; your interact calls would target the new empty one, not the one the user is looking at.
29284
+
29285
+ URL: ${normalized}`
29286
+ }
29287
+ ];
29288
+ if (formFieldValues && Object.keys(formFieldValues).length > 0) {
29289
+ const fieldSummary = Object.entries(formFieldValues).map(([name, value]) => ` ${name}: ${typeof value === "boolean" ? value ? "checked" : "unchecked" : value}`).join(`
29290
+ `);
29291
+ contentParts.push({
29292
+ type: "text",
29293
+ text: `
29294
+ User-provided form field values:
29295
+ ${fieldSummary}`
29296
+ });
29297
+ } else if (elicit_form_inputs && elicitResult && elicitResult.action !== "accept") {
29298
+ contentParts.push({
29299
+ type: "text",
29300
+ text: `
29301
+ Form elicitation was ${elicitResult.action}d by the user.`
29302
+ });
29303
+ }
29304
+ if (fieldInfo.length > 0) {
29305
+ const byPage = new Map;
29306
+ for (const f2 of fieldInfo) {
29307
+ let list = byPage.get(f2.page);
29308
+ if (!list) {
29309
+ list = [];
29310
+ byPage.set(f2.page, list);
29311
+ }
29312
+ list.push(f2);
29313
+ }
29314
+ const lines = [
29315
+ `
29316
+ Form fields (${fieldInfo.length})${disableInteract ? "" : " — use fill_form with {name, value}"}:`
29317
+ ];
29318
+ for (const [pg, fields] of [...byPage.entries()].sort((a2, b) => a2[0] - b[0])) {
29319
+ lines.push(` Page ${pg}:`);
29320
+ for (const f2 of fields) {
29321
+ const label = f2.label ? ` "${f2.label}"` : "";
29322
+ const nameStr = f2.name || "(unnamed)";
29323
+ lines.push(` ${nameStr}${label} [${f2.type}] at (${f2.x},${f2.y}) ${f2.width}×${f2.height}`);
29324
+ }
29325
+ }
29326
+ contentParts.push({ type: "text", text: lines.join(`
29327
+ `) });
29328
+ } else {
29329
+ const fieldNames = viewFieldNames.get(uuid3);
29330
+ if (fieldNames && fieldNames.size > 0) {
29331
+ contentParts.push({
29332
+ type: "text",
29333
+ text: `
29334
+ Form fields${disableInteract ? "" : " available for fill_form"}: ${[...fieldNames].join(", ")}`
29335
+ });
29336
+ }
29337
+ }
28848
29338
  return {
28849
- content: [{ type: "text", text: `Displaying PDF: ${normalized}` }],
29339
+ content: contentParts,
28850
29340
  structuredContent: {
29341
+ viewUUID: uuid3,
28851
29342
  url: normalized,
28852
29343
  initialPage: page,
28853
- totalBytes
29344
+ totalBytes,
29345
+ ...formFieldValues ? { formFieldValues } : {},
29346
+ ...fieldInfo.length > 0 ? { formFields: fieldInfo } : {}
28854
29347
  },
28855
29348
  _meta: {
28856
- viewUUID: randomUUID()
29349
+ viewUUID: uuid3,
29350
+ interactEnabled: !disableInteract,
29351
+ writable,
29352
+ ...debug ? {
29353
+ _debug: {
29354
+ resolved: debugResolved,
29355
+ writable,
29356
+ isWritablePath: debugResolved ? isWritablePath(debugResolved) : undefined,
29357
+ cliLocalFiles: [...cliLocalFiles],
29358
+ allowedLocalFiles: [...allowedLocalFiles],
29359
+ allowedLocalDirs: [...allowedLocalDirs]
29360
+ }
29361
+ } : {}
28857
29362
  }
28858
29363
  };
28859
29364
  });
28860
- fZ(server, RESOURCE_URI, RESOURCE_URI, { mimeType: d }, async () => {
29365
+ if (!disableInteract) {
29366
+ let detectImageDimensions = function(bytes) {
29367
+ if (bytes[0] === 137 && bytes[1] === 80 && bytes[2] === 78 && bytes[3] === 71) {
29368
+ if (bytes.length >= 24) {
29369
+ const width = bytes.readUInt32BE(16);
29370
+ const height = bytes.readUInt32BE(20);
29371
+ return { width, height };
29372
+ }
29373
+ }
29374
+ if (bytes[0] === 255 && bytes[1] === 216) {
29375
+ let offset = 2;
29376
+ while (offset < bytes.length - 8) {
29377
+ if (bytes[offset] !== 255)
29378
+ break;
29379
+ const marker = bytes[offset + 1];
29380
+ if (marker === 192 || marker === 194) {
29381
+ const height = bytes.readUInt16BE(offset + 5);
29382
+ const width = bytes.readUInt16BE(offset + 7);
29383
+ return { width, height };
29384
+ }
29385
+ const segLen = bytes.readUInt16BE(offset + 2);
29386
+ offset += 2 + segLen;
29387
+ }
29388
+ }
29389
+ return null;
29390
+ };
29391
+ const InteractCommandSchema = exports_external.object({
29392
+ action: exports_external.enum([
29393
+ "navigate",
29394
+ "search",
29395
+ "find",
29396
+ "search_navigate",
29397
+ "zoom",
29398
+ "add_annotations",
29399
+ "update_annotations",
29400
+ "remove_annotations",
29401
+ "highlight_text",
29402
+ "fill_form",
29403
+ "get_text",
29404
+ "get_screenshot"
29405
+ ]).describe("Action to perform"),
29406
+ page: exports_external.number().min(1).optional().describe("Page number (for navigate, highlight_text, get_screenshot, get_text)"),
29407
+ query: exports_external.string().optional().describe("Search text (for search / find / highlight_text)"),
29408
+ matchIndex: exports_external.number().min(0).optional().describe("Match index (for search_navigate)"),
29409
+ scale: exports_external.number().min(0.5).max(3).optional().describe("Zoom scale, 1.0 = 100% (for zoom)"),
29410
+ annotations: exports_external.array(exports_external.record(exports_external.string(), exports_external.any())).optional().describe("Annotation objects (see types in description). Each needs: id, type, page. For update_annotations only id+type are required."),
29411
+ ids: exports_external.array(exports_external.string()).optional().describe("Annotation IDs (for remove_annotations)"),
29412
+ color: exports_external.string().optional().describe("Color override (for highlight_text)"),
29413
+ content: exports_external.string().optional().describe("Tooltip/note content (for highlight_text)"),
29414
+ fields: exports_external.array(FormField).optional().describe("Form fields to fill (for fill_form): { name, value } where value is string or boolean"),
29415
+ intervals: exports_external.array(PageInterval).optional().describe("Page ranges for get_text. Each has optional start/end. [{start:1,end:5}], [{}] = all pages. Max 20 pages.")
29416
+ });
29417
+ async function resolveImageAnnotation(ann) {
29418
+ if (!ann.imageData && ann.imageUrl) {
29419
+ const url2 = String(ann.imageUrl);
29420
+ const check2 = validateUrl(url2);
29421
+ if (!check2.valid) {
29422
+ throw new Error(`imageUrl rejected by validateUrl: ${check2.error ?? url2}`);
29423
+ }
29424
+ let imgBytes;
29425
+ if (url2.startsWith("https://")) {
29426
+ const resp = await fetch(url2);
29427
+ if (!resp.ok)
29428
+ throw new Error(`HTTP ${resp.status} for ${url2}`);
29429
+ imgBytes = new Uint8Array(await resp.arrayBuffer());
29430
+ } else {
29431
+ const filePath = isFileUrl(url2) ? fileUrlToPath(url2) : decodeURIComponent(url2);
29432
+ imgBytes = await fs.promises.readFile(path.resolve(filePath));
29433
+ }
29434
+ ann.imageData = Buffer.from(imgBytes).toString("base64");
29435
+ }
29436
+ if (ann.imageData && !ann.mimeType) {
29437
+ const bytes = Buffer.from(ann.imageData, "base64");
29438
+ if (bytes[0] === 137 && bytes[1] === 80 && bytes[2] === 78 && bytes[3] === 71) {
29439
+ ann.mimeType = "image/png";
29440
+ } else {
29441
+ ann.mimeType = "image/jpeg";
29442
+ }
29443
+ }
29444
+ if (ann.imageData && (ann.width == null || ann.height == null)) {
29445
+ const dims = detectImageDimensions(Buffer.from(ann.imageData, "base64"));
29446
+ if (dims) {
29447
+ const maxWidth = 200;
29448
+ const aspectRatio = dims.height / dims.width;
29449
+ ann.width = ann.width ?? Math.min(dims.width, maxWidth);
29450
+ ann.height = ann.height ?? ann.width * aspectRatio;
29451
+ } else {
29452
+ ann.width = ann.width ?? 200;
29453
+ ann.height = ann.height ?? 200;
29454
+ }
29455
+ }
29456
+ ann.x = ann.x ?? 72;
29457
+ ann.y = ann.y ?? 72;
29458
+ }
29459
+ async function processInteractCommand(uuid3, cmd, signal) {
29460
+ const {
29461
+ action,
29462
+ page,
29463
+ query,
29464
+ matchIndex,
29465
+ scale,
29466
+ annotations,
29467
+ ids,
29468
+ color,
29469
+ content,
29470
+ fields,
29471
+ intervals
29472
+ } = cmd;
29473
+ let description;
29474
+ switch (action) {
29475
+ case "navigate":
29476
+ if (page == null)
29477
+ return {
29478
+ content: [{ type: "text", text: "navigate requires `page`" }],
29479
+ isError: true
29480
+ };
29481
+ enqueueCommand(uuid3, { type: "navigate", page });
29482
+ description = `navigate to page ${page}`;
29483
+ break;
29484
+ case "search":
29485
+ if (!query)
29486
+ return {
29487
+ content: [{ type: "text", text: "search requires `query`" }],
29488
+ isError: true
29489
+ };
29490
+ enqueueCommand(uuid3, { type: "search", query });
29491
+ description = `search for "${query}"`;
29492
+ break;
29493
+ case "find":
29494
+ if (!query)
29495
+ return {
29496
+ content: [{ type: "text", text: "find requires `query`" }],
29497
+ isError: true
29498
+ };
29499
+ enqueueCommand(uuid3, { type: "find", query });
29500
+ description = `find "${query}" (silent)`;
29501
+ break;
29502
+ case "search_navigate":
29503
+ if (matchIndex == null)
29504
+ return {
29505
+ content: [
29506
+ {
29507
+ type: "text",
29508
+ text: "search_navigate requires `matchIndex`"
29509
+ }
29510
+ ],
29511
+ isError: true
29512
+ };
29513
+ enqueueCommand(uuid3, { type: "search_navigate", matchIndex });
29514
+ description = `go to match #${matchIndex}`;
29515
+ break;
29516
+ case "zoom":
29517
+ if (scale == null)
29518
+ return {
29519
+ content: [{ type: "text", text: "zoom requires `scale`" }],
29520
+ isError: true
29521
+ };
29522
+ enqueueCommand(uuid3, { type: "zoom", scale });
29523
+ description = `zoom to ${Math.round(scale * 100)}%`;
29524
+ break;
29525
+ case "add_annotations":
29526
+ if (!annotations || annotations.length === 0)
29527
+ return {
29528
+ content: [
29529
+ {
29530
+ type: "text",
29531
+ text: "add_annotations requires `annotations` array"
29532
+ }
29533
+ ],
29534
+ isError: true
29535
+ };
29536
+ try {
29537
+ for (const ann of annotations) {
29538
+ if (ann.type === "image") {
29539
+ await resolveImageAnnotation(ann);
29540
+ }
29541
+ }
29542
+ } catch (err) {
29543
+ return {
29544
+ content: [
29545
+ {
29546
+ type: "text",
29547
+ text: `add_annotations: ${err instanceof Error ? err.message : String(err)}`
29548
+ }
29549
+ ],
29550
+ isError: true
29551
+ };
29552
+ }
29553
+ enqueueCommand(uuid3, {
29554
+ type: "add_annotations",
29555
+ annotations
29556
+ });
29557
+ description = `add ${annotations.length} annotation(s)`;
29558
+ break;
29559
+ case "update_annotations":
29560
+ if (!annotations || annotations.length === 0)
29561
+ return {
29562
+ content: [
29563
+ {
29564
+ type: "text",
29565
+ text: "update_annotations requires `annotations` array"
29566
+ }
29567
+ ],
29568
+ isError: true
29569
+ };
29570
+ enqueueCommand(uuid3, {
29571
+ type: "update_annotations",
29572
+ annotations
29573
+ });
29574
+ description = `update ${annotations.length} annotation(s)`;
29575
+ break;
29576
+ case "remove_annotations":
29577
+ if (!ids || ids.length === 0)
29578
+ return {
29579
+ content: [
29580
+ {
29581
+ type: "text",
29582
+ text: "remove_annotations requires `ids` array"
29583
+ }
29584
+ ],
29585
+ isError: true
29586
+ };
29587
+ enqueueCommand(uuid3, { type: "remove_annotations", ids });
29588
+ description = `remove ${ids.length} annotation(s)`;
29589
+ break;
29590
+ case "highlight_text": {
29591
+ if (!query)
29592
+ return {
29593
+ content: [
29594
+ { type: "text", text: "highlight_text requires `query`" }
29595
+ ],
29596
+ isError: true
29597
+ };
29598
+ const id = `ht_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`;
29599
+ enqueueCommand(uuid3, {
29600
+ type: "highlight_text",
29601
+ id,
29602
+ query,
29603
+ page,
29604
+ color,
29605
+ content
29606
+ });
29607
+ description = `highlight text "${query}"${page ? ` on page ${page}` : ""}`;
29608
+ break;
29609
+ }
29610
+ case "fill_form": {
29611
+ if (!fields || fields.length === 0)
29612
+ return {
29613
+ content: [
29614
+ { type: "text", text: "fill_form requires `fields` array" }
29615
+ ],
29616
+ isError: true
29617
+ };
29618
+ const knownFields = viewFieldNames.get(uuid3);
29619
+ const validFields = [];
29620
+ const unknownNames = [];
29621
+ for (const f2 of fields) {
29622
+ if (knownFields && !knownFields.has(f2.name)) {
29623
+ unknownNames.push(f2.name);
29624
+ } else {
29625
+ validFields.push(f2);
29626
+ }
29627
+ }
29628
+ if (validFields.length > 0) {
29629
+ enqueueCommand(uuid3, { type: "fill_form", fields: validFields });
29630
+ }
29631
+ const parts = [];
29632
+ if (validFields.length > 0) {
29633
+ parts.push(`Filled ${validFields.length} field(s): ${validFields.map((f2) => f2.name).join(", ")}`);
29634
+ }
29635
+ if (unknownNames.length > 0) {
29636
+ parts.push(`Unknown field(s) skipped: ${unknownNames.join(", ")}`);
29637
+ if (knownFields && knownFields.size > 0) {
29638
+ parts.push(`Valid field names: ${[...knownFields].join(", ")}`);
29639
+ }
29640
+ }
29641
+ description = parts.join(". ");
29642
+ if (unknownNames.length > 0 && validFields.length === 0) {
29643
+ return {
29644
+ content: [{ type: "text", text: description }],
29645
+ isError: true
29646
+ };
29647
+ }
29648
+ break;
29649
+ }
29650
+ case "get_text": {
29651
+ const resolvedIntervals = intervals ?? (page ? [{ start: page, end: page }] : [{}]);
29652
+ const requestId = randomUUID();
29653
+ enqueueCommand(uuid3, {
29654
+ type: "get_pages",
29655
+ requestId,
29656
+ intervals: resolvedIntervals,
29657
+ getText: true,
29658
+ getScreenshots: false
29659
+ });
29660
+ let pageData;
29661
+ try {
29662
+ pageData = await waitForPageData(requestId, signal);
29663
+ } catch (err) {
29664
+ return {
29665
+ content: [
29666
+ {
29667
+ type: "text",
29668
+ text: `Error: ${err instanceof Error ? err.message : String(err)}`
29669
+ }
29670
+ ],
29671
+ isError: true
29672
+ };
29673
+ }
29674
+ const textParts = [];
29675
+ for (const entry of pageData) {
29676
+ if (entry.text != null) {
29677
+ textParts.push({
29678
+ type: "text",
29679
+ text: `--- Page ${entry.page} ---
29680
+ ${entry.text}`
29681
+ });
29682
+ }
29683
+ }
29684
+ if (textParts.length === 0) {
29685
+ textParts.push({ type: "text", text: "No text content returned" });
29686
+ }
29687
+ return { content: textParts };
29688
+ }
29689
+ case "get_screenshot": {
29690
+ if (page == null)
29691
+ return {
29692
+ content: [
29693
+ { type: "text", text: "get_screenshot requires `page`" }
29694
+ ],
29695
+ isError: true
29696
+ };
29697
+ const requestId = randomUUID();
29698
+ enqueueCommand(uuid3, {
29699
+ type: "get_pages",
29700
+ requestId,
29701
+ intervals: [{ start: page, end: page }],
29702
+ getText: false,
29703
+ getScreenshots: true
29704
+ });
29705
+ let pageData;
29706
+ try {
29707
+ pageData = await waitForPageData(requestId, signal);
29708
+ } catch (err) {
29709
+ return {
29710
+ content: [
29711
+ {
29712
+ type: "text",
29713
+ text: `Error: ${err instanceof Error ? err.message : String(err)}`
29714
+ }
29715
+ ],
29716
+ isError: true
29717
+ };
29718
+ }
29719
+ const entry = pageData[0];
29720
+ if (entry?.image) {
29721
+ return {
29722
+ content: [
29723
+ {
29724
+ type: "image",
29725
+ data: entry.image,
29726
+ mimeType: "image/jpeg"
29727
+ }
29728
+ ]
29729
+ };
29730
+ }
29731
+ return {
29732
+ content: [{ type: "text", text: "No screenshot returned" }],
29733
+ isError: true
29734
+ };
29735
+ }
29736
+ default:
29737
+ return {
29738
+ content: [{ type: "text", text: `Unknown action: ${action}` }],
29739
+ isError: true
29740
+ };
29741
+ }
29742
+ return {
29743
+ content: [{ type: "text", text: `Queued: ${description}` }]
29744
+ };
29745
+ }
29746
+ server.registerTool("interact", {
29747
+ title: "Interact with PDF",
29748
+ description: `Interact with a PDF viewer: annotate, navigate, search, extract text/screenshots, fill forms.
29749
+ IMPORTANT: viewUUID must be the exact UUID returned by display_pdf (e.g. "a1b2c3d4-..."). Do NOT use arbitrary strings.
29750
+
29751
+ **BATCHING**: Send multiple commands in one call via \`commands\` array. Commands run sequentially. TIP: End with \`get_screenshot\` to verify your changes.
29752
+
29753
+ **ANNOTATION** — add_annotations with array of annotation objects. Each needs: id (unique string), type, page (1-indexed).
29754
+
29755
+ **COORDINATE SYSTEM**: PDF points (1pt = 1/72in), origin at page TOP-LEFT corner. X increases rightward, Y increases downward.
29756
+ - US Letter = 612×792pt. Margins: top≈y=50, bottom≈y=742, left≈x=72, right≈x=540, center≈(306, 396).
29757
+ - Rectangle/circle/stamp x,y is the TOP-LEFT corner. To place a 200×30 box at the TOP of the page: x=72, y=50, width=200, height=30.
29758
+ - For highlights/underlines, each rect's y is the TOP of the highlighted region.
29759
+
29760
+ Annotation types:
29761
+ • highlight: rects:[{x,y,width,height}], color?, content? • underline: rects:[{x,y,w,h}], color?
29762
+ • strikethrough: rects:[{x,y,w,h}], color? • note: x, y, content, color?
29763
+ • rectangle: x, y, width, height, color?, fillColor?, rotation? • circle: x, y, width, height, color?, fillColor?
29764
+ • line: x1, y1, x2, y2, color? • freetext: x, y, content, fontSize?, color?
29765
+ • stamp: x, y, label (any text, e.g. APPROVED, DRAFT, CONFIDENTIAL), color?, rotation?
29766
+ • image: imageUrl (required), x?, y?, width?, height?, mimeType?, rotation?, aspect? — places an image (signature, logo, etc.) on the page. Pass a local file path or HTTPS URL (NO data: URIs, NO base64). Width/height auto-detected if omitted. Users can also drag & drop images directly onto the viewer.
29767
+
29768
+ TIP: For text annotations, prefer highlight_text (auto-finds text) over manual rects.
29769
+
29770
+ Example — add a signature image and a stamp, then screenshot to verify:
29771
+ \`\`\`json
29772
+ {"viewUUID":"…","commands":[
29773
+ {"action":"add_annotations","annotations":[
29774
+ {"id":"sig1","type":"image","page":1,"x":72,"y":700,"imageUrl":"/path/to/signature.png"},
29775
+ {"id":"s1","type":"stamp","page":1,"x":300,"y":400,"label":"APPROVED"}
29776
+ ]},
29777
+ {"action":"get_screenshot","page":1}
29778
+ ]}
29779
+ \`\`\`
29780
+
29781
+ • highlight_text: auto-find and highlight text (query, page?, color?, content?)
29782
+ • update_annotations: partial update (id+type required) • remove_annotations: remove by ids
29783
+
29784
+ **NAVIGATION**: navigate (page), search (query), find (query, silent), search_navigate (matchIndex), zoom (scale 0.5–3.0)
29785
+
29786
+ **TEXT/SCREENSHOTS**:
29787
+ • get_text: extract text from pages. Optional \`page\` for single page, or \`intervals\` for ranges [{start?,end?}]. Max 20 pages.
29788
+ • get_screenshot: capture a single page as PNG image. Requires \`page\`.
29789
+
29790
+ **FORMS** — fill_form: fill fields with \`fields\` array of {name, value}.`,
29791
+ inputSchema: {
29792
+ viewUUID: exports_external.string().describe("The viewUUID of the PDF viewer (from display_pdf result)"),
29793
+ action: exports_external.enum([
29794
+ "navigate",
29795
+ "search",
29796
+ "find",
29797
+ "search_navigate",
29798
+ "zoom",
29799
+ "add_annotations",
29800
+ "update_annotations",
29801
+ "remove_annotations",
29802
+ "highlight_text",
29803
+ "fill_form",
29804
+ "get_text",
29805
+ "get_screenshot"
29806
+ ]).optional().describe("Action to perform (for single command). Use `commands` array for batching."),
29807
+ page: exports_external.number().min(1).optional().describe("Page number (for navigate, highlight_text, get_screenshot, get_text)"),
29808
+ query: exports_external.string().optional().describe("Search text (for search / find / highlight_text)"),
29809
+ matchIndex: exports_external.number().min(0).optional().describe("Match index (for search_navigate)"),
29810
+ scale: exports_external.number().min(0.5).max(3).optional().describe("Zoom scale, 1.0 = 100% (for zoom)"),
29811
+ annotations: exports_external.array(exports_external.record(exports_external.string(), exports_external.any())).optional().describe("Annotation objects (see types in description). Each needs: id, type, page. For update_annotations only id+type are required."),
29812
+ ids: exports_external.array(exports_external.string()).optional().describe("Annotation IDs (for remove_annotations)"),
29813
+ color: exports_external.string().optional().describe("Color override (for highlight_text)"),
29814
+ content: exports_external.string().optional().describe("Tooltip/note content (for highlight_text)"),
29815
+ fields: exports_external.array(FormField).optional().describe("Form fields to fill (for fill_form): { name, value } where value is string or boolean"),
29816
+ intervals: exports_external.array(PageInterval).optional().describe("Page ranges for get_text. Each has optional start/end. [{start:1,end:5}], [{}] = all pages. Max 20 pages."),
29817
+ commands: exports_external.array(InteractCommandSchema).optional().describe("Array of commands to execute sequentially. More efficient than separate calls. Tip: end with get_pages+getScreenshots to verify changes.")
29818
+ }
29819
+ }, async ({
29820
+ viewUUID: uuid3,
29821
+ action,
29822
+ page,
29823
+ query,
29824
+ matchIndex,
29825
+ scale,
29826
+ annotations,
29827
+ ids,
29828
+ color,
29829
+ content,
29830
+ fields,
29831
+ intervals,
29832
+ commands
29833
+ }, extra) => {
29834
+ const commandList = commands ? commands : action ? [
29835
+ {
29836
+ action,
29837
+ page,
29838
+ query,
29839
+ matchIndex,
29840
+ scale,
29841
+ annotations,
29842
+ ids,
29843
+ color,
29844
+ content,
29845
+ fields,
29846
+ intervals
29847
+ }
29848
+ ] : [];
29849
+ if (commandList.length === 0) {
29850
+ return {
29851
+ content: [
29852
+ {
29853
+ type: "text",
29854
+ text: "No action or commands specified. Provide either `action` (single command) or `commands` (batch)."
29855
+ }
29856
+ ],
29857
+ isError: true
29858
+ };
29859
+ }
29860
+ const allContent = [];
29861
+ let hasError = false;
29862
+ for (let i2 = 0;i2 < commandList.length; i2++) {
29863
+ const result = await processInteractCommand(uuid3, commandList[i2], extra.signal);
29864
+ if (result.isError) {
29865
+ hasError = true;
29866
+ }
29867
+ allContent.push(...result.content);
29868
+ if (hasError)
29869
+ break;
29870
+ }
29871
+ return {
29872
+ content: allContent,
29873
+ ...hasError ? { isError: true } : {}
29874
+ };
29875
+ });
29876
+ hZ(server, "submit_page_data", {
29877
+ title: "Submit Page Data",
29878
+ description: "Submit rendered page data for a get_pages request (used by viewer)",
29879
+ inputSchema: {
29880
+ requestId: exports_external.string().describe("The request ID from the get_pages command"),
29881
+ pages: exports_external.array(exports_external.object({
29882
+ page: exports_external.number(),
29883
+ text: exports_external.string().optional(),
29884
+ image: exports_external.string().optional().describe("Base64 PNG image data")
29885
+ })).describe("Page data entries")
29886
+ },
29887
+ _meta: { ui: { visibility: ["app"] } }
29888
+ }, async ({ requestId, pages }) => {
29889
+ const settle = pendingPageRequests.get(requestId);
29890
+ if (settle) {
29891
+ settle(pages);
29892
+ return {
29893
+ content: [
29894
+ { type: "text", text: `Submitted ${pages.length} page(s)` }
29895
+ ]
29896
+ };
29897
+ }
29898
+ return {
29899
+ content: [
29900
+ { type: "text", text: `No pending request for ${requestId}` }
29901
+ ],
29902
+ isError: true
29903
+ };
29904
+ });
29905
+ hZ(server, "poll_pdf_commands", {
29906
+ title: "Poll PDF Commands",
29907
+ description: "Poll for pending commands for a PDF viewer",
29908
+ inputSchema: {
29909
+ viewUUID: exports_external.string().describe("The viewUUID of the PDF viewer")
29910
+ },
29911
+ _meta: { ui: { visibility: ["app"] } }
29912
+ }, async ({ viewUUID: uuid3 }) => {
29913
+ if (commandQueues.has(uuid3)) {
29914
+ await new Promise((r) => setTimeout(r, POLL_BATCH_WAIT_MS));
29915
+ } else {
29916
+ await new Promise((resolve) => {
29917
+ const timer = setTimeout(() => {
29918
+ pollWaiters.delete(uuid3);
29919
+ resolve();
29920
+ }, LONG_POLL_TIMEOUT_MS);
29921
+ const prev = pollWaiters.get(uuid3);
29922
+ if (prev)
29923
+ prev();
29924
+ pollWaiters.set(uuid3, () => {
29925
+ clearTimeout(timer);
29926
+ resolve();
29927
+ });
29928
+ });
29929
+ if (commandQueues.has(uuid3)) {
29930
+ await new Promise((r) => setTimeout(r, POLL_BATCH_WAIT_MS));
29931
+ }
29932
+ }
29933
+ const commands = dequeueCommands(uuid3);
29934
+ return {
29935
+ content: [{ type: "text", text: `${commands.length} command(s)` }],
29936
+ structuredContent: { commands }
29937
+ };
29938
+ });
29939
+ }
29940
+ hZ(server, "save_pdf", {
29941
+ title: "Save PDF",
29942
+ description: "Save annotated PDF bytes back to a local file",
29943
+ inputSchema: {
29944
+ url: exports_external.string().describe("Original PDF URL or local file path"),
29945
+ data: exports_external.string().describe("Base64-encoded PDF bytes")
29946
+ },
29947
+ outputSchema: exports_external.object({
29948
+ filePath: exports_external.string(),
29949
+ mtimeMs: exports_external.number()
29950
+ }),
29951
+ _meta: { ui: { visibility: ["app"] } }
29952
+ }, async ({ url: url2, data }) => {
29953
+ const validation = validateUrl(url2);
29954
+ if (!validation.valid) {
29955
+ return {
29956
+ content: [{ type: "text", text: validation.error }],
29957
+ isError: true
29958
+ };
29959
+ }
29960
+ const filePath = isFileUrl(url2) ? fileUrlToPath(url2) : isLocalPath(url2) ? decodeURIComponent(url2) : null;
29961
+ if (!filePath) {
29962
+ return {
29963
+ content: [
29964
+ { type: "text", text: "Save is only supported for local files" }
29965
+ ],
29966
+ isError: true
29967
+ };
29968
+ }
29969
+ const resolved = path.resolve(filePath);
29970
+ if (!isWritablePath(resolved)) {
29971
+ return {
29972
+ content: [
29973
+ {
29974
+ type: "text",
29975
+ text: "Save refused: file is not under a mounted directory root " + "and was not passed as a CLI argument. MCP file roots are " + "read-only (typically uploaded copies the client doesn't " + "expect to change)."
29976
+ }
29977
+ ],
29978
+ isError: true
29979
+ };
29980
+ }
29981
+ try {
29982
+ const bytes = Buffer.from(data, "base64");
29983
+ await fs.promises.writeFile(resolved, bytes);
29984
+ const { mtimeMs } = await fs.promises.stat(resolved);
29985
+ return {
29986
+ content: [{ type: "text", text: `Saved to ${filePath}` }],
29987
+ structuredContent: { filePath: resolved, mtimeMs }
29988
+ };
29989
+ } catch (err) {
29990
+ return {
29991
+ content: [
29992
+ {
29993
+ type: "text",
29994
+ text: `Failed to save: ${err instanceof Error ? err.message : String(err)}`
29995
+ }
29996
+ ],
29997
+ isError: true
29998
+ };
29999
+ }
30000
+ });
30001
+ mZ(server, RESOURCE_URI, RESOURCE_URI, { mimeType: d }, async () => {
28861
30002
  const html = await fs.promises.readFile(path.join(DIST_DIR, "mcp-app.html"), "utf-8");
28862
30003
  return {
28863
30004
  contents: [
28864
- { uri: RESOURCE_URI, mimeType: d, text: html }
30005
+ {
30006
+ uri: RESOURCE_URI,
30007
+ mimeType: d,
30008
+ text: html,
30009
+ _meta: {
30010
+ ui: {
30011
+ permissions: { clipboardWrite: {} },
30012
+ csp: {
30013
+ connectDomains: [STANDARD_FONT_ORIGIN],
30014
+ resourceDomains: [STANDARD_FONT_ORIGIN]
30015
+ }
30016
+ }
30017
+ }
30018
+ }
28865
30019
  ]
28866
30020
  };
28867
30021
  });
28868
30022
  return server;
28869
30023
  }
28870
30024
  export {
30025
+ writeFlags,
28871
30026
  validateUrl,
30027
+ stopFileWatch,
30028
+ startFileWatch,
28872
30029
  pathToFileUrl,
28873
30030
  normalizeArxivUrl,
30031
+ isWritablePath,
28874
30032
  isFileUrl,
28875
30033
  isArxivUrl,
28876
30034
  isAncestorDir,
28877
30035
  fileUrlToPath,
28878
30036
  createServer,
28879
30037
  createPdfCache,
30038
+ cliLocalFiles,
28880
30039
  allowedLocalFiles,
28881
30040
  allowedLocalDirs,
30041
+ STANDARD_FONT_DATA_URL,
28882
30042
  RESOURCE_URI,
28883
30043
  MAX_CHUNK_BYTES,
28884
30044
  DEFAULT_PDF,