3 namespace BookStack\Actions;
5 use BookStack\Auth\Permissions\PermissionApplicator;
6 use BookStack\Entities\Models\Entity;
7 use BookStack\Util\SimpleListOptions;
8 use Illuminate\Database\Eloquent\Builder;
9 use Illuminate\Support\Collection;
10 use Illuminate\Support\Facades\DB;
14 protected PermissionApplicator $permissions;
16 public function __construct(PermissionApplicator $permissions)
18 $this->permissions = $permissions;
22 * Start a query against all tags in the system.
24 public function queryWithTotals(SimpleListOptions $listOptions, string $nameFilter): Builder
26 $searchTerm = $listOptions->getSearch();
27 $sort = $listOptions->getSort();
28 if ($sort === 'name' && $nameFilter) {
32 $entityTypeCol = DB::getTablePrefix() . 'tags.entity_type';
36 ($searchTerm || $nameFilter) ? 'value' : DB::raw('COUNT(distinct value) as `values`'),
37 DB::raw('COUNT(id) as usages'),
38 DB::raw("SUM(IF({$entityTypeCol} = 'page', 1, 0)) as page_count"),
39 DB::raw("SUM(IF({$entityTypeCol} = 'chapter', 1, 0)) as chapter_count"),
40 DB::raw("SUM(IF({$entityTypeCol} = 'book', 1, 0)) as book_count"),
41 DB::raw("SUM(IF({$entityTypeCol} = 'bookshelf', 1, 0)) as shelf_count"),
43 ->orderBy($sort, $listOptions->getOrder());
46 $query->where('name', '=', $nameFilter);
47 $query->groupBy('value');
48 } elseif ($searchTerm) {
49 $query->groupBy('name', 'value');
51 $query->groupBy('name');
55 $query->where(function (Builder $query) use ($searchTerm) {
56 $query->where('name', 'like', '%' . $searchTerm . '%')
57 ->orWhere('value', 'like', '%' . $searchTerm . '%');
61 return $this->permissions->restrictEntityRelationQuery($query, 'tags', 'entity_id', 'entity_type');
65 * Get tag name suggestions from scanning existing tag names.
66 * If no search term is given the 50 most popular tag names are provided.
68 public function getNameSuggestions(string $searchTerm): Collection
71 ->select('*', DB::raw('count(*) as count'))
75 $query = $query->where('name', 'LIKE', $searchTerm . '%')->orderBy('name', 'asc');
77 $query = $query->orderBy('count', 'desc')->take(50);
80 $query = $this->permissions->restrictEntityRelationQuery($query, 'tags', 'entity_id', 'entity_type');
82 return $query->pluck('name');
86 * Get tag value suggestions from scanning existing tag values.
87 * If no search is given the 50 most popular values are provided.
88 * Passing a tagName will only find values for a tags with a particular name.
90 public function getValueSuggestions(string $searchTerm, string $tagName): Collection
93 ->select('*', DB::raw('count(*) as count'))
97 $query = $query->where('value', 'LIKE', $searchTerm . '%')->orderBy('value', 'desc');
99 $query = $query->orderBy('count', 'desc')->take(50);
103 $query = $query->where('name', '=', $tagName);
106 $query = $this->permissions->restrictEntityRelationQuery($query, 'tags', 'entity_id', 'entity_type');
108 return $query->pluck('value');
112 * Save an array of tags to an entity.
114 public function saveTagsToEntity(Entity $entity, array $tags = []): iterable
116 $entity->tags()->delete();
118 $newTags = collect($tags)->filter(function ($tag) {
119 return boolval(trim($tag['name']));
120 })->map(function ($tag) {
121 return $this->newInstanceFromInput($tag);
124 return $entity->tags()->saveMany($newTags);
128 * Create a new Tag instance from user input.
129 * Input must be an array with a 'name' and an optional 'value' key.
131 protected function newInstanceFromInput(array $input): Tag
134 'name' => trim($input['name']),
135 'value' => trim($input['value'] ?? ''),