UNPKG

mcp-openapi-schema-explorer

Version:
86 lines 4.68 kB
import { ResourceTemplate, } from '@modelcontextprotocol/sdk/server/mcp.js'; import { RenderablePathItem } from '../rendering/path-item.js'; import { createErrorResult } from '../rendering/utils.js'; import { buildPathItemUriSuffix } from '../utils/uri-builder.js'; // Added .js extension // Import shared handler utils import { formatResults, isOpenAPIV3, getValidatedPathItem, // Import new helper getValidatedOperations, // Import new helper } from './handler-utils.js'; // Already has .js const BASE_URI = 'openapi://'; // Removed duplicated FormattedResultItem type - now imported from handler-utils // Removed duplicated formatResults function - now imported from handler-utils // Removed duplicated isOpenAPIV3 function - now imported from handler-utils /** * Handles requests for specific operation details within a path. * Corresponds to the `openapi://paths/{path}/{method*}` template. */ export class OperationHandler { constructor(specLoader, formatter) { this.specLoader = specLoader; this.formatter = formatter; this.handleRequest = async (uri, variables) => { const encodedPath = variables.path; // Correct variable access key: 'method', not 'method*' const methodVar = variables['method']; // Can be string or string[] // Decode the path received from the URI variable const decodedPath = decodeURIComponent(encodedPath || ''); // Use the builder to create the suffix, which will re-encode the path correctly const pathUriSuffix = buildPathItemUriSuffix(decodedPath); const context = { formatter: this.formatter, baseUri: BASE_URI }; let resultItems; try { // Normalize methods: Handle string for single value, array for multiple. let methods = []; if (Array.isArray(methodVar)) { methods = methodVar.map(m => String(m).trim().toLowerCase()); // Ensure elements are strings } else if (typeof methodVar === 'string') { methods = [methodVar.trim().toLowerCase()]; // Treat as single item array } methods = methods.filter(m => m.length > 0); // Remove empty strings if (methods.length === 0) { throw new Error('No valid HTTP method specified.'); } const spec = await this.specLoader.getTransformedSpec({ resourceType: 'schema', // Use 'schema' for now format: 'openapi', }); // Use imported type guard if (!isOpenAPIV3(spec)) { throw new Error('Only OpenAPI v3 specifications are supported'); } // --- Use helper to get validated path item --- const lookupPath = decodedPath.startsWith('/') ? decodedPath : `/${decodedPath}`; const pathItemObj = getValidatedPathItem(spec, lookupPath); // --- Use helper to get validated requested methods --- const validMethods = getValidatedOperations(pathItemObj, methods, lookupPath); // Instantiate RenderablePathItem with the validated pathItemObj const renderablePathItem = new RenderablePathItem(pathItemObj, // pathItemObj retrieved safely via helper lookupPath, // Pass the raw, decoded path pathUriSuffix // Pass the correctly built suffix ); // Use the validated methods returned by the helper resultItems = renderablePathItem.renderOperationDetail(context, validMethods); } catch (error) { // Catch errors from helpers (e.g., path/method not found) or rendering const message = error instanceof Error ? error.message : String(error); console.error(`Error handling request ${uri.href}: ${message}`); // Create a single error item representing the overall request failure resultItems = createErrorResult(uri.href.substring(BASE_URI.length), // Use request URI suffix message); } // Use imported formatResults const contents = formatResults(context, resultItems); return { contents }; }; } getTemplate() { // TODO: Add completion logic if needed return new ResourceTemplate(`${BASE_URI}paths/{path}/{method*}`, { list: undefined, complete: undefined, }); } } //# sourceMappingURL=operation-handler.js.map