bericht/api/_jwt.php
Eduard Wisch bed611cd8b
All checks were successful
Deploy bericht / deploy (push) Successful in 1s
feat: Phase 2.3 + 2.4 — REST-API mit JWT-Auth
- api/_jwt.php: HS256 JWT encode/decode/from_request, Secret aus
  dolibarr_main_instance_unique_id, 7 Tage TTL
- api/_inc.php: gemeinsamer API-Init mit CORS, JSON-Helpers,
  api_authenticate() lädt User aus JWT und prüft bericht/read
- api/auth.php: POST { login, password } → JWT mit user + perms
- api/orders.php:
  - GET /api/orders.php — Liste der Aufträge des Users (Multi-User
    Filter über fk_user_*, Admin sieht alle)
  - GET /api/orders.php?id=X — Auftrags-Detail mit Kunde + Berichten
  - GET /api/orders.php?id=X&action=photos — Anhänge
  - POST /api/orders.php?id=X&action=upload_photo — Foto hochladen,
    Bericht wird automatisch angelegt falls nicht vorhanden
- api/reports.php:
  - GET /api/reports.php?id=X — Bericht-Detail + Seiten
  - POST /api/reports.php?id=X&action=finalize — Status auf final

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
[deploy]
2026-04-08 22:40:52 +02:00

66 lines
2.1 KiB
PHP

<?php
/* Mini JWT Helper für die Bericht-API.
* HS256 only — kein externes Lib nötig.
*
* Secret kommt aus $dolibarr_main_instance_unique_id (siehe conf.php),
* salt'ed mit "bericht-api-v1".
*/
if (!defined('BERICHT_JWT_TTL')) define('BERICHT_JWT_TTL', 7 * 86400); // 7 Tage
function bericht_jwt_secret()
{
global $dolibarr_main_instance_unique_id;
$base = $dolibarr_main_instance_unique_id ?? 'fallback-secret-do-not-use';
return hash('sha256', $base.'|bericht-api-v1');
}
function bericht_b64url_encode($data)
{
return rtrim(strtr(base64_encode($data), '+/', '-_'), '=');
}
function bericht_b64url_decode($data)
{
return base64_decode(strtr($data, '-_', '+/').str_repeat('=', (4 - strlen($data) % 4) % 4));
}
function bericht_jwt_encode(array $payload)
{
$header = array('alg' => 'HS256', 'typ' => 'JWT');
$h = bericht_b64url_encode(json_encode($header));
$p = bericht_b64url_encode(json_encode($payload));
$sig = hash_hmac('sha256', $h.'.'.$p, bericht_jwt_secret(), true);
return $h.'.'.$p.'.'.bericht_b64url_encode($sig);
}
function bericht_jwt_decode($token)
{
$parts = explode('.', $token);
if (count($parts) !== 3) return null;
list($h, $p, $s) = $parts;
$expected = bericht_b64url_encode(hash_hmac('sha256', $h.'.'.$p, bericht_jwt_secret(), true));
if (!hash_equals($expected, $s)) return null;
$payload = json_decode(bericht_b64url_decode($p), true);
if (!is_array($payload)) return null;
if (isset($payload['exp']) && $payload['exp'] < time()) return null;
return $payload;
}
/**
* Liest und validiert das Authorization: Bearer <jwt> Header.
* @return array|null decoded payload
*/
function bericht_jwt_from_request()
{
$hdr = '';
if (isset($_SERVER['HTTP_AUTHORIZATION'])) {
$hdr = $_SERVER['HTTP_AUTHORIZATION'];
} elseif (function_exists('apache_request_headers')) {
$h = apache_request_headers();
if (isset($h['Authorization'])) $hdr = $h['Authorization'];
}
if (!$hdr || stripos($hdr, 'bearer ') !== 0) return null;
$token = trim(substr($hdr, 7));
return bericht_jwt_decode($token);
}