2021-06-26 17:23:15 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Entity;
|
2017-04-20 21:58:54 +02:00
|
|
|
|
2022-08-09 14:25:18 +02:00
|
|
|
use BookStack\Actions\ActivityType;
|
2020-11-22 01:17:45 +01:00
|
|
|
use BookStack\Entities\Models\Page;
|
2017-04-20 21:58:54 +02:00
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
class PageRevisionTest extends TestCase
|
|
|
|
{
|
2022-09-03 13:32:21 +02:00
|
|
|
public function test_revision_links_visible_to_viewer()
|
|
|
|
{
|
2022-09-29 18:31:38 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-03 13:32:21 +02:00
|
|
|
|
|
|
|
$html = $this->withHtml($this->asViewer()->get($page->getUrl()));
|
|
|
|
$html->assertLinkExists($page->getUrl('/revisions'));
|
|
|
|
$html->assertElementContains('a', 'Revisions');
|
|
|
|
$html->assertElementContains('a', 'Revision #1');
|
|
|
|
}
|
|
|
|
|
2019-04-20 14:25:16 +02:00
|
|
|
public function test_page_revision_views_viewable()
|
|
|
|
{
|
|
|
|
$this->asEditor();
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page', 'html' => '<p>new content</p>']);
|
2019-04-20 14:25:16 +02:00
|
|
|
$pageRevision = $page->revisions->last();
|
|
|
|
|
2022-09-28 12:10:06 +02:00
|
|
|
$resp = $this->get($page->getUrl() . '/revisions/' . $pageRevision->id);
|
|
|
|
$resp->assertStatus(200);
|
|
|
|
$resp->assertSee('new content');
|
2019-04-20 14:25:16 +02:00
|
|
|
|
2022-09-28 12:10:06 +02:00
|
|
|
$resp = $this->get($page->getUrl() . '/revisions/' . $pageRevision->id . '/changes');
|
|
|
|
$resp->assertStatus(200);
|
|
|
|
$resp->assertSee('new content');
|
2019-04-20 14:25:16 +02:00
|
|
|
}
|
|
|
|
|
2020-05-23 13:28:14 +02:00
|
|
|
public function test_page_revision_preview_shows_content_of_revision()
|
|
|
|
{
|
|
|
|
$this->asEditor();
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page', 'html' => '<p>new revision content</p>']);
|
2020-05-23 13:28:14 +02:00
|
|
|
$pageRevision = $page->revisions->last();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page', 'html' => '<p>Updated content</p>']);
|
2020-05-23 13:28:14 +02:00
|
|
|
|
|
|
|
$revisionView = $this->get($page->getUrl() . '/revisions/' . $pageRevision->id);
|
|
|
|
$revisionView->assertStatus(200);
|
|
|
|
$revisionView->assertSee('new revision content');
|
|
|
|
}
|
|
|
|
|
2019-04-20 14:25:16 +02:00
|
|
|
public function test_page_revision_restore_updates_content()
|
|
|
|
{
|
|
|
|
$this->asEditor();
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page abc123', 'html' => '<p>new contente def456</p>']);
|
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page again', 'html' => '<p>new content</p>']);
|
2021-06-26 17:23:15 +02:00
|
|
|
$page = Page::find($page->id);
|
2019-04-20 14:25:16 +02:00
|
|
|
|
|
|
|
$pageView = $this->get($page->getUrl());
|
|
|
|
$pageView->assertDontSee('abc123');
|
|
|
|
$pageView->assertDontSee('def456');
|
|
|
|
|
|
|
|
$revToRestore = $page->revisions()->where('name', 'like', '%abc123')->first();
|
|
|
|
$restoreReq = $this->put($page->getUrl() . '/revisions/' . $revToRestore->id . '/restore');
|
2021-06-26 17:23:15 +02:00
|
|
|
$page = Page::find($page->id);
|
2019-04-20 14:25:16 +02:00
|
|
|
|
|
|
|
$restoreReq->assertStatus(302);
|
|
|
|
$restoreReq->assertRedirect($page->getUrl());
|
|
|
|
|
|
|
|
$pageView = $this->get($page->getUrl());
|
|
|
|
$pageView->assertSee('abc123');
|
|
|
|
$pageView->assertSee('def456');
|
|
|
|
}
|
2017-04-20 21:58:54 +02:00
|
|
|
|
2021-02-06 14:51:05 +01:00
|
|
|
public function test_page_revision_restore_with_markdown_retains_markdown_content()
|
|
|
|
{
|
|
|
|
$this->asEditor();
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page abc123', 'markdown' => '## New Content def456']);
|
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page again', 'markdown' => '## New Content Updated']);
|
2021-02-06 15:14:38 +01:00
|
|
|
$page = Page::find($page->id);
|
2021-02-06 14:51:05 +01:00
|
|
|
|
|
|
|
$pageView = $this->get($page->getUrl());
|
|
|
|
$pageView->assertDontSee('abc123');
|
|
|
|
$pageView->assertDontSee('def456');
|
|
|
|
|
|
|
|
$revToRestore = $page->revisions()->where('name', 'like', '%abc123')->first();
|
|
|
|
$restoreReq = $this->put($page->getUrl() . '/revisions/' . $revToRestore->id . '/restore');
|
|
|
|
$page = Page::find($page->id);
|
|
|
|
|
|
|
|
$restoreReq->assertStatus(302);
|
|
|
|
$restoreReq->assertRedirect($page->getUrl());
|
|
|
|
|
|
|
|
$pageView = $this->get($page->getUrl());
|
|
|
|
$this->assertDatabaseHas('pages', [
|
2021-06-26 17:23:15 +02:00
|
|
|
'id' => $page->id,
|
2021-02-06 15:14:38 +01:00
|
|
|
'markdown' => '## New Content def456',
|
2021-02-06 14:51:05 +01:00
|
|
|
]);
|
|
|
|
$pageView->assertSee('abc123');
|
|
|
|
$pageView->assertSee('def456');
|
|
|
|
}
|
|
|
|
|
2021-01-02 17:42:05 +01:00
|
|
|
public function test_page_revision_restore_sets_new_revision_with_summary()
|
|
|
|
{
|
|
|
|
$this->asEditor();
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['name' => 'updated page abc123', 'html' => '<p>new contente def456</p>', 'summary' => 'My first update']);
|
|
|
|
$this->createRevisions($page, 1, ['html' => '<p>new content</p>']);
|
2021-01-02 17:42:05 +01:00
|
|
|
$page->refresh();
|
|
|
|
|
|
|
|
$revToRestore = $page->revisions()->where('name', 'like', '%abc123')->first();
|
|
|
|
$this->put($page->getUrl() . '/revisions/' . $revToRestore->id . '/restore');
|
|
|
|
$page->refresh();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('page_revisions', [
|
|
|
|
'page_id' => $page->id,
|
2021-06-26 17:23:15 +02:00
|
|
|
'text' => 'new contente def456',
|
|
|
|
'type' => 'version',
|
2021-01-02 17:42:05 +01:00
|
|
|
'summary' => "Restored from #{$revToRestore->id}; My first update",
|
|
|
|
]);
|
2022-08-09 14:25:18 +02:00
|
|
|
|
|
|
|
$detail = "Revision #{$revToRestore->revision_number} (ID: {$revToRestore->id}) for page ID {$revToRestore->page_id}";
|
|
|
|
$this->assertActivityExists(ActivityType::REVISION_RESTORE, null, $detail);
|
2021-01-02 17:42:05 +01:00
|
|
|
}
|
|
|
|
|
2017-04-20 21:58:54 +02:00
|
|
|
public function test_page_revision_count_increments_on_update()
|
|
|
|
{
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2017-04-20 21:58:54 +02:00
|
|
|
$startCount = $page->revision_count;
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1);
|
2017-04-20 21:58:54 +02:00
|
|
|
|
2021-06-26 17:23:15 +02:00
|
|
|
$this->assertTrue(Page::find($page->id)->revision_count === $startCount + 1);
|
2017-04-20 21:58:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function test_revision_count_shown_in_page_meta()
|
|
|
|
{
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 2);
|
2018-09-16 17:14:09 +02:00
|
|
|
|
2017-04-20 21:58:54 +02:00
|
|
|
$pageView = $this->get($page->getUrl());
|
|
|
|
$pageView->assertSee('Revision #' . $page->revision_count);
|
|
|
|
}
|
|
|
|
|
2021-06-26 17:23:15 +02:00
|
|
|
public function test_revision_deletion()
|
|
|
|
{
|
2022-09-29 18:31:38 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 2);
|
2018-09-15 17:35:51 +02:00
|
|
|
$beforeRevisionCount = $page->revisions->count();
|
|
|
|
|
|
|
|
// Delete the first revision
|
2018-09-16 17:14:09 +02:00
|
|
|
$revision = $page->revisions->get(1);
|
2018-09-15 17:35:51 +02:00
|
|
|
$resp = $this->asEditor()->delete($revision->getUrl('/delete/'));
|
2019-08-25 18:21:25 +02:00
|
|
|
$resp->assertRedirect($page->getUrl('/revisions'));
|
2018-09-15 17:35:51 +02:00
|
|
|
|
2022-03-26 17:44:34 +01:00
|
|
|
$page->refresh();
|
2018-09-15 17:35:51 +02:00
|
|
|
$afterRevisionCount = $page->revisions->count();
|
|
|
|
|
|
|
|
$this->assertTrue($beforeRevisionCount === ($afterRevisionCount + 1));
|
|
|
|
|
2022-08-09 14:25:18 +02:00
|
|
|
$detail = "Revision #{$revision->revision_number} (ID: {$revision->id}) for page ID {$revision->page_id}";
|
|
|
|
$this->assertActivityExists(ActivityType::REVISION_DELETE, null, $detail);
|
|
|
|
|
2018-09-15 17:35:51 +02:00
|
|
|
// Try to delete the latest revision
|
2018-09-15 21:42:36 +02:00
|
|
|
$beforeRevisionCount = $page->revisions->count();
|
2022-03-26 17:44:34 +01:00
|
|
|
$resp = $this->asEditor()->delete($page->currentRevision->getUrl('/delete/'));
|
2019-10-05 13:55:01 +02:00
|
|
|
$resp->assertRedirect($page->getUrl('/revisions'));
|
2018-09-15 21:42:36 +02:00
|
|
|
|
2022-03-26 17:44:34 +01:00
|
|
|
$page->refresh();
|
2018-09-15 21:42:36 +02:00
|
|
|
$afterRevisionCount = $page->revisions->count();
|
|
|
|
$this->assertTrue($beforeRevisionCount === $afterRevisionCount);
|
2018-09-15 17:35:51 +02:00
|
|
|
}
|
2018-09-22 18:30:42 +02:00
|
|
|
|
|
|
|
public function test_revision_limit_enforced()
|
|
|
|
{
|
|
|
|
config()->set('app.revision_limit', 2);
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 12);
|
2018-09-22 18:30:42 +02:00
|
|
|
|
|
|
|
$revisionCount = $page->revisions()->count();
|
|
|
|
$this->assertEquals(2, $revisionCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_false_revision_limit_allows_many_revisions()
|
|
|
|
{
|
|
|
|
config()->set('app.revision_limit', false);
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 12);
|
2018-09-22 18:30:42 +02:00
|
|
|
|
|
|
|
$revisionCount = $page->revisions()->count();
|
|
|
|
$this->assertEquals(12, $revisionCount);
|
|
|
|
}
|
2022-04-23 16:03:58 +02:00
|
|
|
|
|
|
|
public function test_revision_list_shows_editor_type()
|
|
|
|
{
|
2022-09-29 23:11:16 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['html' => 'new page html']);
|
2022-04-23 16:03:58 +02:00
|
|
|
|
2022-09-28 12:10:06 +02:00
|
|
|
$resp = $this->asAdmin()->get($page->refresh()->getUrl('/revisions'));
|
2022-11-01 15:53:36 +01:00
|
|
|
$this->withHtml($resp)->assertElementContains('.item-list-row > div:nth-child(2)', 'WYSIWYG)');
|
|
|
|
$this->withHtml($resp)->assertElementNotContains('.item-list-row > div:nth-child(2)', 'Markdown)');
|
2022-04-23 16:03:58 +02:00
|
|
|
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 1, ['markdown' => '# Some markdown content']);
|
2022-04-23 16:03:58 +02:00
|
|
|
$resp = $this->get($page->refresh()->getUrl('/revisions'));
|
2022-11-01 15:53:36 +01:00
|
|
|
$this->withHtml($resp)->assertElementContains('.item-list-row > div:nth-child(2)', 'Markdown)');
|
2022-04-23 16:03:58 +02:00
|
|
|
}
|
2022-09-28 12:10:06 +02:00
|
|
|
|
|
|
|
public function test_revision_restore_action_only_visible_with_permission()
|
|
|
|
{
|
2022-09-29 18:31:38 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 2);
|
|
|
|
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
$this->actingAs($viewer);
|
|
|
|
$respHtml = $this->withHtml($this->get($page->getUrl('/revisions')));
|
|
|
|
$respHtml->assertElementNotContains('.actions a', 'Restore');
|
|
|
|
$respHtml->assertElementNotExists('form[action$="/restore"]');
|
|
|
|
|
|
|
|
$this->giveUserPermissions($viewer, ['page-update-all']);
|
|
|
|
|
|
|
|
$respHtml = $this->withHtml($this->get($page->getUrl('/revisions')));
|
|
|
|
$respHtml->assertElementContains('.actions a', 'Restore');
|
|
|
|
$respHtml->assertElementExists('form[action$="/restore"]');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_revision_delete_action_only_visible_with_permission()
|
|
|
|
{
|
2022-09-29 18:31:38 +02:00
|
|
|
$page = $this->entities->page();
|
2022-09-28 12:10:06 +02:00
|
|
|
$this->createRevisions($page, 2);
|
|
|
|
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
$this->actingAs($viewer);
|
|
|
|
$respHtml = $this->withHtml($this->get($page->getUrl('/revisions')));
|
|
|
|
$respHtml->assertElementNotContains('.actions a', 'Delete');
|
|
|
|
$respHtml->assertElementNotExists('form[action$="/delete"]');
|
|
|
|
|
|
|
|
$this->giveUserPermissions($viewer, ['page-delete-all']);
|
|
|
|
|
|
|
|
$respHtml = $this->withHtml($this->get($page->getUrl('/revisions')));
|
|
|
|
$respHtml->assertElementContains('.actions a', 'Delete');
|
|
|
|
$respHtml->assertElementExists('form[action$="/delete"]');
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function createRevisions(Page $page, int $times, array $attrs = [])
|
|
|
|
{
|
|
|
|
$user = user();
|
|
|
|
|
|
|
|
for ($i = 0; $i < $times; $i++) {
|
|
|
|
$data = ['name' => 'Page update' . $i, 'summary' => 'Update entry' . $i];
|
|
|
|
if (!isset($attrs['markdown'])) {
|
|
|
|
$data['html'] = '<p>My update page</p>';
|
|
|
|
}
|
|
|
|
$this->asAdmin()->put($page->getUrl(), array_merge($data, $attrs));
|
|
|
|
$page->refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->actingAs($user);
|
|
|
|
}
|
2021-06-26 17:23:15 +02:00
|
|
|
}
|