feat: add Image Webhook plugin

This commit is contained in:
Benjamin Nussbaum 2026-01-05 18:09:39 +01:00
parent 809965e81c
commit 3def60ae3e
11 changed files with 817 additions and 7 deletions

View file

@ -0,0 +1,196 @@
<?php
use App\Models\Plugin;
use App\Models\User;
use Illuminate\Http\UploadedFile;
use Illuminate\Support\Facades\Storage;
use Illuminate\Support\Str;
uses(Illuminate\Foundation\Testing\RefreshDatabase::class);
beforeEach(function (): void {
Storage::fake('public');
Storage::disk('public')->makeDirectory('/images/generated');
});
test('can upload image to image webhook plugin via multipart form', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
$image = UploadedFile::fake()->image('test.png', 800, 480);
$response = $this->post("/api/plugin_settings/{$plugin->uuid}/image", [
'image' => $image,
]);
$response->assertOk()
->assertJsonStructure([
'message',
'image_url',
]);
$plugin->refresh();
expect($plugin->current_image)
->not->toBeNull()
->not->toBe($plugin->uuid); // Should be a new UUID, not the plugin's UUID
// File should exist with the new UUID
Storage::disk('public')->assertExists("images/generated/{$plugin->current_image}.png");
// Image URL should contain the new UUID
expect($response->json('image_url'))
->toContain($plugin->current_image);
});
test('can upload image to image webhook plugin via raw binary', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
// Create a simple PNG image binary
$pngData = base64_decode('iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNk+M9QDwADhgGAWjR9awAAAABJRU5ErkJggg==');
$response = $this->call('POST', "/api/plugin_settings/{$plugin->uuid}/image", [], [], [], [
'CONTENT_TYPE' => 'image/png',
], $pngData);
$response->assertOk()
->assertJsonStructure([
'message',
'image_url',
]);
$plugin->refresh();
expect($plugin->current_image)
->not->toBeNull()
->not->toBe($plugin->uuid); // Should be a new UUID, not the plugin's UUID
// File should exist with the new UUID
Storage::disk('public')->assertExists("images/generated/{$plugin->current_image}.png");
// Image URL should contain the new UUID
expect($response->json('image_url'))
->toContain($plugin->current_image);
});
test('can upload image to image webhook plugin via base64 data URI', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
// Create a simple PNG image as base64 data URI
$base64Image = '';
$response = $this->postJson("/api/plugin_settings/{$plugin->uuid}/image", [
'image' => $base64Image,
]);
$response->assertOk()
->assertJsonStructure([
'message',
'image_url',
]);
$plugin->refresh();
expect($plugin->current_image)
->not->toBeNull()
->not->toBe($plugin->uuid); // Should be a new UUID, not the plugin's UUID
// File should exist with the new UUID
Storage::disk('public')->assertExists("images/generated/{$plugin->current_image}.png");
// Image URL should contain the new UUID
expect($response->json('image_url'))
->toContain($plugin->current_image);
});
test('returns 400 for non-image-webhook plugin', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->create([
'user_id' => $user->id,
'plugin_type' => 'recipe',
]);
$image = UploadedFile::fake()->image('test.png', 800, 480);
$response = $this->post("/api/plugin_settings/{$plugin->uuid}/image", [
'image' => $image,
]);
$response->assertStatus(400)
->assertJson(['error' => 'Plugin is not an image webhook plugin']);
});
test('returns 404 for non-existent plugin', function (): void {
$image = UploadedFile::fake()->image('test.png', 800, 480);
$response = $this->post('/api/plugin_settings/'.Str::uuid().'/image', [
'image' => $image,
]);
$response->assertNotFound();
});
test('returns 400 for unsupported image format', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
// Create a fake GIF file (not supported)
$gifData = base64_decode('R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7');
$response = $this->call('POST', "/api/plugin_settings/{$plugin->uuid}/image", [], [], [], [
'CONTENT_TYPE' => 'image/gif',
], $gifData);
$response->assertStatus(400)
->assertJson(['error' => 'Unsupported image format. Expected PNG or BMP.']);
});
test('returns 400 for JPG image format', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
// Create a fake JPG file (not supported)
$jpgData = base64_decode('/9j/4AAQSkZJRgABAQEAYABgAAD/2wBDAAEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQH/2wBDAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQH/wAARCAABAAEDASIAAhEBAxEB/8QAFQABAQAAAAAAAAAAAAAAAAAAAAv/xAAUEAEAAAAAAAAAAAAAAAAAAAAA/8QAFQEBAQAAAAAAAAAAAAAAAAAAAAX/xAAUEQEAAAAAAAAAAAAAAAAAAAAA/9oADAMBAAIRAxEAPwA/8A8A');
$response = $this->call('POST', "/api/plugin_settings/{$plugin->uuid}/image", [], [], [], [
'CONTENT_TYPE' => 'image/jpeg',
], $jpgData);
$response->assertStatus(400)
->assertJson(['error' => 'Unsupported image format. Expected PNG or BMP.']);
});
test('returns 400 when no image data provided', function (): void {
$user = User::factory()->create();
$plugin = Plugin::factory()->imageWebhook()->create([
'user_id' => $user->id,
]);
$response = $this->postJson("/api/plugin_settings/{$plugin->uuid}/image", []);
$response->assertStatus(400)
->assertJson(['error' => 'No image data provided']);
});
test('image webhook plugin isDataStale returns false', function (): void {
$plugin = Plugin::factory()->imageWebhook()->create();
expect($plugin->isDataStale())->toBeFalse();
});
test('image webhook plugin factory creates correct plugin type', function (): void {
$plugin = Plugin::factory()->imageWebhook()->create();
expect($plugin)
->plugin_type->toBe('image_webhook')
->data_strategy->toBe('static');
});

View file

@ -685,11 +685,11 @@ test('plugin render includes utc_offset and time_zone_iana in trmnl.user context
* [Input, Expected Result, Forbidden String]
*/
dataset('xss_vectors', [
'standard_script' => ['Safe <script>alert(1)</script>', 'Safe ', '<script>'],
'standard_script' => ['Safe <script>alert(1)</script>', 'Safe ', '<script>'],
'attribute_event_handlers' => ['<a onmouseover="alert(1)">Link</a>', '<a>Link</a>', 'onmouseover'],
'javascript_protocol' => ['<a href="javascript:alert(1)">Click</a>', '<a>Click</a>', 'javascript:'],
'iframe_injection' => ['Watch <iframe src="https://x.com"></iframe>', 'Watch ', '<iframe>'],
'img_onerror_fallback' => ['Photo <img src=x onerror=alert(1)>', 'Photo <img src="x" alt="x">', 'onerror'],
'javascript_protocol' => ['<a href="javascript:alert(1)">Click</a>', '<a>Click</a>', 'javascript:'],
'iframe_injection' => ['Watch <iframe src="https://x.com"></iframe>', 'Watch ', '<iframe>'],
'img_onerror_fallback' => ['Photo <img src=x onerror=alert(1)>', 'Photo <img src="x" alt="x">', 'onerror'],
]);
test('plugin model sanitizes template fields on save', function (string $input, string $expected, string $forbidden): void {
@ -731,8 +731,8 @@ test('plugin model preserves multi_string csv format', function (): void {
'data_strategy' => 'static',
'polling_verb' => 'get',
'configuration' => [
'tags' => 'laravel,pest,security'
]
'tags' => 'laravel,pest,security',
],
]);
expect($plugin->fresh()->configuration['tags'])->toBe('laravel,pest,security');