{
"event": "PreToolUse",
"tool_name": "Read",
"tool_input": {
"file_path": "\/var\/www\/dev.campus.systemische-tools.de\/src\/Domain\/Entity\/Task.php",
"limit": 80
}
}
{
"tool_response": {
"type": "text",
"file": {
"filePath": "\/var\/www\/dev.campus.systemische-tools.de\/src\/Framework\/Controller.php",
"content": "<?php\n\ndeclare(strict_types=1);\n\nnamespace Framework;\n\n\/\/ @responsibility: Basis-Controller mit CSRF, JSON-Response, View-Rendering\n\nabstract class Controller\n{\n protected function csrfToken(): string\n {\n return CsrfService::getToken();\n }\n\n protected function csrfField(): string\n {\n return CsrfService::getTokenField();\n }\n\n protected function validateCsrf(): bool\n {\n $token = $_POST['_csrf_token'] ?? $_SERVER['HTTP_X_CSRF_TOKEN'] ?? null;\n\n return CsrfService::validateToken($token);\n }\n\n protected function requireCsrf(): void\n {\n if (!$this->validateCsrf()) {\n if ($this->isJsonRequest()) {\n $this->json(['error' => 'CSRF token invalid'], 403);\n } else {\n $this->text('CSRF token invalid', 403);\n }\n exit;\n }\n }\n\n private function isJsonRequest(): bool\n {\n $accept = $_SERVER['HTTP_ACCEPT'] ?? '';\n $contentType = $_SERVER['CONTENT_TYPE'] ?? '';\n\n return str_contains($accept, 'application\/json') || str_contains($contentType, 'application\/json');\n }\n\n protected function view(string $name, array $data = []): void\n {\n $data['csrfField'] = $this->csrfField();\n $data['csrfToken'] = $this->csrfToken();\n extract($data);\n $file = VIEW_PATH . '\/' . str_replace('.', '\/', $name) . '.php';\n\n if (file_exists($file)) {\n require $file;\n } else {\n throw new \\Exception(\"View not found: {$name}\");\n }\n }\n\n protected function json(mixed $data, int $status = 200): void\n {\n http_response_code($status);\n header('Content-Type: application\/json');\n echo json_encode($data, JSON_UNESCAPED_UNICODE);\n }\n\n protected function redirect(string $url, int $status = 302): never\n {\n http_response_code($status);\n header(\"Location: {$url}\");\n exit;\n }\n\n \/**\n * Render a partial template (for HTMX responses).\n *\n * @param string $name Partial name (e.g., 'chat\/message' or 'content\/version')\n * @param array<string, mixed> $data Data to pass to partial\n *\/\n protected function partial(string $name, array $data = []): void\n {\n $data['csrfField'] = $this->csrfField();\n $data['csrfToken'] = $this->csrfToken();\n extract($data);\n\n \/\/ Try module-specific partials first, then global partials\n $paths = [\n VIEW_PATH . '\/' . str_replace('.', '\/', $name) . '.php',\n VIEW_PATH . '\/partials\/' . str_replace('.', '\/', $name) . '.php',\n ];\n\n foreach ($paths as $file) {\n if (file_exists($file)) {\n require $file;\n\n return;\n }\n }",
"numLines": 100,
"startLine": 1,
"totalLines": 288
}
}
}