create(array_merge( AiEditingSetting::defaults(), [ 'runware_mode' => 'fake', 'queue_auto_dispatch' => false, ] )); $event = Event::factory()->create(['status' => 'published']); $photo = Photo::factory()->for($event)->create([ 'tenant_id' => $event->tenant_id, 'status' => 'approved', ]); $style = AiStyle::query()->create([ 'key' => 'fake-style', 'name' => 'Fake Style', 'provider' => 'runware', 'provider_model' => 'runware-default', 'requires_source_image' => true, 'is_active' => true, ]); $request = AiEditRequest::query()->create([ 'tenant_id' => $event->tenant_id, 'event_id' => $event->id, 'photo_id' => $photo->id, 'style_id' => $style->id, 'provider' => 'runware', 'provider_model' => 'runware-default', 'status' => AiEditRequest::STATUS_QUEUED, 'safety_state' => 'pending', 'prompt' => 'Transform image style.', 'idempotency_key' => 'job-fake-1', 'queued_at' => now(), ]); ProcessAiEditRequest::dispatchSync($request->id); $request->refresh(); $this->assertSame(AiEditRequest::STATUS_SUCCEEDED, $request->status); $this->assertNotNull($request->started_at); $this->assertNotNull($request->completed_at); $this->assertSame(1, $request->outputs()->count()); $this->assertSame(1, $request->providerRuns()->count()); $this->assertSame('succeeded', $request->providerRuns()->first()?->status); $this->assertSame(1, $request->usageLedgers()->count()); $this->assertSame(AiUsageLedger::TYPE_DEBIT, $request->usageLedgers()->first()?->entry_type); $this->assertSame('unentitled', $request->usageLedgers()->first()?->package_context); ProcessAiEditRequest::dispatchSync($request->id); $request->refresh(); $this->assertSame(1, $request->usageLedgers()->count()); } public function test_it_marks_request_failed_when_runware_is_not_configured(): void { config([ 'services.runware.api_key' => null, ]); AiEditingSetting::query()->create(array_merge( AiEditingSetting::defaults(), [ 'runware_mode' => 'live', 'queue_auto_dispatch' => false, ] )); $event = Event::factory()->create(['status' => 'published']); $photo = Photo::factory()->for($event)->create([ 'tenant_id' => $event->tenant_id, 'status' => 'approved', ]); $style = AiStyle::query()->create([ 'key' => 'live-style', 'name' => 'Live Style', 'provider' => 'runware', 'provider_model' => 'runware-default', 'requires_source_image' => true, 'is_active' => true, ]); $request = AiEditRequest::query()->create([ 'tenant_id' => $event->tenant_id, 'event_id' => $event->id, 'photo_id' => $photo->id, 'style_id' => $style->id, 'provider' => 'runware', 'provider_model' => 'runware-default', 'status' => AiEditRequest::STATUS_QUEUED, 'safety_state' => 'pending', 'prompt' => 'Transform image style.', 'idempotency_key' => 'job-live-1', 'queued_at' => now(), ]); ProcessAiEditRequest::dispatchSync($request->id); $request->refresh(); $this->assertSame(AiEditRequest::STATUS_FAILED, $request->status); $this->assertSame('provider_not_configured', $request->failure_code); $this->assertNotNull($request->completed_at); $this->assertSame(0, $request->outputs()->count()); $this->assertSame(1, $request->providerRuns()->count()); $this->assertSame('failed', $request->providerRuns()->first()?->status); } public function test_it_blocks_request_when_provider_flags_output_as_unsafe(): void { AiEditingSetting::query()->create(array_merge( AiEditingSetting::defaults(), [ 'runware_mode' => 'fake', 'queue_auto_dispatch' => false, ] )); $event = Event::factory()->create(['status' => 'published']); $photo = Photo::factory()->for($event)->create([ 'tenant_id' => $event->tenant_id, 'status' => 'approved', ]); $style = AiStyle::query()->create([ 'key' => 'unsafe-style', 'name' => 'Unsafe Style', 'provider' => 'runware', 'provider_model' => 'runware-default', 'requires_source_image' => true, 'is_active' => true, ]); $request = AiEditRequest::query()->create([ 'tenant_id' => $event->tenant_id, 'event_id' => $event->id, 'photo_id' => $photo->id, 'style_id' => $style->id, 'provider' => 'runware', 'provider_model' => 'runware-default', 'status' => AiEditRequest::STATUS_QUEUED, 'safety_state' => 'pending', 'prompt' => 'Transform image style.', 'idempotency_key' => 'job-fake-unsafe-1', 'queued_at' => now(), 'metadata' => ['fake_nsfw' => true], ]); ProcessAiEditRequest::dispatchSync($request->id); $request->refresh(); $this->assertSame(AiEditRequest::STATUS_BLOCKED, $request->status); $this->assertSame('blocked', $request->safety_state); $this->assertSame('output_policy_blocked', $request->failure_code); $this->assertSame(['provider_nsfw_content'], $request->safety_reasons); $this->assertNotNull($request->completed_at); $this->assertSame(0, $request->outputs()->count()); $this->assertSame(1, $request->providerRuns()->count()); $this->assertIsArray($request->metadata); $this->assertSame('output_block', $request->metadata['abuse']['type'] ?? null); } public function test_it_marks_request_failed_when_provider_returns_processing_without_task_id(): void { AiEditingSetting::query()->create(array_merge( AiEditingSetting::defaults(), [ 'runware_mode' => 'live', 'queue_auto_dispatch' => false, ] )); $event = Event::factory()->create(['status' => 'published']); $photo = Photo::factory()->for($event)->create([ 'tenant_id' => $event->tenant_id, 'status' => 'approved', ]); $style = AiStyle::query()->create([ 'key' => 'processing-no-task', 'name' => 'Processing Without Task ID', 'provider' => 'runware', 'provider_model' => 'runware-default', 'requires_source_image' => true, 'is_active' => true, ]); $request = AiEditRequest::query()->create([ 'tenant_id' => $event->tenant_id, 'event_id' => $event->id, 'photo_id' => $photo->id, 'style_id' => $style->id, 'provider' => 'runware', 'provider_model' => 'runware-default', 'status' => AiEditRequest::STATUS_QUEUED, 'safety_state' => 'pending', 'prompt' => 'Transform image style.', 'idempotency_key' => 'job-processing-no-task', 'queued_at' => now(), ]); $provider = \Mockery::mock(RunwareAiImageProvider::class); $provider->shouldReceive('submit') ->once() ->andReturn(new AiProviderResult(status: 'processing', providerTaskId: '')); $this->app->instance(RunwareAiImageProvider::class, $provider); ProcessAiEditRequest::dispatchSync($request->id); $request->refresh(); $run = $request->providerRuns()->first(); $this->assertSame(AiEditRequest::STATUS_FAILED, $request->status); $this->assertSame('provider_task_id_missing', $request->failure_code); $this->assertNotNull($request->completed_at); $this->assertNotNull($run); $this->assertSame('failed', $run?->status); $this->assertSame('Provider returned processing state without task identifier.', $run?->error_message); } public function test_process_job_failed_hook_marks_request_as_failed(): void { $event = Event::factory()->create(['status' => 'published']); $photo = Photo::factory()->for($event)->create([ 'tenant_id' => $event->tenant_id, 'status' => 'approved', ]); $style = AiStyle::query()->create([ 'key' => 'failed-hook-style', 'name' => 'Failed Hook Style', 'provider' => 'runware', 'provider_model' => 'runware-default', 'requires_source_image' => true, 'is_active' => true, ]); $request = AiEditRequest::query()->create([ 'tenant_id' => $event->tenant_id, 'event_id' => $event->id, 'photo_id' => $photo->id, 'style_id' => $style->id, 'provider' => 'runware', 'provider_model' => 'runware-default', 'status' => AiEditRequest::STATUS_PROCESSING, 'safety_state' => 'pending', 'prompt' => 'Transform image style.', 'idempotency_key' => 'job-failed-hook-1', 'queued_at' => now()->subMinute(), 'started_at' => now()->subSeconds(30), ]); $job = new ProcessAiEditRequest($request->id); $job->failed(new RuntimeException('Queue worker timeout')); $request->refresh(); $this->assertSame(AiEditRequest::STATUS_FAILED, $request->status); $this->assertSame('queue_job_failed', $request->failure_code); $this->assertSame('Queue worker timeout', $request->failure_message); $this->assertNotNull($request->completed_at); } }