Skip to main content
Glama
kadykov

OpenAPI Schema Explorer

operation-handler.ts4.59 kB
import { ReadResourceTemplateCallback, ResourceTemplate, } from '@modelcontextprotocol/sdk/server/mcp.js'; import { Variables } from '@modelcontextprotocol/sdk/shared/uriTemplate.js'; import { SpecLoaderService } from '../types.js'; import { IFormatter } from '../services/formatters.js'; import { RenderablePathItem } from '../rendering/path-item.js'; import { RenderContext, RenderResultItem } from '../rendering/types.js'; import { createErrorResult } from '../rendering/utils.js'; import { buildPathItemUriSuffix } from '../utils/uri-builder.js'; // Added .js extension // Import shared handler utils import { formatResults, isOpenAPIV3, FormattedResultItem, 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( private specLoader: SpecLoaderService, private formatter: IFormatter ) {} getTemplate(): ResourceTemplate { // TODO: Add completion logic if needed return new ResourceTemplate(`${BASE_URI}paths/{path}/{method*}`, { list: undefined, complete: undefined, }); } handleRequest: ReadResourceTemplateCallback = async ( uri: URL, variables: Variables ): Promise<{ contents: FormattedResultItem[] }> => { const encodedPath = variables.path as string; // 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: RenderContext = { formatter: this.formatter, baseUri: BASE_URI }; let resultItems: RenderResultItem[]; try { // Normalize methods: Handle string for single value, array for multiple. let methods: string[] = []; 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: unknown) { // 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 }; }; }

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/kadykov/mcp-openapi-schema-explorer'

If you have feedback or need assistance with the MCP directory API, please join our Discord server