2021-06-26 11:23:15 -04:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace BookStack\Http\Controllers;
|
2019-10-05 07:55:01 -04:00
|
|
|
|
2022-08-09 08:25:18 -04:00
|
|
|
use BookStack\Actions\ActivityType;
|
2019-10-05 07:55:01 -04:00
|
|
|
use BookStack\Entities\Repos\PageRepo;
|
2021-06-26 11:23:15 -04:00
|
|
|
use BookStack\Entities\Tools\PageContent;
|
2019-10-05 07:55:01 -04:00
|
|
|
use BookStack\Exceptions\NotFoundException;
|
2022-08-09 08:25:18 -04:00
|
|
|
use BookStack\Facades\Activity;
|
2020-11-29 14:08:13 -05:00
|
|
|
use Ssddanbrown\HtmlDiff\Diff;
|
2019-10-05 07:55:01 -04:00
|
|
|
|
|
|
|
class PageRevisionController extends Controller
|
|
|
|
{
|
2022-08-10 12:50:35 -04:00
|
|
|
protected PageRepo $pageRepo;
|
2019-10-05 07:55:01 -04:00
|
|
|
|
|
|
|
public function __construct(PageRepo $pageRepo)
|
|
|
|
{
|
|
|
|
$this->pageRepo = $pageRepo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the last revisions for this page.
|
2021-06-26 11:23:15 -04:00
|
|
|
*
|
2019-10-05 07:55:01 -04:00
|
|
|
* @throws NotFoundException
|
|
|
|
*/
|
|
|
|
public function index(string $bookSlug, string $pageSlug)
|
|
|
|
{
|
|
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
2022-08-10 12:50:35 -04:00
|
|
|
$revisions = $page->revisions()->select([
|
2022-08-16 06:27:22 -04:00
|
|
|
'id', 'page_id', 'name', 'created_at', 'created_by', 'updated_at',
|
|
|
|
'type', 'revision_number', 'summary',
|
|
|
|
])
|
2022-08-10 12:50:35 -04:00
|
|
|
->selectRaw("IF(markdown = '', false, true) as is_markdown")
|
|
|
|
->with(['page.book', 'createdBy'])
|
|
|
|
->get();
|
|
|
|
|
|
|
|
$this->setPageTitle(trans('entities.pages_revisions_named', ['pageName' => $page->getShortName()]));
|
2021-06-26 11:23:15 -04:00
|
|
|
|
2019-10-05 07:55:01 -04:00
|
|
|
return view('pages.revisions', [
|
2022-08-10 12:50:35 -04:00
|
|
|
'revisions' => $revisions,
|
2022-08-16 06:27:22 -04:00
|
|
|
'page' => $page,
|
2019-10-05 07:55:01 -04:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows a preview of a single revision.
|
2021-06-26 11:23:15 -04:00
|
|
|
*
|
2019-10-05 07:55:01 -04:00
|
|
|
* @throws NotFoundException
|
|
|
|
*/
|
|
|
|
public function show(string $bookSlug, string $pageSlug, int $revisionId)
|
|
|
|
{
|
|
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
|
|
|
$revision = $page->revisions()->where('id', '=', $revisionId)->first();
|
|
|
|
if ($revision === null) {
|
|
|
|
throw new NotFoundException();
|
|
|
|
}
|
|
|
|
|
|
|
|
$page->fill($revision->toArray());
|
2020-05-23 07:28:14 -04:00
|
|
|
// TODO - Refactor PageContent so we don't need to juggle this
|
|
|
|
$page->html = $revision->html;
|
|
|
|
$page->html = (new PageContent($page))->render();
|
2019-10-05 07:55:01 -04:00
|
|
|
|
|
|
|
$this->setPageTitle(trans('entities.pages_revision_named', ['pageName' => $page->getShortName()]));
|
2021-06-26 11:23:15 -04:00
|
|
|
|
2019-10-05 07:55:01 -04:00
|
|
|
return view('pages.revision', [
|
2021-06-26 11:23:15 -04:00
|
|
|
'page' => $page,
|
|
|
|
'book' => $page->book,
|
|
|
|
'diff' => null,
|
|
|
|
'revision' => $revision,
|
2019-10-05 07:55:01 -04:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the changes of a single revision.
|
2021-06-26 11:23:15 -04:00
|
|
|
*
|
2019-10-05 07:55:01 -04:00
|
|
|
* @throws NotFoundException
|
|
|
|
*/
|
|
|
|
public function changes(string $bookSlug, string $pageSlug, int $revisionId)
|
|
|
|
{
|
|
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
|
|
|
$revision = $page->revisions()->where('id', '=', $revisionId)->first();
|
|
|
|
if ($revision === null) {
|
|
|
|
throw new NotFoundException();
|
|
|
|
}
|
|
|
|
|
|
|
|
$prev = $revision->getPrevious();
|
|
|
|
$prevContent = $prev->html ?? '';
|
2020-11-29 14:08:13 -05:00
|
|
|
$diff = Diff::excecute($prevContent, $revision->html);
|
2019-10-05 07:55:01 -04:00
|
|
|
|
|
|
|
$page->fill($revision->toArray());
|
2020-05-23 07:28:14 -04:00
|
|
|
// TODO - Refactor PageContent so we don't need to juggle this
|
|
|
|
$page->html = $revision->html;
|
|
|
|
$page->html = (new PageContent($page))->render();
|
2019-10-05 07:55:01 -04:00
|
|
|
$this->setPageTitle(trans('entities.pages_revision_named', ['pageName'=>$page->getShortName()]));
|
|
|
|
|
|
|
|
return view('pages.revision', [
|
2021-06-26 11:23:15 -04:00
|
|
|
'page' => $page,
|
|
|
|
'book' => $page->book,
|
|
|
|
'diff' => $diff,
|
|
|
|
'revision' => $revision,
|
2019-10-05 07:55:01 -04:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Restores a page using the content of the specified revision.
|
2021-06-26 11:23:15 -04:00
|
|
|
*
|
2019-10-05 07:55:01 -04:00
|
|
|
* @throws NotFoundException
|
|
|
|
*/
|
|
|
|
public function restore(string $bookSlug, string $pageSlug, int $revisionId)
|
|
|
|
{
|
|
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
|
|
|
$this->checkOwnablePermission('page-update', $page);
|
|
|
|
|
|
|
|
$page = $this->pageRepo->restoreRevision($page, $revisionId);
|
|
|
|
|
|
|
|
return redirect($page->getUrl());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Deletes a revision using the id of the specified revision.
|
2021-06-26 11:23:15 -04:00
|
|
|
*
|
2019-10-05 07:55:01 -04:00
|
|
|
* @throws NotFoundException
|
|
|
|
*/
|
|
|
|
public function destroy(string $bookSlug, string $pageSlug, int $revId)
|
|
|
|
{
|
|
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
|
|
|
$this->checkOwnablePermission('page-delete', $page);
|
|
|
|
|
|
|
|
$revision = $page->revisions()->where('id', '=', $revId)->first();
|
|
|
|
if ($revision === null) {
|
|
|
|
throw new NotFoundException("Revision #{$revId} not found");
|
|
|
|
}
|
|
|
|
|
2022-03-26 12:44:34 -04:00
|
|
|
// Check if it's the latest revision, cannot delete the latest revision.
|
|
|
|
if (intval($page->currentRevision->id ?? null) === intval($revId)) {
|
2019-10-05 07:55:01 -04:00
|
|
|
$this->showErrorNotification(trans('entities.revision_cannot_delete_latest'));
|
2021-06-26 11:23:15 -04:00
|
|
|
|
2019-10-05 07:55:01 -04:00
|
|
|
return redirect($page->getUrl('/revisions'));
|
|
|
|
}
|
|
|
|
|
|
|
|
$revision->delete();
|
2022-08-09 08:25:18 -04:00
|
|
|
Activity::add(ActivityType::REVISION_DELETE, $revision);
|
2019-10-05 07:55:01 -04:00
|
|
|
$this->showSuccessNotification(trans('entities.revision_delete_success'));
|
2021-06-26 11:23:15 -04:00
|
|
|
|
2019-10-05 07:55:01 -04:00
|
|
|
return redirect($page->getUrl('/revisions'));
|
|
|
|
}
|
|
|
|
}
|