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
54 lines
1.4 KiB
PHP
54 lines
1.4 KiB
PHP
<?php
|
|
|
|
// SPDX-License-Identifier: EUPL-1.2
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Core\Mcp\Resources;
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
use Illuminate\Support\Facades\Schema;
|
|
use Laravel\Mcp\Request;
|
|
use Laravel\Mcp\Response;
|
|
use Laravel\Mcp\Server\Resource;
|
|
|
|
final class DatabaseSchema extends Resource
|
|
{
|
|
protected string $description = 'Database schema information for Host Hub';
|
|
|
|
public function handle(Request $request): Response
|
|
{
|
|
$schema = [];
|
|
|
|
foreach ($this->tables() as $tableName) {
|
|
$schema[$tableName] = $this->describeTable($tableName);
|
|
}
|
|
|
|
return Response::text((string) json_encode($schema, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES));
|
|
}
|
|
|
|
protected function tables(): array
|
|
{
|
|
try {
|
|
return collect(DB::select('SHOW TABLES'))
|
|
->map(static fn (object $row): string => (string) array_values((array) $row)[0])
|
|
->all();
|
|
} catch (\Throwable) {
|
|
return Schema::getTableListing();
|
|
}
|
|
}
|
|
|
|
protected function describeTable(string $tableName): array
|
|
{
|
|
$driver = DB::getDriverName();
|
|
|
|
try {
|
|
return array_map(static fn (object $column): array => (array) $column, DB::select(sprintf(
|
|
$driver === 'sqlite' ? 'PRAGMA table_info("%s")' : 'DESCRIBE `%s`',
|
|
$tableName,
|
|
)));
|
|
} catch (\Throwable) {
|
|
return [];
|
|
}
|
|
}
|
|
}
|