transport.js 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  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 transport_exports = {};
  30. __export(transport_exports, {
  31. start: () => start
  32. });
  33. module.exports = __toCommonJS(transport_exports);
  34. var import_assert = __toESM(require("assert"));
  35. var import_http = __toESM(require("http"));
  36. var import_crypto = __toESM(require("crypto"));
  37. var import_utilsBundle = require("playwright-core/lib/utilsBundle");
  38. var mcpBundle = __toESM(require("./bundle"));
  39. var mcpServer = __toESM(require("./server"));
  40. async function start(serverBackendFactory, options) {
  41. if (options.port !== void 0) {
  42. const httpServer = await startHttpServer(options);
  43. startHttpTransport(httpServer, serverBackendFactory);
  44. } else {
  45. await startStdioTransport(serverBackendFactory);
  46. }
  47. }
  48. async function startStdioTransport(serverBackendFactory) {
  49. await mcpServer.connect(serverBackendFactory, new mcpBundle.StdioServerTransport(), false);
  50. }
  51. const testDebug = (0, import_utilsBundle.debug)("pw:mcp:test");
  52. async function handleSSE(serverBackendFactory, req, res, url, sessions) {
  53. if (req.method === "POST") {
  54. const sessionId = url.searchParams.get("sessionId");
  55. if (!sessionId) {
  56. res.statusCode = 400;
  57. return res.end("Missing sessionId");
  58. }
  59. const transport = sessions.get(sessionId);
  60. if (!transport) {
  61. res.statusCode = 404;
  62. return res.end("Session not found");
  63. }
  64. return await transport.handlePostMessage(req, res);
  65. } else if (req.method === "GET") {
  66. const transport = new mcpBundle.SSEServerTransport("/sse", res);
  67. sessions.set(transport.sessionId, transport);
  68. testDebug(`create SSE session: ${transport.sessionId}`);
  69. await mcpServer.connect(serverBackendFactory, transport, false);
  70. res.on("close", () => {
  71. testDebug(`delete SSE session: ${transport.sessionId}`);
  72. sessions.delete(transport.sessionId);
  73. });
  74. return;
  75. }
  76. res.statusCode = 405;
  77. res.end("Method not allowed");
  78. }
  79. async function handleStreamable(serverBackendFactory, req, res, sessions) {
  80. const sessionId = req.headers["mcp-session-id"];
  81. if (sessionId) {
  82. const transport = sessions.get(sessionId);
  83. if (!transport) {
  84. res.statusCode = 404;
  85. res.end("Session not found");
  86. return;
  87. }
  88. return await transport.handleRequest(req, res);
  89. }
  90. if (req.method === "POST") {
  91. const transport = new mcpBundle.StreamableHTTPServerTransport({
  92. sessionIdGenerator: () => import_crypto.default.randomUUID(),
  93. onsessioninitialized: async (sessionId2) => {
  94. testDebug(`create http session: ${transport.sessionId}`);
  95. await mcpServer.connect(serverBackendFactory, transport, true);
  96. sessions.set(sessionId2, transport);
  97. }
  98. });
  99. transport.onclose = () => {
  100. if (!transport.sessionId)
  101. return;
  102. sessions.delete(transport.sessionId);
  103. testDebug(`delete http session: ${transport.sessionId}`);
  104. };
  105. await transport.handleRequest(req, res);
  106. return;
  107. }
  108. res.statusCode = 400;
  109. res.end("Invalid request");
  110. }
  111. function startHttpTransport(httpServer, serverBackendFactory) {
  112. const sseSessions = /* @__PURE__ */ new Map();
  113. const streamableSessions = /* @__PURE__ */ new Map();
  114. httpServer.on("request", async (req, res) => {
  115. const url2 = new URL(`http://localhost${req.url}`);
  116. if (url2.pathname.startsWith("/sse"))
  117. await handleSSE(serverBackendFactory, req, res, url2, sseSessions);
  118. else
  119. await handleStreamable(serverBackendFactory, req, res, streamableSessions);
  120. });
  121. const url = httpAddressToString(httpServer.address());
  122. const message = [
  123. `Listening on ${url}`,
  124. "Put this in your client config:",
  125. JSON.stringify({
  126. "mcpServers": {
  127. "playwright": {
  128. "url": `${url}/mcp`
  129. }
  130. }
  131. }, void 0, 2),
  132. "For legacy SSE transport support, you can use the /sse endpoint instead."
  133. ].join("\n");
  134. console.error(message);
  135. }
  136. async function startHttpServer(config) {
  137. const { host, port } = config;
  138. const httpServer = import_http.default.createServer();
  139. await new Promise((resolve, reject) => {
  140. httpServer.on("error", reject);
  141. httpServer.listen(port, host, () => {
  142. resolve();
  143. httpServer.removeListener("error", reject);
  144. });
  145. });
  146. return httpServer;
  147. }
  148. function httpAddressToString(address) {
  149. (0, import_assert.default)(address, "Could not bind server socket");
  150. if (typeof address === "string")
  151. return address;
  152. const resolvedPort = address.port;
  153. let resolvedHost = address.family === "IPv4" ? address.address : `[${address.address}]`;
  154. if (resolvedHost === "0.0.0.0" || resolvedHost === "[::]")
  155. resolvedHost = "localhost";
  156. return `http://${resolvedHost}:${resolvedPort}`;
  157. }
  158. // Annotate the CommonJS export names for ESM import in node:
  159. 0 && (module.exports = {
  160. start
  161. });