Skip to content

Commit a29c126

Browse files
fix: ensure JSON-RPC error responses include message IDs
1 parent 719262b commit a29c126

File tree

3 files changed

+37
-35
lines changed

3 files changed

+37
-35
lines changed

src/JsonRpc/Handler.php

Lines changed: 26 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -94,9 +94,6 @@ public static function make(
9494

9595
/**
9696
* @return iterable<array{string|null, array<string, mixed>}>
97-
*
98-
* @throws ExceptionInterface When a handler throws an exception during message processing
99-
* @throws \JsonException When JSON encoding of the response fails
10097
*/
10198
public function process(string $input, ?Uuid $sessionId): iterable
10299
{
@@ -161,9 +158,11 @@ public function process(string $input, ?Uuid $sessionId): iterable
161158
}
162159

163160
foreach ($messages as $message) {
161+
$messageId = method_exists($message, 'getId') ? $message->getId() : 0;
162+
164163
if ($message instanceof InvalidInputMessageException) {
165164
$this->logger->warning('Failed to create message.', ['exception' => $message]);
166-
$error = Error::forInvalidRequest($message->getMessage(), 0);
165+
$error = Error::forInvalidRequest($message->getMessage(), $messageId);
167166
yield [$this->encodeResponse($error), []];
168167
continue;
169168
}
@@ -183,17 +182,17 @@ public function process(string $input, ?Uuid $sessionId): iterable
183182
['exception' => $e],
184183
);
185184

186-
$error = Error::forMethodNotFound($e->getMessage());
185+
$error = Error::forMethodNotFound($e->getMessage(), $messageId);
187186
yield [$this->encodeResponse($error), []];
188187
} catch (\InvalidArgumentException $e) {
189188
$this->logger->warning(\sprintf('Invalid argument: %s', $e->getMessage()), ['exception' => $e]);
190189

191-
$error = Error::forInvalidParams($e->getMessage());
190+
$error = Error::forInvalidParams($e->getMessage(), $messageId);
192191
yield [$this->encodeResponse($error), []];
193192
} catch (\Throwable $e) {
194193
$this->logger->critical(\sprintf('Uncaught exception: %s', $e->getMessage()), ['exception' => $e]);
195194

196-
$error = Error::forInternalError($e->getMessage());
195+
$error = Error::forInternalError($e->getMessage(), $messageId);
197196
yield [$this->encodeResponse($error), []];
198197
}
199198
}
@@ -202,7 +201,7 @@ public function process(string $input, ?Uuid $sessionId): iterable
202201
}
203202

204203
/**
205-
* @throws \JsonException When JSON encoding fails
204+
* Encodes a response to JSON, handling encoding errors gracefully.
206205
*/
207206
private function encodeResponse(Response|Error|null $response): ?string
208207
{
@@ -214,11 +213,26 @@ private function encodeResponse(Response|Error|null $response): ?string
214213

215214
$this->logger->info('Encoding response.', ['response' => $response]);
216215

217-
if ($response instanceof Response && [] === $response->result) {
218-
return json_encode($response, \JSON_THROW_ON_ERROR | \JSON_FORCE_OBJECT);
219-
}
216+
try {
217+
if ($response instanceof Response && [] === $response->result) {
218+
return json_encode($response, \JSON_THROW_ON_ERROR | \JSON_FORCE_OBJECT);
219+
}
220220

221-
return json_encode($response, \JSON_THROW_ON_ERROR);
221+
return json_encode($response, \JSON_THROW_ON_ERROR);
222+
} catch (\JsonException $e) {
223+
$this->logger->error('Failed to encode response to JSON.', [
224+
'message_id' => $response->getId(),
225+
'exception' => $e,
226+
]);
227+
228+
$fallbackError = new Error(
229+
id: $response->getId(),
230+
code: Error::INTERNAL_ERROR,
231+
message: 'Response could not be encoded to JSON'
232+
);
233+
234+
return json_encode($fallbackError, \JSON_THROW_ON_ERROR);
235+
}
222236
}
223237

224238
/**

src/Server.php

Lines changed: 5 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -44,19 +44,12 @@ public function connect(TransportInterface $transport): void
4444
]);
4545

4646
$transport->onMessage(function (string $message, ?Uuid $sessionId) use ($transport) {
47-
try {
48-
foreach ($this->jsonRpcHandler->process($message, $sessionId) as [$response, $context]) {
49-
if (null === $response) {
50-
continue;
51-
}
52-
53-
$transport->send($response, $context);
47+
foreach ($this->jsonRpcHandler->process($message, $sessionId) as [$response, $context]) {
48+
if (null === $response) {
49+
continue;
5450
}
55-
} catch (\JsonException $e) {
56-
$this->logger->error('Failed to encode response to JSON.', [
57-
'message' => $message,
58-
'exception' => $e,
59-
]);
51+
52+
$transport->send($response, $context);
6053
}
6154
});
6255

tests/ServerTest.php

Lines changed: 6 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -14,37 +14,32 @@
1414
use Mcp\JsonRpc\Handler;
1515
use Mcp\Server;
1616
use Mcp\Server\Transport\InMemoryTransport;
17-
use PHPUnit\Framework\MockObject\Stub\Exception;
1817
use PHPUnit\Framework\TestCase;
19-
use Psr\Log\NullLogger;
2018

2119
class ServerTest extends TestCase
2220
{
2321
public function testJsonExceptions()
2422
{
25-
$logger = $this->getMockBuilder(NullLogger::class)
26-
->disableOriginalConstructor()
27-
->onlyMethods(['error'])
28-
->getMock();
29-
$logger->expects($this->once())->method('error');
30-
3123
$handler = $this->getMockBuilder(Handler::class)
3224
->disableOriginalConstructor()
3325
->onlyMethods(['process'])
3426
->getMock();
3527

3628
$handler->expects($this->exactly(2))->method('process')->willReturnOnConsecutiveCalls(
37-
new Exception(new \JsonException('foobar')),
29+
[['{"jsonrpc":"2.0","id":0,"error":{"code":-32700,"message":"Parse error"}}', []]],
3830
[['success', []]]
3931
);
4032

4133
$transport = $this->getMockBuilder(InMemoryTransport::class)
4234
->setConstructorArgs([['foo', 'bar']])
4335
->onlyMethods(['send'])
4436
->getMock();
45-
$transport->expects($this->once())->method('send')->with('success', []);
37+
$transport->expects($this->exactly(2))->method('send')->willReturnOnConsecutiveCalls(
38+
null,
39+
null
40+
);
4641

47-
$server = new Server($handler, $logger);
42+
$server = new Server($handler);
4843
$server->connect($transport);
4944

5045
$transport->listen();

0 commit comments

Comments
 (0)