agent/php/Mcp/Services/ToolRateLimiter.php
Snider 91551dec9b feat(mcp): implement extended RFC services + transport (#842)
Additive-only — no existing files modified.

Services (php/Mcp/Services/):
- CircuitBreaker (3-state, Cache::add trial lock)
- DataRedactor (28 sensitive + 16 PII keys, partial-redact algorithm)
- McpHealthService (YAML registry + JSON-RPC stdio ping protocolVersion 2024-11-05)
- McpMetricsService (p50/p95/p99 linear interpolation)
- McpWebhookDispatcher (mcp.tool.executed → WebhookEndpoints)
- OpenApiGenerator (OpenAPI 3.0.3)
- ToolRateLimiter (Cache::put first, Cache::increment after — no reset)
- AgentSessionService (php/Mod/Mcp/Services/ namespace per spec)

Transport (php/Mcp/Transport/):
- McpContext (transport-agnostic callbacks)
- Contracts/McpToolHandler interface

Resources (php/Mcp/Resources/):
- AppConfig, ContentResource, DatabaseSchema

Config: php/resources/mcp/registry.yaml.
Pest Feature tests _Good/_Bad/_Ugly per AX-10 for each new class.

Co-authored-by: Codex <noreply@openai.com>
Closes tasks.lthn.sh/view.php?id=842
2026-04-25 05:50:16 +01:00

117 lines
3.3 KiB
PHP

<?php
// SPDX-License-Identifier: EUPL-1.2
declare(strict_types=1);
namespace Core\Mcp\Services;
use Illuminate\Support\Facades\Cache;
final class ToolRateLimiter
{
protected const CACHE_PREFIX = 'mcp_rate_limit:';
public function check(string $identifier, string $toolName): array
{
if (! config('mcp.rate_limiting.enabled', true)) {
return ['limited' => false, 'remaining' => PHP_INT_MAX, 'retry_after' => null];
}
$limit = $this->limitForTool($toolName);
$cacheKey = $this->cacheKey($identifier, $toolName);
$current = (int) Cache::get($cacheKey, 0);
$decaySeconds = (int) config('mcp.rate_limiting.decay_seconds', 60);
if ($current >= $limit) {
$ttl = $this->ttl($cacheKey, $decaySeconds);
return [
'limited' => true,
'remaining' => 0,
'retry_after' => $ttl > 0 ? $ttl : $decaySeconds,
];
}
return [
'limited' => false,
'remaining' => max($limit - $current - 1, 0),
'retry_after' => null,
];
}
public function hit(string $identifier, string $toolName): void
{
if (! config('mcp.rate_limiting.enabled', true)) {
return;
}
$cacheKey = $this->cacheKey($identifier, $toolName);
$current = (int) Cache::get($cacheKey, 0);
$decaySeconds = (int) config('mcp.rate_limiting.decay_seconds', 60);
if ($current === 0) {
Cache::put($cacheKey, 1, $decaySeconds);
return;
}
Cache::increment($cacheKey);
}
public function clear(string $identifier, ?string $toolName = null): void
{
if ($toolName !== null) {
Cache::forget($this->cacheKey($identifier, $toolName));
return;
}
foreach (array_keys((array) config('mcp.rate_limiting.per_tool', [])) as $configuredTool) {
Cache::forget($this->cacheKey($identifier, (string) $configuredTool));
}
Cache::forget($this->cacheKey($identifier, '*'));
}
public function getStatus(string $identifier, string $toolName): array
{
$limit = $this->limitForTool($toolName);
$cacheKey = $this->cacheKey($identifier, $toolName);
$current = (int) Cache::get($cacheKey, 0);
$ttl = $this->ttl($cacheKey, (int) config('mcp.rate_limiting.decay_seconds', 60));
return [
'limit' => $limit,
'remaining' => max($limit - $current, 0),
'reset_at' => $ttl > 0 ? now()->addSeconds($ttl)->toIso8601String() : null,
];
}
protected function limitForTool(string $toolName): int
{
$perTool = (array) config('mcp.rate_limiting.per_tool', []);
if (array_key_exists($toolName, $perTool)) {
return (int) $perTool[$toolName];
}
return (int) config('mcp.rate_limiting.calls_per_minute', 60);
}
protected function cacheKey(string $identifier, string $toolName): string
{
return self::CACHE_PREFIX.$identifier.':'.$toolName;
}
protected function ttl(string $cacheKey, int $default): int
{
try {
$ttl = Cache::ttl($cacheKey);
return is_int($ttl) ? $ttl : $default;
} catch (\Throwable) {
return $default;
}
}
}