implemented step 03 by Gemini

This commit is contained in:
2026-06-12 16:11:21 +02:00
parent 3c6344e94f
commit ed7dfe7795

69
public/ajax.php Normal file
View File

@ -0,0 +1,69 @@
<?php
declare(strict_types=1);
require_once __DIR__ . '/../vendor/autoload.php';
// Set headers
header('Content-Type: application/json; charset=utf-8');
// Error handling
set_error_handler(function ($errno, $errstr, $errfile, $errline) {
throw new ErrorException($errstr, 0, $errno, $errfile, $errline);
});
function sendResponse(bool $success, $dataOrError, int $httpStatus = 200): void {
http_response_code($httpStatus);
$response = ['success' => $success];
if ($success) {
$response['data'] = $dataOrError;
} else {
$response['error'] = $dataOrError;
}
echo json_encode($response, JSON_UNESCAPED_UNICODE | JSON_THROW_ON_ERROR);
exit;
}
try {
// Only POST allowed
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
sendResponse(false, ['code' => 'METHOD_NOT_ALLOWED', 'message' => 'Only POST requests are allowed.'], 405);
}
// Read JSON input
$input = file_get_contents('php://input');
$data = json_decode($input, true);
if (json_last_error() !== JSON_ERROR_NONE) {
sendResponse(false, ['code' => 'INVALID_JSON', 'message' => 'Invalid JSON input.'], 400);
}
// Validate action
$action = $data['action'] ?? null;
if (!$action) {
sendResponse(false, ['code' => 'MISSING_ACTION', 'message' => 'Action is required.'], 400);
}
// Check X-User-ID header (except for initSession if we want to allow it)
$userId = $_SERVER['HTTP_X_USER_ID'] ?? null;
if (!$userId && $action !== 'initSession') {
sendResponse(false, ['code' => 'UNAUTHORIZED', 'message' => 'X-User-ID header is missing.'], 401);
}
// Router
switch ($action) {
case 'ping':
sendResponse(true, ['message' => 'pong', 'timestamp' => time()]);
break;
default:
sendResponse(false, ['code' => 'UNKNOWN_ACTION', 'message' => "Action '$action' is not defined."], 404);
break;
}
} catch (Throwable $e) {
sendResponse(false, [
'code' => 'INTERNAL_SERVER_ERROR',
'message' => $e->getMessage()
], 500);
}