mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-01-22 23:37:31 +08:00
3625f12abe
Creates a new organsied formatting system for webhook data, with interfaces for extending with custom model formatting rules. Allows easy usage & extension of the default bookstack formatting behaviour when customizing webhook events via theme system, and keeps default data customizations organised. This also makes the following webhook data changes: - owned_by/created_by/updated_by user details are loaded for events with Entity details. (POTENTIALLY BREAKING CHANGE). - current_revision details are loaded for page update/create events. Added testing to cover added model formatting rules. For #3279 and #3218
140 lines
4.3 KiB
PHP
140 lines
4.3 KiB
PHP
<?php
|
|
|
|
namespace BookStack\Http\Controllers;
|
|
|
|
use BookStack\Entities\Repos\PageRepo;
|
|
use BookStack\Entities\Tools\PageContent;
|
|
use BookStack\Exceptions\NotFoundException;
|
|
use Ssddanbrown\HtmlDiff\Diff;
|
|
|
|
class PageRevisionController extends Controller
|
|
{
|
|
protected $pageRepo;
|
|
|
|
/**
|
|
* PageRevisionController constructor.
|
|
*/
|
|
public function __construct(PageRepo $pageRepo)
|
|
{
|
|
$this->pageRepo = $pageRepo;
|
|
}
|
|
|
|
/**
|
|
* Shows the last revisions for this page.
|
|
*
|
|
* @throws NotFoundException
|
|
*/
|
|
public function index(string $bookSlug, string $pageSlug)
|
|
{
|
|
$page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
|
|
$this->setPageTitle(trans('entities.pages_revisions_named', ['pageName'=>$page->getShortName()]));
|
|
|
|
return view('pages.revisions', [
|
|
'page' => $page,
|
|
'current' => $page,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Shows a preview of a single revision.
|
|
*
|
|
* @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());
|
|
// TODO - Refactor PageContent so we don't need to juggle this
|
|
$page->html = $revision->html;
|
|
$page->html = (new PageContent($page))->render();
|
|
|
|
$this->setPageTitle(trans('entities.pages_revision_named', ['pageName' => $page->getShortName()]));
|
|
|
|
return view('pages.revision', [
|
|
'page' => $page,
|
|
'book' => $page->book,
|
|
'diff' => null,
|
|
'revision' => $revision,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Shows the changes of a single revision.
|
|
*
|
|
* @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 ?? '';
|
|
$diff = Diff::excecute($prevContent, $revision->html);
|
|
|
|
$page->fill($revision->toArray());
|
|
// TODO - Refactor PageContent so we don't need to juggle this
|
|
$page->html = $revision->html;
|
|
$page->html = (new PageContent($page))->render();
|
|
$this->setPageTitle(trans('entities.pages_revision_named', ['pageName'=>$page->getShortName()]));
|
|
|
|
return view('pages.revision', [
|
|
'page' => $page,
|
|
'book' => $page->book,
|
|
'diff' => $diff,
|
|
'revision' => $revision,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Restores a page using the content of the specified revision.
|
|
*
|
|
* @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.
|
|
*
|
|
* @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");
|
|
}
|
|
|
|
// Check if it's the latest revision, cannot delete the latest revision.
|
|
if (intval($page->currentRevision->id ?? null) === intval($revId)) {
|
|
$this->showErrorNotification(trans('entities.revision_cannot_delete_latest'));
|
|
|
|
return redirect($page->getUrl('/revisions'));
|
|
}
|
|
|
|
$revision->delete();
|
|
$this->showSuccessNotification(trans('entities.revision_delete_success'));
|
|
|
|
return redirect($page->getUrl('/revisions'));
|
|
}
|
|
}
|