php-llm / mcp-sdk
Model Context Protocol SDK for Client and Server applications in PHP
dev-main
2025-03-31 22:46 UTC
Requires
- php: ^8.2
- php-llm/llm-chain: ^0.19.0
- psr/log: ^3.0
- symfony/uid: ^6.4 || ^7.0
Requires (Dev)
- php-cs-fixer/shim: ^3.70
- phpstan/phpstan: ^2.1
- phpunit/phpunit: ^11.5
- rector/rector: ^2.0
- symfony/console: ^6.4 || ^7.0
Suggests
- symfony/console: To use SymfonyConsoleTransport for STDIO
This package is auto-updated.
Last update: 2025-03-31 22:47:17 UTC
README
Model Context Protocol SDK for Client and Server applications in PHP. Currently only support Tool Calls as Server via Server-Sent Events (SSE) and STDIO.
See Demo App for a working example and MCP Bundle for Symfony integration.
Installation
composer require php-llm/mcp-sdk
Usage with Symfony
Server integration points for are tailored to Symfony Console and HttpFoundation (Laravel compatible).
Console Command for STDIO Server
namespace App\Command; use PhpLlm\McpSdk\Server; use PhpLlm\McpSdk\Server\Transport\Stdio\SymfonyConsoleTransport; use Symfony\Component\Console\Attribute\AsCommand; use Symfony\Component\Console\Command\Command; use Symfony\Component\Console\Input\InputInterface; use Symfony\Component\Console\Output\OutputInterface; #[AsCommand('mcp', 'Starts an MCP server')] final class McpCommand extends Command { public function __construct( private readonly Server $server, ) { parent::__construct(); } protected function execute(InputInterface $input, OutputInterface $output): int { $this->server->connect( new SymfonyConsoleTransport($input, $output) ); return Command::SUCCESS; } }
Controller for Server-Sent Events Server
namespace App\Controller; use PhpLlm\McpSdk\Server; use PhpLlm\McpSdk\Server\Transport\Sse\Store\CachePoolStore; use PhpLlm\McpSdk\Server\Transport\Sse\StreamTransport; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpFoundation\StreamedResponse; use Symfony\Component\HttpKernel\Attribute\AsController; use Symfony\Component\Routing\Attribute\Route; use Symfony\Component\Routing\Generator\UrlGeneratorInterface; use Symfony\Component\Uid\Uuid; #[AsController] #[Route('/mcp', name: 'mcp_')] final readonly class McpController { public function __construct( private Server $server, private CachePoolStore $store, private UrlGeneratorInterface $urlGenerator, ) { } #[Route('/sse', name: 'sse', methods: ['GET'])] public function sse(): StreamedResponse { $id = Uuid::v4(); $endpoint = $this->urlGenerator->generate('mcp_messages', ['id' => $id], UrlGeneratorInterface::ABSOLUTE_URL); $transport = new StreamTransport($endpoint, $this->store, $id); return new StreamedResponse(fn() => $this->server->connect($transport), headers: [ 'Content-Type' => 'text/event-stream', 'Cache-Control' => 'no-cache', 'X-Accel-Buffering' => 'no', ]); } #[Route('/messages/{id}', name: 'messages', methods: ['POST'])] public function messages(Request $request, Uuid $id): Response { $this->store->push($id, $request->getContent()); return new Response(); } }
Exposing Tools
Under the hood the SDK uses LLM Chain's ToolBox
to register, analyze and
execute tools. In combination with its Symfony Bundle you can expose
tools with #[AsTool]
attribute.
use PhpLlm\LlmChain\ToolBox\Attribute\AsTool; #[AsTool('company_name', 'Provides the name of your company')] final class CompanyName { public function __invoke(): string { return 'ACME Corp.' } }
See LLM Chain Documentation for more information.