2020-11-06 13:54:39 +01:00
|
|
|
<?php namespace Tests;
|
|
|
|
|
|
|
|
use BookStack\Entities\Book;
|
|
|
|
use BookStack\Entities\Deletion;
|
|
|
|
use BookStack\Entities\Page;
|
2020-11-07 14:58:23 +01:00
|
|
|
use DB;
|
|
|
|
use Illuminate\Support\Carbon;
|
2020-11-06 13:54:39 +01:00
|
|
|
|
|
|
|
class RecycleBinTest extends TestCase
|
|
|
|
{
|
|
|
|
public function test_recycle_bin_routes_permissions()
|
|
|
|
{
|
|
|
|
$page = Page::query()->first();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
$deletion = Deletion::query()->firstOrFail();
|
|
|
|
|
|
|
|
$routes = [
|
|
|
|
'GET:/settings/recycle-bin',
|
|
|
|
'POST:/settings/recycle-bin/empty',
|
|
|
|
"GET:/settings/recycle-bin/{$deletion->id}/destroy",
|
|
|
|
"GET:/settings/recycle-bin/{$deletion->id}/restore",
|
|
|
|
"POST:/settings/recycle-bin/{$deletion->id}/restore",
|
|
|
|
"DELETE:/settings/recycle-bin/{$deletion->id}",
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach($routes as $route) {
|
|
|
|
[$method, $url] = explode(':', $route);
|
|
|
|
$resp = $this->call($method, $url);
|
|
|
|
$this->assertPermissionError($resp);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->giveUserPermissions($editor, ['restrictions-manage-all']);
|
|
|
|
|
|
|
|
foreach($routes as $route) {
|
|
|
|
[$method, $url] = explode(':', $route);
|
|
|
|
$resp = $this->call($method, $url);
|
|
|
|
$this->assertPermissionError($resp);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->giveUserPermissions($editor, ['settings-manage']);
|
|
|
|
|
|
|
|
foreach($routes as $route) {
|
2020-11-07 14:58:23 +01:00
|
|
|
DB::beginTransaction();
|
2020-11-06 13:54:39 +01:00
|
|
|
[$method, $url] = explode(':', $route);
|
|
|
|
$resp = $this->call($method, $url);
|
|
|
|
$this->assertNotPermissionError($resp);
|
2020-11-07 14:58:23 +01:00
|
|
|
DB::rollBack();
|
2020-11-06 13:54:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_recycle_bin_view()
|
|
|
|
{
|
|
|
|
$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());
|
|
|
|
|
|
|
|
$viewReq = $this->asAdmin()->get('/settings/recycle-bin');
|
|
|
|
$viewReq->assertElementContains('table.table', $page->name);
|
|
|
|
$viewReq->assertElementContains('table.table', $editor->name);
|
|
|
|
$viewReq->assertElementContains('table.table', $book->name);
|
|
|
|
$viewReq->assertElementContains('table.table', $book->pages_count . ' Pages');
|
|
|
|
$viewReq->assertElementContains('table.table', $book->chapters_count . ' Chapters');
|
|
|
|
}
|
2020-11-07 14:19:23 +01:00
|
|
|
|
|
|
|
public function test_recycle_bin_empty()
|
|
|
|
{
|
|
|
|
$page = Page::query()->first();
|
|
|
|
$book = Book::query()->where('id' , '!=', $page->book_id)->whereHas('pages')->whereHas('chapters')->with(['pages', 'chapters'])->firstOrFail();
|
|
|
|
$editor = $this->getEditor();
|
|
|
|
$this->actingAs($editor)->delete($page->getUrl());
|
|
|
|
$this->actingAs($editor)->delete($book->getUrl());
|
|
|
|
|
|
|
|
$this->assertTrue(Deletion::query()->count() === 2);
|
|
|
|
$emptyReq = $this->asAdmin()->post('/settings/recycle-bin/empty');
|
|
|
|
$emptyReq->assertRedirect('/settings/recycle-bin');
|
|
|
|
|
|
|
|
$this->assertTrue(Deletion::query()->count() === 0);
|
|
|
|
$this->assertDatabaseMissing('books', ['id' => $book->id]);
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $page->id]);
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $book->pages->first()->id]);
|
|
|
|
$this->assertDatabaseMissing('chapters', ['id' => $book->chapters->first()->id]);
|
|
|
|
|
|
|
|
$itemCount = 2 + $book->pages->count() + $book->chapters->count();
|
|
|
|
$redirectReq = $this->get('/settings/recycle-bin');
|
|
|
|
$redirectReq->assertNotificationContains('Deleted '.$itemCount.' total items from the recycle bin');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_entity_restore()
|
|
|
|
{
|
|
|
|
$book = Book::query()->whereHas('pages')->whereHas('chapters')->with(['pages', 'chapters'])->firstOrFail();
|
|
|
|
$this->asEditor()->delete($book->getUrl());
|
|
|
|
$deletion = Deletion::query()->firstOrFail();
|
|
|
|
|
2020-11-07 14:58:23 +01:00
|
|
|
$this->assertEquals($book->pages->count(), DB::table('pages')->where('book_id', '=', $book->id)->whereNotNull('deleted_at')->count());
|
|
|
|
$this->assertEquals($book->chapters->count(), DB::table('chapters')->where('book_id', '=', $book->id)->whereNotNull('deleted_at')->count());
|
2020-11-07 14:19:23 +01:00
|
|
|
|
|
|
|
$restoreReq = $this->asAdmin()->post("/settings/recycle-bin/{$deletion->id}/restore");
|
|
|
|
$restoreReq->assertRedirect('/settings/recycle-bin');
|
|
|
|
$this->assertTrue(Deletion::query()->count() === 0);
|
|
|
|
|
2020-11-07 14:58:23 +01:00
|
|
|
$this->assertEquals($book->pages->count(), DB::table('pages')->where('book_id', '=', $book->id)->whereNull('deleted_at')->count());
|
|
|
|
$this->assertEquals($book->chapters->count(), DB::table('chapters')->where('book_id', '=', $book->id)->whereNull('deleted_at')->count());
|
2020-11-07 14:19:23 +01:00
|
|
|
|
|
|
|
$itemCount = 1 + $book->pages->count() + $book->chapters->count();
|
|
|
|
$redirectReq = $this->get('/settings/recycle-bin');
|
|
|
|
$redirectReq->assertNotificationContains('Restored '.$itemCount.' total items from the recycle bin');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_permanent_delete()
|
|
|
|
{
|
|
|
|
$book = Book::query()->whereHas('pages')->whereHas('chapters')->with(['pages', 'chapters'])->firstOrFail();
|
|
|
|
$this->asEditor()->delete($book->getUrl());
|
|
|
|
$deletion = Deletion::query()->firstOrFail();
|
|
|
|
|
|
|
|
$deleteReq = $this->asAdmin()->delete("/settings/recycle-bin/{$deletion->id}");
|
|
|
|
$deleteReq->assertRedirect('/settings/recycle-bin');
|
|
|
|
$this->assertTrue(Deletion::query()->count() === 0);
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('books', ['id' => $book->id]);
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $book->pages->first()->id]);
|
|
|
|
$this->assertDatabaseMissing('chapters', ['id' => $book->chapters->first()->id]);
|
|
|
|
|
|
|
|
$itemCount = 1 + $book->pages->count() + $book->chapters->count();
|
|
|
|
$redirectReq = $this->get('/settings/recycle-bin');
|
|
|
|
$redirectReq->assertNotificationContains('Deleted '.$itemCount.' total items from the recycle bin');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_permanent_entity_delete_updates_existing_activity_with_entity_name()
|
|
|
|
{
|
|
|
|
$page = Page::query()->firstOrFail();
|
|
|
|
$this->asEditor()->delete($page->getUrl());
|
|
|
|
$deletion = $page->deletions()->firstOrFail();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('activities', [
|
|
|
|
'key' => 'page_delete',
|
|
|
|
'entity_id' => $page->id,
|
|
|
|
'entity_type' => $page->getMorphClass(),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->asAdmin()->delete("/settings/recycle-bin/{$deletion->id}");
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('activities', [
|
|
|
|
'key' => 'page_delete',
|
|
|
|
'entity_id' => $page->id,
|
|
|
|
'entity_type' => $page->getMorphClass(),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('activities', [
|
|
|
|
'key' => 'page_delete',
|
|
|
|
'entity_id' => 0,
|
|
|
|
'entity_type' => '',
|
|
|
|
'extra' => $page->name,
|
|
|
|
]);
|
|
|
|
}
|
2020-11-07 14:58:23 +01:00
|
|
|
|
|
|
|
public function test_auto_clear_functionality_works()
|
|
|
|
{
|
|
|
|
config()->set('app.recycle_bin_lifetime', 5);
|
|
|
|
$page = Page::query()->firstOrFail();
|
|
|
|
$otherPage = Page::query()->where('id', '!=', $page->id)->firstOrFail();
|
|
|
|
|
|
|
|
$this->asEditor()->delete($page->getUrl());
|
|
|
|
$this->assertDatabaseHas('pages', ['id' => $page->id]);
|
|
|
|
$this->assertEquals(1, Deletion::query()->count());
|
|
|
|
|
|
|
|
Carbon::setTestNow(Carbon::now()->addDays(6));
|
|
|
|
$this->asEditor()->delete($otherPage->getUrl());
|
|
|
|
$this->assertEquals(1, Deletion::query()->count());
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $page->id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_auto_clear_functionality_with_negative_time_keeps_forever()
|
|
|
|
{
|
|
|
|
config()->set('app.recycle_bin_lifetime', -1);
|
|
|
|
$page = Page::query()->firstOrFail();
|
|
|
|
$otherPage = Page::query()->where('id', '!=', $page->id)->firstOrFail();
|
|
|
|
|
|
|
|
$this->asEditor()->delete($page->getUrl());
|
|
|
|
$this->assertEquals(1, Deletion::query()->count());
|
|
|
|
|
|
|
|
Carbon::setTestNow(Carbon::now()->addDays(6000));
|
|
|
|
$this->asEditor()->delete($otherPage->getUrl());
|
|
|
|
$this->assertEquals(2, Deletion::query()->count());
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('pages', ['id' => $page->id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_auto_clear_functionality_with_zero_time_deletes_instantly()
|
|
|
|
{
|
|
|
|
config()->set('app.recycle_bin_lifetime', 0);
|
|
|
|
$page = Page::query()->firstOrFail();
|
|
|
|
|
|
|
|
$this->asEditor()->delete($page->getUrl());
|
|
|
|
$this->assertDatabaseMissing('pages', ['id' => $page->id]);
|
|
|
|
$this->assertEquals(0, Deletion::query()->count());
|
|
|
|
}
|
2020-11-06 13:54:39 +01:00
|
|
|
}
|