2022-04-06 22:57:18 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Api;
|
|
|
|
|
|
|
|
use BookStack\Entities\Models\Book;
|
|
|
|
use BookStack\Entities\Models\Deletion;
|
|
|
|
use BookStack\Entities\Models\Page;
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
class RecycleBinApiTest extends TestCase
|
|
|
|
{
|
|
|
|
use TestsApi;
|
|
|
|
|
|
|
|
protected string $baseEndpoint = '/api/recycle_bin';
|
|
|
|
|
|
|
|
protected array $endpointMap = [
|
|
|
|
['get', '/api/recycle_bin'],
|
|
|
|
['put', '/api/recycle_bin/1'],
|
|
|
|
['delete', '/api/recycle_bin/1'],
|
|
|
|
];
|
|
|
|
|
|
|
|
public function test_settings_manage_permission_needed_for_all_endpoints()
|
|
|
|
{
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->giveUserPermissions($editor, ['settings-manage']);
|
|
|
|
$this->actingAs($editor);
|
|
|
|
|
|
|
|
foreach ($this->endpointMap as [$method, $uri]) {
|
|
|
|
$resp = $this->json($method, $uri);
|
|
|
|
$resp->assertStatus(403);
|
|
|
|
$resp->assertJson($this->permissionErrorResponse());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-24 10:16:45 +02:00
|
|
|
public function test_restrictions_manage_all_permission_needed_for_all_endpoints()
|
2022-04-06 22:57:18 +02:00
|
|
|
{
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->giveUserPermissions($editor, ['restrictions-manage-all']);
|
|
|
|
$this->actingAs($editor);
|
2022-04-24 10:16:45 +02:00
|
|
|
|
2022-04-06 22:57:18 +02:00
|
|
|
foreach ($this->endpointMap as [$method, $uri]) {
|
|
|
|
$resp = $this->json($method, $uri);
|
|
|
|
$resp->assertStatus(403);
|
|
|
|
$resp->assertJson($this->permissionErrorResponse());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_index_endpoint_returns_expected_page()
|
|
|
|
{
|
|
|
|
$this->actingAsAuthorizedUser();
|
2022-04-20 22:58:16 +02:00
|
|
|
|
2022-04-06 22:57:18 +02:00
|
|
|
$page = Page::query()->first();
|
|
|
|
$book = Book::query()->whereHas('pages')->whereHas('chapters')->withCount(['pages', 'chapters'])->first();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
$this->actingAs($editor)->delete($book->getUrl());
|
|
|
|
|
|
|
|
$deletions = Deletion::query()->orderBy('id')->get();
|
|
|
|
|
|
|
|
$resp = $this->getJson($this->baseEndpoint);
|
|
|
|
|
|
|
|
$expectedData = $deletions
|
|
|
|
->zip([$page, $book])
|
|
|
|
->map(function (Collection $data) use ($editor) {
|
|
|
|
return [
|
|
|
|
'id' => $data[0]->id,
|
|
|
|
'deleted_by' => $editor->getKey(),
|
|
|
|
'created_at' => $data[0]->created_at->toJson(),
|
|
|
|
'updated_at' => $data[0]->updated_at->toJson(),
|
|
|
|
'deletable_type' => $data[1]->getMorphClass(),
|
2022-04-20 22:58:16 +02:00
|
|
|
'deletable_id' => $data[1]->getKey(),
|
2022-04-06 22:57:18 +02:00
|
|
|
];
|
|
|
|
});
|
|
|
|
|
|
|
|
$resp->assertJson([
|
2022-04-24 10:16:45 +02:00
|
|
|
'data' => $expectedData->values()->all(),
|
2022-04-20 22:58:16 +02:00
|
|
|
'total' => 2,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_index_endpoint_returns_children()
|
|
|
|
{
|
|
|
|
$this->actingAsAuthorizedUser();
|
2022-04-24 10:16:45 +02:00
|
|
|
|
2022-04-20 22:58:16 +02:00
|
|
|
$book = Book::query()->whereHas('pages')->whereHas('chapters')->withCount(['pages', 'chapters'])->first();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($book->getUrl());
|
|
|
|
|
|
|
|
$deletion = Deletion::query()->orderBy('id')->first();
|
|
|
|
|
|
|
|
$resp = $this->getJson($this->baseEndpoint);
|
|
|
|
|
|
|
|
$expectedData = [
|
|
|
|
[
|
|
|
|
'id' => $deletion->getKey(),
|
|
|
|
'deleted_by' => $editor->getKey(),
|
|
|
|
'created_at' => $deletion->created_at->toJson(),
|
|
|
|
'updated_at' => $deletion->updated_at->toJson(),
|
|
|
|
'deletable_type' => $book->getMorphClass(),
|
|
|
|
'deletable_id' => $book->getKey(),
|
|
|
|
'children' => [
|
2022-04-24 10:16:45 +02:00
|
|
|
'BookStack\Page' => $book->pages_count,
|
|
|
|
'BookStack\Chapter' => $book->chapters_count,
|
2022-04-20 22:58:16 +02:00
|
|
|
],
|
|
|
|
'parent' => null,
|
2022-04-24 10:16:45 +02:00
|
|
|
],
|
2022-04-20 22:58:16 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
$resp->assertJson([
|
2022-04-24 10:16:45 +02:00
|
|
|
'data' => $expectedData,
|
2022-04-20 22:58:16 +02:00
|
|
|
'total' => 1,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_index_endpoint_returns_parent()
|
|
|
|
{
|
|
|
|
$this->actingAsAuthorizedUser();
|
|
|
|
|
|
|
|
$page = Page::query()->whereHas('chapter')->with('chapter')->first();
|
|
|
|
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
|
|
|
|
$deletion = Deletion::query()->orderBy('id')->first();
|
|
|
|
|
|
|
|
$resp = $this->getJson($this->baseEndpoint);
|
|
|
|
|
|
|
|
$expectedData = [
|
|
|
|
[
|
|
|
|
'id' => $deletion->getKey(),
|
|
|
|
'deleted_by' => $editor->getKey(),
|
|
|
|
'created_at' => $deletion->created_at->toJson(),
|
|
|
|
'updated_at' => $deletion->updated_at->toJson(),
|
|
|
|
'deletable_type' => $page->getMorphClass(),
|
|
|
|
'deletable_id' => $page->getKey(),
|
|
|
|
'parent' => [
|
|
|
|
'type' => 'BookStack\Chapter',
|
2022-04-24 10:16:45 +02:00
|
|
|
'id' => $page->chapter->getKey(),
|
2022-04-20 22:58:16 +02:00
|
|
|
],
|
|
|
|
'children' => null,
|
2022-04-24 10:16:45 +02:00
|
|
|
],
|
2022-04-20 22:58:16 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
$resp->assertJson([
|
2022-04-24 10:16:45 +02:00
|
|
|
'data' => $expectedData,
|
|
|
|
'total' => 1,
|
2022-04-06 22:57:18 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_restore_endpoint()
|
|
|
|
{
|
|
|
|
$this->actingAsAuthorizedUser();
|
2022-04-24 10:16:45 +02:00
|
|
|
|
2022-04-06 22:57:18 +02:00
|
|
|
$page = Page::query()->first();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
$page->refresh();
|
|
|
|
|
|
|
|
$deletion = Deletion::query()->orderBy('id')->first();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('pages', [
|
2022-04-24 10:16:45 +02:00
|
|
|
'id' => $page->getKey(),
|
|
|
|
'deleted_at' => $page->deleted_at,
|
2022-04-06 22:57:18 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
$this->putJson($this->baseEndpoint . '/' . $deletion->getKey());
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('pages', [
|
2022-04-24 10:16:45 +02:00
|
|
|
'id' => $page->getKey(),
|
|
|
|
'deleted_at' => null,
|
2022-04-06 22:57:18 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_destroy_endpoint()
|
|
|
|
{
|
|
|
|
$this->actingAsAuthorizedUser();
|
2022-04-24 10:16:45 +02:00
|
|
|
|
2022-04-06 22:57:18 +02:00
|
|
|
$page = Page::query()->first();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
$page->refresh();
|
|
|
|
|
|
|
|
$deletion = Deletion::query()->orderBy('id')->first();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('pages', [
|
2022-04-24 10:16:45 +02:00
|
|
|
'id' => $page->getKey(),
|
|
|
|
'deleted_at' => $page->deleted_at,
|
2022-04-06 22:57:18 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
$this->deleteJson($this->baseEndpoint . '/' . $deletion->getKey());
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $page->getKey()]);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function actingAsAuthorizedUser()
|
|
|
|
{
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->giveUserPermissions($editor, ['restrictions-manage-all']);
|
|
|
|
$this->giveUserPermissions($editor, ['settings-manage']);
|
|
|
|
$this->actingAs($editor);
|
|
|
|
}
|
|
|
|
}
|