14758136b6
Milestone 5 (media): - POST /api/v1/media — multipart upload, attaches to feature/issue/project/ phase/layer, idempotent by uuid, authz member + 'upload media'. Added uuid+client_updated_at to media. - Bundle now includes a 'media' array (URLs) for the project's project/feature/ issue attachments (delta-aware). Milestone 6 (hardening + docs): - sync_logs table/model: every applied op is logged; /sync short-circuits on a repeated op uuid -> 'duplicate' (true idempotency for updates too, not just creates). - Rate limiting on login (10/min), sync (60/min), media (120/min). - docs/openapi.yaml: OpenAPI 3 contract for the mobile team. Tests: 18 passing (added media upload idempotency + sync_logs idempotency). The mobile API (Milestones 1-6) is now feature-complete on the webapp side. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
29 lines
1.1 KiB
PHP
29 lines
1.1 KiB
PHP
<?php
|
|
|
|
use App\Http\Controllers\Api\V1\AuthController;
|
|
use App\Http\Controllers\Api\V1\MediaController;
|
|
use App\Http\Controllers\Api\V1\ProjectApiController;
|
|
use App\Http\Controllers\Api\V1\SyncController;
|
|
use Illuminate\Support\Facades\Route;
|
|
|
|
Route::prefix('v1')->group(function () {
|
|
|
|
// Público
|
|
Route::post('login', [AuthController::class, 'login'])->middleware('throttle:10,1');
|
|
|
|
// Protegido: token Sanctum con ability 'mobile-sync'
|
|
Route::middleware(['auth:sanctum', 'ability:mobile-sync'])->group(function () {
|
|
Route::get('me', [AuthController::class, 'me']);
|
|
Route::post('logout', [AuthController::class, 'logout']);
|
|
|
|
// PULL
|
|
Route::get('projects', [ProjectApiController::class, 'index']);
|
|
Route::get('projects/{project}/bundle', [ProjectApiController::class, 'bundle']);
|
|
Route::get('templates', [ProjectApiController::class, 'templates']);
|
|
|
|
// PUSH
|
|
Route::post('sync', [SyncController::class, 'sync'])->middleware('throttle:60,1');
|
|
Route::post('media', [MediaController::class, 'upload'])->middleware('throttle:120,1');
|
|
});
|
|
});
|