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 };
};
}