BookStack/app/Repos/PageRepo.php

443 lines
14 KiB
PHP
Raw Normal View History

<?php namespace BookStack\Repos;
2015-07-13 04:31:15 +08:00
use Activity;
use BookStack\Book;
use BookStack\Chapter;
use BookStack\Exceptions\NotFoundException;
use BookStack\Services\RestrictionService;
use Illuminate\Http\Request;
2015-08-09 19:06:52 +08:00
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Log;
2015-07-13 04:31:15 +08:00
use Illuminate\Support\Str;
use BookStack\Page;
use BookStack\PageRevision;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
2015-07-13 04:31:15 +08:00
class PageRepo
{
protected $page;
2015-08-09 19:06:52 +08:00
protected $pageRevision;
protected $restrictionService;
2015-07-13 04:31:15 +08:00
/**
* PageRepo constructor.
* @param Page $page
2015-08-09 19:06:52 +08:00
* @param PageRevision $pageRevision
* @param RestrictionService $restrictionService
2015-07-13 04:31:15 +08:00
*/
public function __construct(Page $page, PageRevision $pageRevision, RestrictionService $restrictionService)
2015-07-13 04:31:15 +08:00
{
$this->page = $page;
2015-08-09 19:06:52 +08:00
$this->pageRevision = $pageRevision;
$this->restrictionService = $restrictionService;
2015-07-13 04:31:15 +08:00
}
/**
* Base query for getting pages, Takes restrictions into account.
* @return mixed
*/
private function pageQuery()
{
return $this->restrictionService->enforcePageRestrictions($this->page, 'view');
}
/**
* Get a page via a specific ID.
* @param $id
* @return mixed
*/
2015-07-13 04:31:15 +08:00
public function getById($id)
{
return $this->pageQuery()->findOrFail($id);
2015-07-13 04:31:15 +08:00
}
/**
* Get a page identified by the given slug.
* @param $slug
* @param $bookId
* @return mixed
* @throws NotFoundException
*/
2015-07-16 05:55:49 +08:00
public function getBySlug($slug, $bookId)
2015-07-13 04:31:15 +08:00
{
$page = $this->pageQuery()->where('slug', '=', $slug)->where('book_id', '=', $bookId)->first();
if ($page === null) throw new NotFoundException('Page not found');
2015-12-29 01:19:23 +08:00
return $page;
2015-07-13 04:31:15 +08:00
}
/**
* Search through page revisions and retrieve
* the last page in the current book that
* has a slug equal to the one given.
* @param $pageSlug
* @param $bookSlug
* @return null | Page
*/
public function findPageUsingOldSlug($pageSlug, $bookSlug)
{
$revision = $this->pageRevision->where('slug', '=', $pageSlug)
->whereHas('page', function($query) {
$this->restrictionService->enforcePageRestrictions($query);
})
->where('book_slug', '=', $bookSlug)->orderBy('created_at', 'desc')
->with('page')->first();
return $revision !== null ? $revision->page : null;
}
/**
* Get a new Page instance from the given input.
* @param $input
* @return Page
*/
2015-07-13 04:31:15 +08:00
public function newFromInput($input)
{
$page = $this->page->fill($input);
return $page;
}
/**
* Count the pages with a particular slug within a book.
* @param $slug
* @param $bookId
* @return mixed
*/
2015-07-13 04:31:15 +08:00
public function countBySlug($slug, $bookId)
{
return $this->page->where('slug', '=', $slug)->where('book_id', '=', $bookId)->count();
}
/**
* Save a new page into the system.
* Input validation must be done beforehand.
* @param array $input
* @param Book $book
* @param int $chapterId
* @return Page
*/
public function saveNew(array $input, Book $book, $chapterId = null)
{
$page = $this->newFromInput($input);
$page->slug = $this->findSuitableSlug($page->name, $book->id);
if ($chapterId) $page->chapter_id = $chapterId;
$page->html = $this->formatHtml($input['html']);
$page->text = strip_tags($page->html);
$page->created_by = auth()->user()->id;
$page->updated_by = auth()->user()->id;
$book->pages()->save($page);
return $page;
}
/**
* Formats a page's html to be tagged correctly
* within the system.
* @param string $htmlText
* @return string
*/
protected function formatHtml($htmlText)
{
if($htmlText == '') return $htmlText;
libxml_use_internal_errors(true);
$doc = new \DOMDocument();
$doc->loadHTML(mb_convert_encoding($htmlText, 'HTML-ENTITIES', 'UTF-8'));
$container = $doc->documentElement;
$body = $container->childNodes->item(0);
$childNodes = $body->childNodes;
// Ensure no duplicate ids are used
$idArray = [];
foreach ($childNodes as $index => $childNode) {
/** @var \DOMElement $childNode */
if (get_class($childNode) !== 'DOMElement') continue;
// Overwrite id if not a BookStack custom id
if ($childNode->hasAttribute('id')) {
$id = $childNode->getAttribute('id');
if (strpos($id, 'bkmrk') === 0 && array_search($id, $idArray) === false) {
$idArray[] = $id;
continue;
};
}
// Create an unique id for the element
// Uses the content as a basis to ensure output is the same every time
// the same content is passed through.
$contentId = 'bkmrk-' . substr(strtolower(preg_replace('/\s+/', '-', trim($childNode->nodeValue))), 0, 20);
$newId = urlencode($contentId);
$loopIndex = 0;
while (in_array($newId, $idArray)) {
$newId = urlencode($contentId . '-' . $loopIndex);
$loopIndex++;
}
$childNode->setAttribute('id', $newId);
$idArray[] = $newId;
}
// Generate inner html as a string
$html = '';
foreach ($childNodes as $childNode) {
$html .= $doc->saveHTML($childNode);
}
return $html;
}
2015-07-13 04:31:15 +08:00
/**
* Gets pages by a search term.
* Highlights page content for showing in results.
* @param string $term
* @param array $whereTerms
* @param int $count
* @param array $paginationAppends
* @return mixed
*/
public function getBySearch($term, $whereTerms = [], $count = 20, $paginationAppends = [])
2015-07-17 02:53:24 +08:00
{
preg_match_all('/"(.*?)"/', $term, $matches);
if (count($matches[1]) > 0) {
$terms = $matches[1];
$term = trim(preg_replace('/"(.*?)"/', '', $term));
} else {
$terms = [];
}
if (!empty($term)) {
$terms = array_merge($terms, explode(' ', $term));
}
$pages = $this->restrictionService->enforcePageRestrictions($this->page->fullTextSearchQuery(['name', 'text'], $terms, $whereTerms))
->paginate($count)->appends($paginationAppends);
// Add highlights to page text.
$words = join('|', explode(' ', preg_quote(trim($term), '/')));
//lookahead/behind assertions ensures cut between words
$s = '\s\x00-/:-@\[-`{-~'; //character set for start/end of words
foreach ($pages as $page) {
preg_match_all('#(?<=[' . $s . ']).{1,30}((' . $words . ').{1,30})+(?=[' . $s . '])#uis', $page->text, $matches, PREG_SET_ORDER);
//delimiter between occurrences
$results = [];
foreach ($matches as $line) {
$results[] = htmlspecialchars($line[0], 0, 'UTF-8');
}
$matchLimit = 6;
if (count($results) > $matchLimit) {
$results = array_slice($results, 0, $matchLimit);
}
$result = join('... ', $results);
//highlight
$result = preg_replace('#' . $words . '#iu', "<span class=\"highlight\">\$0</span>", $result);
if (strlen($result) < 5) {
$result = $page->getExcerpt(80);
}
$page->searchSnippet = $result;
2015-07-17 02:53:24 +08:00
}
return $pages;
2015-07-17 02:53:24 +08:00
}
/**
* Search for image usage.
* @param $imageString
* @return mixed
*/
public function searchForImage($imageString)
{
$pages = $this->pageQuery()->where('html', 'like', '%' . $imageString . '%')->get();
foreach ($pages as $page) {
$page->url = $page->getUrl();
$page->html = '';
$page->text = '';
}
return count($pages) > 0 ? $pages : false;
}
2015-08-09 19:06:52 +08:00
/**
* Updates a page with any fillable data and saves it into the database.
* @param Page $page
* @param int $book_id
* @param string $input
2015-08-09 19:06:52 +08:00
* @return Page
*/
public function updatePage(Page $page, $book_id, $input)
2015-08-09 19:06:52 +08:00
{
2015-10-19 02:40:33 +08:00
// Save a revision before updating
if ($page->html !== $input['html'] || $page->name !== $input['name']) {
$this->saveRevision($page);
}
// Prevent slug being updated if no name change
if ($page->name !== $input['name']) {
$page->slug = $this->findSuitableSlug($input['name'], $book_id, $page->id);
}
2015-10-19 02:40:33 +08:00
// Update with new details
$page->fill($input);
$page->html = $this->formatHtml($input['html']);
2015-08-09 19:06:52 +08:00
$page->text = strip_tags($page->html);
2015-10-19 02:40:33 +08:00
$page->updated_by = auth()->user()->id;
2015-08-09 19:06:52 +08:00
$page->save();
2015-10-19 02:40:33 +08:00
return $page;
}
/**
* Restores a revision's content back into a page.
* @param Page $page
* @param Book $book
* @param int $revisionId
* @return Page
*/
public function restoreRevision(Page $page, Book $book, $revisionId)
{
2015-08-09 19:06:52 +08:00
$this->saveRevision($page);
2015-10-19 02:40:33 +08:00
$revision = $this->getRevisionById($revisionId);
$page->fill($revision->toArray());
$page->slug = $this->findSuitableSlug($page->name, $book->id, $page->id);
$page->text = strip_tags($page->html);
$page->updated_by = auth()->user()->id;
$page->save();
2015-08-09 19:06:52 +08:00
return $page;
}
/**
* Saves a page revision into the system.
* @param Page $page
* @return $this
*/
public function saveRevision(Page $page)
2015-08-09 19:06:52 +08:00
{
$revision = $this->pageRevision->fill($page->toArray());
$revision->page_id = $page->id;
$revision->slug = $page->slug;
$revision->book_slug = $page->book->slug;
2015-10-19 02:40:33 +08:00
$revision->created_by = auth()->user()->id;
$revision->created_at = $page->updated_at;
2015-08-09 19:06:52 +08:00
$revision->save();
// Clear old revisions
if ($this->pageRevision->where('page_id', '=', $page->id)->count() > 50) {
2015-08-09 19:06:52 +08:00
$this->pageRevision->where('page_id', '=', $page->id)
->orderBy('created_at', 'desc')->skip(50)->take(5)->delete();
}
return $revision;
}
/**
* Gets a single revision via it's id.
* @param $id
* @return mixed
*/
public function getRevisionById($id)
{
return $this->pageRevision->findOrFail($id);
}
2015-07-21 05:05:26 +08:00
/**
* Checks if a slug exists within a book already.
* @param $slug
* @param $bookId
* @param bool|false $currentId
* @return bool
2015-07-21 05:05:26 +08:00
*/
public function doesSlugExist($slug, $bookId, $currentId = false)
2015-07-21 05:05:26 +08:00
{
$query = $this->page->where('slug', '=', $slug)->where('book_id', '=', $bookId);
2015-10-19 02:40:33 +08:00
if ($currentId) $query = $query->where('id', '!=', $currentId);
return $query->count() > 0;
2015-07-22 03:13:29 +08:00
}
2015-10-19 02:40:33 +08:00
/**
2015-11-22 02:11:46 +08:00
* Changes the related book for the specified page.
2015-10-19 02:40:33 +08:00
* Changes the book id of any relations to the page that store the book id.
* @param int $bookId
* @param Page $page
* @return Page
*/
2015-11-22 02:11:46 +08:00
public function changeBook($bookId, Page $page)
2015-09-06 21:35:53 +08:00
{
$page->book_id = $bookId;
foreach ($page->activity as $activity) {
2015-09-06 21:35:53 +08:00
$activity->book_id = $bookId;
$activity->save();
}
2015-11-22 02:11:46 +08:00
$page->slug = $this->findSuitableSlug($page->name, $bookId, $page->id);
2015-09-06 21:35:53 +08:00
$page->save();
return $page;
}
2015-07-22 03:13:29 +08:00
/**
* Gets a suitable slug for the resource
* @param $name
* @param $bookId
* @param bool|false $currentId
* @return string
2015-07-22 03:13:29 +08:00
*/
public function findSuitableSlug($name, $bookId, $currentId = false)
2015-07-22 03:13:29 +08:00
{
$slug = Str::slug($name);
while ($this->doesSlugExist($slug, $bookId, $currentId)) {
$slug .= '-' . substr(md5(rand(1, 500)), 0, 3);
2015-07-22 03:13:29 +08:00
}
return $slug;
2015-07-21 05:05:26 +08:00
}
/**
* Destroy a given page along with its dependencies.
* @param $page
*/
public function destroy($page)
{
Activity::removeEntity($page);
$page->views()->delete();
$page->revisions()->delete();
$page->restrictions()->delete();
$page->delete();
}
/**
* Get the latest pages added to the system.
* @param $count
*/
public function getRecentlyCreatedPaginated($count = 20)
{
return $this->pageQuery()->orderBy('created_at', 'desc')->paginate($count);
}
/**
* Get the latest pages added to the system.
* @param $count
*/
public function getRecentlyUpdatedPaginated($count = 20)
{
return $this->pageQuery()->orderBy('updated_at', 'desc')->paginate($count);
}
/**
* Updates pages restrictions from a request
* @param $request
* @param $page
*/
public function updateRestrictionsFromRequest($request, $page)
{
// TODO - extract into shared repo
$page->restricted = $request->has('restricted') && $request->get('restricted') === 'true';
$page->restrictions()->delete();
if ($request->has('restrictions')) {
foreach($request->get('restrictions') as $roleId => $restrictions) {
foreach ($restrictions as $action => $value) {
$page->restrictions()->create([
'role_id' => $roleId,
'action' => strtolower($action)
]);
}
}
}
$page->save();
}
}