agent/php/Mcp/Resources/DatabaseSchema.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

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 [];
}
}
}