create(['is_active' => true]); $plainTextToken = $user->createToken('API Client', ['press-releases:read'])->plainTextToken; $tokenId = $user->tokens()->value('id'); $this->withHeader('Authorization', "Bearer {$plainTextToken}") ->getJson('/api/v1/categories') ->assertOk(); $log = ApiUsageLog::query()->firstOrFail(); expect($log->user_id)->toBe($user->id) ->and($log->personal_access_token_id)->toBe($tokenId) ->and($log->method)->toBe('GET') ->and($log->path)->toBe('/api/v1/categories') ->and($log->status_code)->toBe(200) ->and(json_encode($log->toArray()))->not->toContain($plainTextToken); }); test('legacy api key rejections are logged', function () { /** @var TestCase $this */ $this->getJson('/api/v1/categories?api_key=legacy-secret') ->assertStatus(410); $log = ApiUsageLog::query()->firstOrFail(); expect($log->user_id)->toBeNull() ->and($log->personal_access_token_id)->toBeNull() ->and($log->path)->toBe('/api/v1/categories') ->and($log->status_code)->toBe(410) ->and(json_encode($log->toArray()))->not->toContain('legacy-secret'); }); test('inactive users cannot use existing api tokens', function () { /** @var TestCase $this */ $user = User::factory()->create(['is_active' => false]); $plainTextToken = $user->createToken('Old API Client', ['press-releases:read'])->plainTextToken; $this->withHeader('Authorization', "Bearer {$plainTextToken}") ->getJson('/api/v1/categories') ->assertForbidden() ->assertJsonPath('message', 'API access is disabled for inactive users.'); expect(ApiUsageLog::query()->value('status_code'))->toBe(403); }); test('api rate limit is scoped to each personal access token', function () { /** @var TestCase $this */ $user = User::factory()->create(['is_active' => true]); $firstToken = $user->createToken('Busy Client', ['press-releases:read'])->plainTextToken; $secondToken = $user->createToken('Other Client', ['press-releases:read'])->plainTextToken; expect(strtok($firstToken, '|'))->not->toBe(strtok($secondToken, '|')); for ($request = 1; $request <= 60; $request++) { $this->withHeader('Authorization', "Bearer {$firstToken}") ->getJson('/api/v1/categories') ->assertOk(); } $this->withHeader('Authorization', "Bearer {$firstToken}") ->getJson('/api/v1/categories') ->assertStatus(429) ->assertJsonPath('message', 'API rate limit exceeded.'); $this->flushHeaders(); $this->withHeader('Authorization', "Bearer {$secondToken}") ->getJson('/api/v1/categories') ->assertOk(); });