Update npm registry domain from privatenpm.hanson.xyz to npm.internal.hanson.xyz 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
153 lines
6.1 KiB
JavaScript
Executable File
153 lines
6.1 KiB
JavaScript
Executable File
"use strict";
|
|
var __create = Object.create;
|
|
var __defProp = Object.defineProperty;
|
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
var __getProtoOf = Object.getPrototypeOf;
|
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
var __export = (target, all) => {
|
|
for (var name in all)
|
|
__defProp(target, name, { get: all[name], enumerable: true });
|
|
};
|
|
var __copyProps = (to, from, except, desc) => {
|
|
if (from && typeof from === "object" || typeof from === "function") {
|
|
for (let key of __getOwnPropNames(from))
|
|
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
}
|
|
return to;
|
|
};
|
|
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
// If the importer is in node compatibility mode or this is not an ESM
|
|
// file that has been converted to a CommonJS file using a Babel-
|
|
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
mod
|
|
));
|
|
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
var http_exports = {};
|
|
__export(http_exports, {
|
|
addressToString: () => addressToString,
|
|
startMcpHttpServer: () => startMcpHttpServer
|
|
});
|
|
module.exports = __toCommonJS(http_exports);
|
|
var import_assert = __toESM(require("assert"));
|
|
var import_crypto = __toESM(require("crypto"));
|
|
var import_utilsBundle = require("playwright-core/lib/utilsBundle");
|
|
var mcpBundle = __toESM(require("playwright-core/lib/mcpBundle"));
|
|
var import_utils = require("playwright-core/lib/utils");
|
|
var mcpServer = __toESM(require("./server"));
|
|
const testDebug = (0, import_utilsBundle.debug)("pw:mcp:test");
|
|
async function startMcpHttpServer(config, serverBackendFactory, allowedHosts) {
|
|
const httpServer = (0, import_utils.createHttpServer)();
|
|
await (0, import_utils.startHttpServer)(httpServer, config);
|
|
return await installHttpTransport(httpServer, serverBackendFactory, allowedHosts);
|
|
}
|
|
function addressToString(address, options) {
|
|
(0, import_assert.default)(address, "Could not bind server socket");
|
|
if (typeof address === "string")
|
|
throw new Error("Unexpected address type: " + address);
|
|
let host = address.family === "IPv4" ? address.address : `[${address.address}]`;
|
|
if (options.normalizeLoopback && (host === "0.0.0.0" || host === "[::]" || host === "[::1]" || host === "127.0.0.1"))
|
|
host = "localhost";
|
|
return `${options.protocol}://${host}:${address.port}`;
|
|
}
|
|
async function installHttpTransport(httpServer, serverBackendFactory, allowedHosts) {
|
|
const url = addressToString(httpServer.address(), { protocol: "http", normalizeLoopback: true });
|
|
const host = new URL(url).host;
|
|
allowedHosts = (allowedHosts || [host]).map((h) => h.toLowerCase());
|
|
const allowAnyHost = allowedHosts.includes("*");
|
|
const sseSessions = /* @__PURE__ */ new Map();
|
|
const streamableSessions = /* @__PURE__ */ new Map();
|
|
httpServer.on("request", async (req, res) => {
|
|
if (!allowAnyHost) {
|
|
const host2 = req.headers.host?.toLowerCase();
|
|
if (!host2) {
|
|
res.statusCode = 400;
|
|
return res.end("Missing host");
|
|
}
|
|
if (!allowedHosts.includes(host2)) {
|
|
res.statusCode = 403;
|
|
return res.end("Access is only allowed at " + allowedHosts.join(", "));
|
|
}
|
|
}
|
|
const url2 = new URL(`http://localhost${req.url}`);
|
|
if (url2.pathname === "/killkillkill" && req.method === "GET") {
|
|
res.statusCode = 200;
|
|
res.end("Killing process");
|
|
process.emit("SIGINT");
|
|
return;
|
|
}
|
|
if (url2.pathname.startsWith("/sse"))
|
|
await handleSSE(serverBackendFactory, req, res, url2, sseSessions);
|
|
else
|
|
await handleStreamable(serverBackendFactory, req, res, streamableSessions);
|
|
});
|
|
return url;
|
|
}
|
|
async function handleSSE(serverBackendFactory, req, res, url, sessions) {
|
|
if (req.method === "POST") {
|
|
const sessionId = url.searchParams.get("sessionId");
|
|
if (!sessionId) {
|
|
res.statusCode = 400;
|
|
return res.end("Missing sessionId");
|
|
}
|
|
const transport = sessions.get(sessionId);
|
|
if (!transport) {
|
|
res.statusCode = 404;
|
|
return res.end("Session not found");
|
|
}
|
|
return await transport.handlePostMessage(req, res);
|
|
} else if (req.method === "GET") {
|
|
const transport = new mcpBundle.SSEServerTransport("/sse", res);
|
|
sessions.set(transport.sessionId, transport);
|
|
testDebug(`create SSE session: ${transport.sessionId}`);
|
|
await mcpServer.connect(serverBackendFactory, transport, false);
|
|
res.on("close", () => {
|
|
testDebug(`delete SSE session: ${transport.sessionId}`);
|
|
sessions.delete(transport.sessionId);
|
|
});
|
|
return;
|
|
}
|
|
res.statusCode = 405;
|
|
res.end("Method not allowed");
|
|
}
|
|
async function handleStreamable(serverBackendFactory, req, res, sessions) {
|
|
const sessionId = req.headers["mcp-session-id"];
|
|
if (sessionId) {
|
|
const transport = sessions.get(sessionId);
|
|
if (!transport) {
|
|
res.statusCode = 404;
|
|
res.end("Session not found");
|
|
return;
|
|
}
|
|
return await transport.handleRequest(req, res);
|
|
}
|
|
if (req.method === "POST") {
|
|
const transport = new mcpBundle.StreamableHTTPServerTransport({
|
|
sessionIdGenerator: () => import_crypto.default.randomUUID(),
|
|
onsessioninitialized: async (sessionId2) => {
|
|
testDebug(`create http session: ${transport.sessionId}`);
|
|
await mcpServer.connect(serverBackendFactory, transport, true);
|
|
sessions.set(sessionId2, transport);
|
|
}
|
|
});
|
|
transport.onclose = () => {
|
|
if (!transport.sessionId)
|
|
return;
|
|
sessions.delete(transport.sessionId);
|
|
testDebug(`delete http session: ${transport.sessionId}`);
|
|
};
|
|
await transport.handleRequest(req, res);
|
|
return;
|
|
}
|
|
res.statusCode = 400;
|
|
res.end("Invalid request");
|
|
}
|
|
// Annotate the CommonJS export names for ESM import in node:
|
|
0 && (module.exports = {
|
|
addressToString,
|
|
startMcpHttpServer
|
|
});
|