server.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. "use strict";
  2. var __create = Object.create;
  3. var __defProp = Object.defineProperty;
  4. var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
  5. var __getOwnPropNames = Object.getOwnPropertyNames;
  6. var __getProtoOf = Object.getPrototypeOf;
  7. var __hasOwnProp = Object.prototype.hasOwnProperty;
  8. var __export = (target, all) => {
  9. for (var name in all)
  10. __defProp(target, name, { get: all[name], enumerable: true });
  11. };
  12. var __copyProps = (to, from, except, desc) => {
  13. if (from && typeof from === "object" || typeof from === "function") {
  14. for (let key of __getOwnPropNames(from))
  15. if (!__hasOwnProp.call(to, key) && key !== except)
  16. __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
  17. }
  18. return to;
  19. };
  20. var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
  21. // If the importer is in node compatibility mode or this is not an ESM
  22. // file that has been converted to a CommonJS file using a Babel-
  23. // compatible transform (i.e. "__esModule" has not been set), then set
  24. // "default" to the CommonJS "module.exports" for node compatibility.
  25. isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
  26. mod
  27. ));
  28. var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
  29. var server_exports = {};
  30. __export(server_exports, {
  31. connect: () => connect,
  32. createServer: () => createServer
  33. });
  34. module.exports = __toCommonJS(server_exports);
  35. var import_utilsBundle = require("playwright-core/lib/utilsBundle");
  36. var mcpBundle = __toESM(require("./bundle"));
  37. const serverDebug = (0, import_utilsBundle.debug)("pw:mcp:server");
  38. const errorsDebug = (0, import_utilsBundle.debug)("pw:mcp:errors");
  39. async function connect(serverBackendFactory, transport, runHeartbeat) {
  40. const backend = serverBackendFactory();
  41. const server = createServer(backend, runHeartbeat);
  42. await server.connect(transport);
  43. }
  44. function createServer(backend, runHeartbeat) {
  45. let initializedCallback = () => {
  46. };
  47. const initializedPromise = new Promise((resolve) => initializedCallback = resolve);
  48. const server = new mcpBundle.Server({ name: backend.name, version: backend.version }, {
  49. capabilities: {
  50. tools: {}
  51. }
  52. });
  53. server.setRequestHandler(mcpBundle.ListToolsRequestSchema, async () => {
  54. serverDebug("listTools");
  55. await initializedPromise;
  56. const tools = await backend.listTools();
  57. return { tools };
  58. });
  59. let heartbeatRunning = false;
  60. server.setRequestHandler(mcpBundle.CallToolRequestSchema, async (request) => {
  61. serverDebug("callTool", request);
  62. await initializedPromise;
  63. if (runHeartbeat && !heartbeatRunning) {
  64. heartbeatRunning = true;
  65. startHeartbeat(server);
  66. }
  67. try {
  68. return await backend.callTool(request.params.name, request.params.arguments || {});
  69. } catch (error) {
  70. return {
  71. content: [{ type: "text", text: "### Result\n" + String(error) }],
  72. isError: true
  73. };
  74. }
  75. });
  76. addServerListener(server, "initialized", async () => {
  77. try {
  78. const capabilities = server.getClientCapabilities();
  79. let clientRoots = [];
  80. if (capabilities?.roots) {
  81. const { roots } = await server.listRoots(void 0, { timeout: 2e3 }).catch(() => ({ roots: [] }));
  82. clientRoots = roots;
  83. }
  84. const clientVersion = server.getClientVersion() ?? { name: "unknown", version: "unknown" };
  85. await backend.initialize?.(clientVersion, clientRoots);
  86. initializedCallback();
  87. } catch (e) {
  88. errorsDebug(e);
  89. }
  90. });
  91. addServerListener(server, "close", () => backend.serverClosed?.());
  92. return server;
  93. }
  94. const startHeartbeat = (server) => {
  95. const beat = () => {
  96. Promise.race([
  97. server.ping(),
  98. new Promise((_, reject) => setTimeout(() => reject(new Error("ping timeout")), 5e3))
  99. ]).then(() => {
  100. setTimeout(beat, 3e3);
  101. }).catch(() => {
  102. void server.close();
  103. });
  104. };
  105. beat();
  106. };
  107. function addServerListener(server, event, listener) {
  108. const oldListener = server[`on${event}`];
  109. server[`on${event}`] = () => {
  110. oldListener?.();
  111. listener();
  112. };
  113. }
  114. // Annotate the CommonJS export names for ESM import in node:
  115. 0 && (module.exports = {
  116. connect,
  117. createServer
  118. });