5 use BookStack\Entities\Models\Page;
6 use BookStack\Exports\Import;
7 use Tests\Api\TestsApi;
8 use Tests\Exports\ZipTestHelper;
11 class ImportsApiTest extends TestCase
15 protected string $baseEndpoint = '/api/imports';
17 public function test_upload_and_run(): void
19 $book = $this->entities->book();
20 $zip = ZipTestHelper::zipUploadFromData([
22 'name' => 'My API import page',
25 'name' => 'My api tag',
26 'value' => 'api test value'
32 $resp = $this->actingAsApiAdmin()->call('POST', $this->baseEndpoint, [], [], ['file' => $zip]);
33 $resp->assertStatus(200);
35 $importId = $resp->json('id');
36 $import = Import::query()->findOrFail($importId);
37 $this->assertEquals('page', $import->type);
39 $resp = $this->post($this->baseEndpoint . "/{$import->id}", [
40 'parent_type' => 'book',
41 'parent_id' => $book->id,
44 'name' => 'My API import page',
45 'book_id' => $book->id,
48 $page = Page::query()->where('name', '=', 'My API import page')->first();
49 $this->assertEquals('My api tag', $page->tags()->first()->name);
52 public function test_upload_validation_error(): void
54 $zip = ZipTestHelper::zipUploadFromData([
58 'name' => 'My api tag',
59 'value' => 'api test value'
65 $resp = $this->actingAsApiAdmin()->call('POST', $this->baseEndpoint, [], [], ['file' => $zip]);
66 $resp->assertStatus(422);
67 $message = $resp->json('message');
69 $this->assertStringContainsString('ZIP upload failed with the following validation errors:', $message);
70 $this->assertStringContainsString('[page.name] The name field is required.', $message);
73 public function test_list(): void
75 $imports = Import::factory()->count(10)->create();
77 $resp = $this->actingAsApiAdmin()->get($this->baseEndpoint);
78 $resp->assertJsonCount(10, 'data');
79 $resp->assertJsonPath('total', 10);
81 $firstImport = $imports->first();
82 $resp = $this->actingAsApiAdmin()->get($this->baseEndpoint . '?filter[id]=' . $firstImport->id);
83 $resp->assertJsonCount(1, 'data');
84 $resp->assertJsonPath('data.0.id', $firstImport->id);
85 $resp->assertJsonPath('data.0.name', $firstImport->name);
86 $resp->assertJsonPath('data.0.size', $firstImport->size);
87 $resp->assertJsonPath('data.0.type', $firstImport->type);
90 public function test_list_visibility_limited(): void
92 $user = $this->users->editor();
93 $admin = $this->users->admin();
94 $userImport = Import::factory()->create(['name' => 'MySuperUserImport', 'created_by' => $user->id]);
95 $adminImport = Import::factory()->create(['name' => 'MySuperAdminImport', 'created_by' => $admin->id]);
96 $this->permissions->grantUserRolePermissions($user, ['content-import']);
98 $resp = $this->actingAsForApi($user)->get($this->baseEndpoint);
99 $resp->assertJsonCount(1, 'data');
100 $resp->assertJsonPath('data.0.name', 'MySuperUserImport');
102 $this->permissions->grantUserRolePermissions($user, ['settings-manage']);
104 $resp = $this->actingAsForApi($user)->get($this->baseEndpoint);
105 $resp->assertJsonCount(2, 'data');
106 $resp->assertJsonPath('data.1.name', 'MySuperAdminImport');
109 public function test_read(): void
111 $zip = ZipTestHelper::zipUploadFromData([
113 'name' => 'My API import book',
116 'name' => 'My import page',
119 'name' => 'My api tag',
120 'value' => 'api test value'
128 $resp = $this->actingAsApiAdmin()->call('POST', $this->baseEndpoint, [], [], ['file' => $zip]);
129 $resp->assertStatus(200);
131 $resp = $this->get($this->baseEndpoint . "/{$resp->json('id')}");
132 $resp->assertStatus(200);
134 $resp->assertJsonPath('details.name', 'My API import book');
135 $resp->assertJsonPath('details.pages.0.name', 'My import page');
136 $resp->assertJsonPath('details.pages.0.tags.0.name', 'My api tag');
137 $resp->assertJsonMissingPath('metadata');
140 public function test_delete(): void
142 $import = Import::factory()->create();
144 $resp = $this->actingAsApiAdmin()->delete($this->baseEndpoint . "/{$import->id}");
145 $resp->assertStatus(204);
148 public function test_content_import_permissions_needed(): void
150 $user = $this->users->viewer();
151 $this->permissions->grantUserRolePermissions($user, ['access-api']);
152 $this->actingAsForApi($user);
154 ['GET', $this->baseEndpoint],
155 ['POST', $this->baseEndpoint],
156 ['GET', $this->baseEndpoint . "/1"],
157 ['POST', $this->baseEndpoint . "/1"],
158 ['DELETE', $this->baseEndpoint . "/1"],
161 foreach ($requests as $request) {
162 [$method, $endpoint] = $request;
163 $resp = $this->json($method, $endpoint);
164 $resp->assertStatus(403);
167 $this->permissions->grantUserRolePermissions($user, ['content-import']);
169 foreach ($requests as $request) {
170 [$method, $endpoint] = $request;
171 $resp = $this->call($method, $endpoint);
172 $this->assertNotEquals(403, $resp->status(), "A {$method} request to {$endpoint} returned 403");