Skip to content

Commit 061d152

Browse files
committed
fix: session id for initialization
1 parent 0d88a0f commit 061d152

File tree

2 files changed

+26
-7
lines changed

2 files changed

+26
-7
lines changed

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "mcp-framework",
3-
"version": "0.2.3",
3+
"version": "0.2.4",
44

55
"description": "Framework for building Model Context Protocol (MCP) servers in Typescript",
66
"type": "module",

src/transports/http/server.ts

+25-6
Original file line numberDiff line numberDiff line change
@@ -349,14 +349,32 @@ export class HttpStreamTransport extends AbstractTransport {
349349
logger.debug(`Handling GET request to ${this._config.endpoint}`);
350350
const acceptHeader = req.headers.accept || '';
351351
if (!acceptHeader.includes(SSE_CONTENT_TYPE) && !acceptHeader.includes('*/*')) throw this.httpError(406, `Not Acceptable: GET requires Accept header including ${SSE_CONTENT_TYPE}`);
352+
352353
const sessionIdHeader = getRequestHeader(req.headers, this._config.session.headerName);
353354
let session: SessionData | undefined;
354-
if (this._config.session.enabled) { session = this.validateSession(sessionIdHeader, req, true); session.lastActivity = Date.now(); }
355-
await this.handleAuthentication(req, res, `GET session ${session?.id || 'N/A'}`, session);
355+
356+
if (this._config.session.enabled && sessionIdHeader) {
357+
// If a session ID is provided, validate it
358+
session = this.validateSession(sessionIdHeader, req, false);
359+
session.lastActivity = Date.now();
360+
logger.debug(`Found valid session: ${session.id}`);
361+
await this.handleAuthentication(req, res, `GET session ${session.id}`, session);
362+
} else if (this._config.session.enabled) {
363+
// Allow initial GET requests without session ID during initialization phase
364+
logger.debug(`GET request without session ID - allowing as potential initialization connection`);
365+
await this.handleAuthentication(req, res, `GET initialization`, undefined);
366+
} else {
367+
// Sessions disabled
368+
await this.handleAuthentication(req, res, `GET (sessions disabled)`, undefined);
369+
}
370+
356371
const lastEventId = getRequestHeader(req.headers, "Last-Event-ID");
357-
if (lastEventId && !this._config.resumability.enabled) logger.warn(`Client sent Last-Event-ID (${lastEventId}) but resumability is disabled.`);
372+
if (lastEventId && !this._config.resumability.enabled) {
373+
logger.warn(`Client sent Last-Event-ID (${lastEventId}) but resumability is disabled.`);
374+
}
375+
358376
this.setupSSEConnection(req, res, session?.id, lastEventId);
359-
logger.debug(`Established SSE stream for GET request (Session: ${session?.id || 'N/A'})`);
377+
logger.debug(`Established SSE stream for GET request (Session: ${session?.id || 'initialization phase'})`);
360378
}
361379

362380
private async handleDelete(req: IncomingMessage, res: ServerResponse): Promise<void> {
@@ -520,8 +538,9 @@ export class HttpStreamTransport extends AbstractTransport {
520538
throw this.httpError(400, `Bad Request: Missing required session header ${headerName}`, -32601, undefined, requestId);
521539
}
522540
else {
523-
logger.error(`Programming error: validateSession called for initialization request with isMandatory=false`);
524-
throw this.httpError(500, "Internal Server Error: Session validation incorrectly called for initialization", -32603, undefined, requestId);
541+
// This is a valid case for initialization or when sessionId is optional
542+
logger.debug(`No session ID provided and not mandatory - acceptable for initialization`);
543+
return undefined as any; // Will be caught by typescript at call site
525544
}
526545
}
527546

0 commit comments

Comments
 (0)