3 namespace Tests\Search;
5 use BookStack\Activity\Models\Tag;
6 use BookStack\Entities\Models\Book;
9 class EntitySearchTest extends TestCase
11 public function test_page_search()
13 $book = $this->entities->book();
14 $page = $book->pages->first();
16 $search = $this->asEditor()->get('/search?term=' . urlencode($page->name));
17 $search->assertSee('Search Results');
18 $search->assertSeeText($page->name, true);
21 public function test_bookshelf_search()
23 $shelf = $this->entities->shelf();
25 $search = $this->asEditor()->get('/search?term=' . urlencode($shelf->name) . ' {type:bookshelf}');
26 $search->assertSee('Search Results');
27 $search->assertSeeText($shelf->name, true);
30 public function test_search_shows_pagination()
32 $search = $this->asEditor()->get('/search?term=a');
33 $this->withHtml($search)->assertLinkExists('/search?term=a&page=2', '2');
36 public function test_invalid_page_search()
38 $resp = $this->asEditor()->get('/search?term=' . urlencode('<p>test</p>'));
39 $resp->assertSee('Search Results');
40 $resp->assertStatus(200);
41 $this->get('/search?term=cat+-')->assertStatus(200);
44 public function test_empty_search_shows_search_page()
46 $res = $this->asEditor()->get('/search');
47 $res->assertStatus(200);
50 public function test_searching_accents_and_small_terms()
52 $page = $this->entities->newPage(['name' => 'My new test quaffleachits', 'html' => 'some áéííúü¿¡ test content a2 orange dog']);
55 $accentSearch = $this->get('/search?term=' . urlencode('áéíí'));
56 $accentSearch->assertStatus(200)->assertSee($page->name);
58 $smallSearch = $this->get('/search?term=' . urlencode('a2'));
59 $smallSearch->assertStatus(200)->assertSee($page->name);
62 public function test_book_search()
64 $book = Book::first();
65 $page = $book->pages->last();
66 $chapter = $book->chapters->last();
68 $pageTestResp = $this->asEditor()->get('/search/book/' . $book->id . '?term=' . urlencode($page->name));
69 $pageTestResp->assertSee($page->name);
71 $chapterTestResp = $this->asEditor()->get('/search/book/' . $book->id . '?term=' . urlencode($chapter->name));
72 $chapterTestResp->assertSee($chapter->name);
75 public function test_chapter_search()
77 $chapter = $this->entities->chapterHasPages();
78 $page = $chapter->pages[0];
80 $pageTestResp = $this->asEditor()->get('/search/chapter/' . $chapter->id . '?term=' . urlencode($page->name));
81 $pageTestResp->assertSee($page->name);
84 public function test_tag_search()
97 $pageA = $this->entities->page();
98 $pageA->tags()->saveMany($newTags);
100 $pageB = $this->entities->page();
101 $pageB->tags()->create(['name' => 'animal', 'value' => 'dog']);
104 $tNameSearch = $this->get('/search?term=%5Banimal%5D');
105 $tNameSearch->assertSee($pageA->name)->assertSee($pageB->name);
107 $tNameSearch2 = $this->get('/search?term=%5Bcolor%5D');
108 $tNameSearch2->assertSee($pageA->name)->assertDontSee($pageB->name);
110 $tNameValSearch = $this->get('/search?term=%5Banimal%3Dcat%5D');
111 $tNameValSearch->assertSee($pageA->name)->assertDontSee($pageB->name);
114 public function test_exact_searches()
116 $page = $this->entities->newPage(['name' => 'My new test page', 'html' => 'this is a story about an orange donkey']);
118 $exactSearchA = $this->asEditor()->get('/search?term=' . urlencode('"story about an orange"'));
119 $exactSearchA->assertStatus(200)->assertSee($page->name);
121 $exactSearchB = $this->asEditor()->get('/search?term=' . urlencode('"story not about an orange"'));
122 $exactSearchB->assertStatus(200)->assertDontSee($page->name);
125 public function test_negated_searches()
127 $page = $this->entities->newPage(['name' => 'My new test negation page', 'html' => '<p>An angry tortoise wore trumpeted plimsoles</p>']);
128 $page->tags()->saveMany([new Tag(['name' => 'DonkCount', 'value' => '500'])]);
129 $page->created_by = $this->users->admin()->id;
132 $editor = $this->users->editor();
133 $this->actingAs($editor);
135 $exactSearch = $this->get('/search?term=' . urlencode('negation -"tortoise"'));
136 $exactSearch->assertStatus(200)->assertDontSeeText($page->name);
138 $tagSearchA = $this->get('/search?term=' . urlencode('negation [DonkCount=500]'));
139 $tagSearchA->assertStatus(200)->assertSeeText($page->name);
140 $tagSearchB = $this->get('/search?term=' . urlencode('negation -[DonkCount=500]'));
141 $tagSearchB->assertStatus(200)->assertDontSeeText($page->name);
143 $filterSearchA = $this->get('/search?term=' . urlencode('negation -{created_by:me}'));
144 $filterSearchA->assertStatus(200)->assertSeeText($page->name);
145 $page->created_by = $editor->id;
147 $filterSearchB = $this->get('/search?term=' . urlencode('negation -{created_by:me}'));
148 $filterSearchB->assertStatus(200)->assertDontSeeText($page->name);
151 public function test_search_terms_with_delimiters_are_converted_to_exact_matches()
154 $page = $this->entities->newPage(['name' => 'Delimiter test', 'html' => '<p>1.1 2,2 3?3 4:4 5;5 (8) <9> "10" \'11\' `12`</p>']);
155 $terms = explode(' ', '1.1 2,2 3?3 4:4 5;5 (8) <9> "10" \'11\' `12`');
157 foreach ($terms as $term) {
158 $search = $this->get('/search?term=' . urlencode($term));
159 $search->assertSee($page->name);
163 public function test_search_filters()
165 $page = $this->entities->newPage(['name' => 'My new test quaffleachits', 'html' => 'this is about an orange donkey danzorbhsing']);
166 $editor = $this->users->editor();
167 $this->actingAs($editor);
169 // Viewed filter searches
170 $this->get('/search?term=' . urlencode('danzorbhsing {not_viewed_by_me}'))->assertSee($page->name);
171 $this->get('/search?term=' . urlencode('danzorbhsing {viewed_by_me}'))->assertDontSee($page->name);
172 $this->get($page->getUrl());
173 $this->get('/search?term=' . urlencode('danzorbhsing {not_viewed_by_me}'))->assertDontSee($page->name);
174 $this->get('/search?term=' . urlencode('danzorbhsing {viewed_by_me}'))->assertSee($page->name);
177 $this->get('/search?term=' . urlencode('danzorbhsing {created_by:me}'))->assertDontSee($page->name);
178 $this->get('/search?term=' . urlencode('danzorbhsing {updated_by:me}'))->assertDontSee($page->name);
179 $this->get('/search?term=' . urlencode('danzorbhsing {owned_by:me}'))->assertDontSee($page->name);
180 $this->get('/search?term=' . urlencode('danzorbhsing {updated_by:' . $editor->slug . '}'))->assertDontSee($page->name);
181 $page->created_by = $editor->id;
183 $this->get('/search?term=' . urlencode('danzorbhsing {created_by:me}'))->assertSee($page->name);
184 $this->get('/search?term=' . urlencode('danzorbhsing {created_by: ' . $editor->slug . '}'))->assertSee($page->name);
185 $this->get('/search?term=' . urlencode('danzorbhsing {updated_by:me}'))->assertDontSee($page->name);
186 $this->get('/search?term=' . urlencode('danzorbhsing {owned_by:me}'))->assertDontSee($page->name);
187 $page->updated_by = $editor->id;
189 $this->get('/search?term=' . urlencode('danzorbhsing {updated_by:me}'))->assertSee($page->name);
190 $this->get('/search?term=' . urlencode('danzorbhsing {updated_by:' . $editor->slug . '}'))->assertSee($page->name);
191 $this->get('/search?term=' . urlencode('danzorbhsing {owned_by:me}'))->assertDontSee($page->name);
192 $page->owned_by = $editor->id;
194 $this->get('/search?term=' . urlencode('danzorbhsing {owned_by:me}'))->assertSee($page->name);
195 $this->get('/search?term=' . urlencode('danzorbhsing {owned_by:' . $editor->slug . '}'))->assertSee($page->name);
198 $this->get('/search?term=' . urlencode('{in_name:danzorbhsing}'))->assertDontSee($page->name);
199 $this->get('/search?term=' . urlencode('{in_body:danzorbhsing}'))->assertSee($page->name);
200 $this->get('/search?term=' . urlencode('{in_name:test quaffleachits}'))->assertSee($page->name);
201 $this->get('/search?term=' . urlencode('{in_body:test quaffleachits}'))->assertDontSee($page->name);
204 $this->get('/search?term=' . urlencode('danzorbhsing {is_restricted}'))->assertDontSee($page->name);
205 $this->permissions->setEntityPermissions($page, ['view'], [$editor->roles->first()]);
206 $this->get('/search?term=' . urlencode('danzorbhsing {is_restricted}'))->assertSee($page->name);
209 $this->get('/search?term=' . urlencode('danzorbhsing {updated_after:2037-01-01}'))->assertDontSee($page->name);
210 $this->get('/search?term=' . urlencode('danzorbhsing {updated_before:2037-01-01}'))->assertSee($page->name);
211 $page->updated_at = '2037-02-01';
213 $this->get('/search?term=' . urlencode('danzorbhsing {updated_after:2037-01-01}'))->assertSee($page->name);
214 $this->get('/search?term=' . urlencode('danzorbhsing {updated_before:2037-01-01}'))->assertDontSee($page->name);
216 $this->get('/search?term=' . urlencode('danzorbhsing {created_after:2037-01-01}'))->assertDontSee($page->name);
217 $this->get('/search?term=' . urlencode('danzorbhsing {created_before:2037-01-01}'))->assertSee($page->name);
218 $page->created_at = '2037-02-01';
220 $this->get('/search?term=' . urlencode('danzorbhsing {created_after:2037-01-01}'))->assertSee($page->name);
221 $this->get('/search?term=' . urlencode('danzorbhsing {created_before:2037-01-01}'))->assertDontSee($page->name);
224 public function test_entity_selector_search()
226 $page = $this->entities->newPage(['name' => 'my ajax search test', 'html' => 'ajax test']);
227 $notVisitedPage = $this->entities->page();
229 // Visit the page to make popular
230 $this->asEditor()->get($page->getUrl());
232 $normalSearch = $this->get('/search/entity-selector?term=' . urlencode($page->name));
233 $normalSearch->assertSee($page->name);
235 $bookSearch = $this->get('/search/entity-selector?types=book&term=' . urlencode($page->name));
236 $bookSearch->assertDontSee($page->name);
238 $defaultListTest = $this->get('/search/entity-selector');
239 $defaultListTest->assertSee($page->name);
240 $defaultListTest->assertDontSee($notVisitedPage->name);
243 public function test_entity_selector_search_shows_breadcrumbs()
245 $chapter = $this->entities->chapter();
246 $page = $chapter->pages->first();
249 $pageSearch = $this->get('/search/entity-selector?term=' . urlencode($page->name));
250 $pageSearch->assertSee($page->name);
251 $pageSearch->assertSee($chapter->getShortName(42));
252 $pageSearch->assertSee($page->book->getShortName(42));
254 $chapterSearch = $this->get('/search/entity-selector?term=' . urlencode($chapter->name));
255 $chapterSearch->assertSee($chapter->name);
256 $chapterSearch->assertSee($chapter->book->getShortName(42));
259 public function test_entity_selector_shows_breadcrumbs_on_default_view()
261 $page = $this->entities->pageWithinChapter();
262 $this->asEditor()->get($page->chapter->getUrl());
264 $resp = $this->asEditor()->get('/search/entity-selector?types=book,chapter&permission=page-create');
265 $html = $this->withHtml($resp);
266 $html->assertElementContains('.chapter.entity-list-item', $page->chapter->name);
267 $html->assertElementContains('.chapter.entity-list-item .entity-item-snippet', $page->book->getShortName(42));
270 public function test_entity_selector_search_reflects_items_without_permission()
272 $page = $this->entities->page();
273 $baseSelector = 'a[data-entity-type="page"][data-entity-id="' . $page->id . '"]';
274 $searchUrl = '/search/entity-selector?permission=update&term=' . urlencode($page->name);
276 $resp = $this->asEditor()->get($searchUrl);
277 $this->withHtml($resp)->assertElementContains($baseSelector, $page->name);
278 $this->withHtml($resp)->assertElementNotContains($baseSelector, "You don't have the required permissions to select this item");
280 $resp = $this->actingAs($this->users->viewer())->get($searchUrl);
281 $this->withHtml($resp)->assertElementContains($baseSelector, $page->name);
282 $this->withHtml($resp)->assertElementContains($baseSelector, "You don't have the required permissions to select this item");
285 public function test_entity_template_selector_search()
287 $templatePage = $this->entities->newPage(['name' => 'Template search test', 'html' => 'template test']);
288 $templatePage->template = true;
289 $templatePage->save();
291 $nonTemplatePage = $this->entities->newPage(['name' => 'Nontemplate page', 'html' => 'nontemplate', 'template' => false]);
293 // Visit both to make popular
294 $this->asEditor()->get($templatePage->getUrl());
295 $this->get($nonTemplatePage->getUrl());
297 $normalSearch = $this->get('/search/entity-selector-templates?term=test');
298 $normalSearch->assertSee($templatePage->name);
299 $normalSearch->assertDontSee($nonTemplatePage->name);
301 $normalSearch = $this->get('/search/entity-selector-templates?term=beans');
302 $normalSearch->assertDontSee($templatePage->name);
303 $normalSearch->assertDontSee($nonTemplatePage->name);
305 $defaultListTest = $this->get('/search/entity-selector-templates');
306 $defaultListTest->assertSee($templatePage->name);
307 $defaultListTest->assertDontSee($nonTemplatePage->name);
309 $this->permissions->disableEntityInheritedPermissions($templatePage);
311 $normalSearch = $this->get('/search/entity-selector-templates?term=test');
312 $normalSearch->assertDontSee($templatePage->name);
314 $defaultListTest = $this->get('/search/entity-selector-templates');
315 $defaultListTest->assertDontSee($templatePage->name);
318 public function test_search_works_on_updated_page_content()
320 $page = $this->entities->page();
323 $update = $this->put($page->getUrl(), [
324 'name' => $page->name,
325 'html' => '<p>dog pandabearmonster spaghetti</p>',
328 $search = $this->asEditor()->get('/search?term=pandabearmonster');
329 $search->assertStatus(200);
330 $search->assertSeeText($page->name);
331 $search->assertSee($page->getUrl());
334 public function test_search_ranks_common_words_lower()
336 $this->entities->newPage(['name' => 'Test page A', 'html' => '<p>dog biscuit dog dog</p>']);
337 $this->entities->newPage(['name' => 'Test page B', 'html' => '<p>cat biscuit</p>']);
339 $search = $this->asEditor()->get('/search?term=cat+dog+biscuit');
340 $this->withHtml($search)->assertElementContains('.entity-list > .page:nth-child(1)', 'Test page A');
341 $this->withHtml($search)->assertElementContains('.entity-list > .page:nth-child(2)', 'Test page B');
343 for ($i = 0; $i < 2; $i++) {
344 $this->entities->newPage(['name' => 'Test page ' . $i, 'html' => '<p>dog</p>']);
347 $search = $this->asEditor()->get('/search?term=cat+dog+biscuit');
348 $this->withHtml($search)->assertElementContains('.entity-list > .page:nth-child(1)', 'Test page B');
349 $this->withHtml($search)->assertElementContains('.entity-list > .page:nth-child(2)', 'Test page A');
352 public function test_matching_terms_in_search_results_are_highlighted()
354 $this->entities->newPage(['name' => 'My Meowie Cat', 'html' => '<p>A superimportant page about meowieable animals</p>', 'tags' => [
355 ['name' => 'Animal', 'value' => 'MeowieCat'],
356 ['name' => 'SuperImportant'],
359 $search = $this->asEditor()->get('/search?term=SuperImportant+Meowie');
361 $search->assertSee('My <strong>Meowie</strong> Cat', false);
363 $search->assertSee('A <strong>superimportant</strong> page about <strong>meowie</strong>able animals', false);
365 $this->withHtml($search)->assertElementContains('.tag-name.highlight', 'SuperImportant');
367 $this->withHtml($search)->assertElementContains('.tag-value.highlight', 'MeowieCat');
370 public function test_match_highlighting_works_with_multibyte_content()
372 $this->entities->newPage([
373 'name' => 'Test Page',
374 'html' => '<p>На мен ми трябва нещо добро test</p>',
377 $search = $this->asEditor()->get('/search?term=' . urlencode('На мен ми трябва нещо добро'));
378 $search->assertSee('<strong>На</strong> <strong>мен</strong> <strong>ми</strong> <strong>трябва</strong> <strong>нещо</strong> <strong>добро</strong> test', false);
381 public function test_match_highlighting_is_efficient_with_large_frequency_in_content()
383 $content = str_repeat('superbeans ', 10000);
384 $this->entities->newPage([
385 'name' => 'Test Page',
386 'html' => "<p>{$content}</p>",
389 $time = microtime(true);
390 $resp = $this->asEditor()->get('/search?term=' . urlencode('superbeans'));
391 $this->assertLessThan(0.5, microtime(true) - $time);
393 $resp->assertSee('<strong>superbeans</strong>', false);
396 public function test_html_entities_in_item_details_remains_escaped_in_search_results()
398 $this->entities->newPage(['name' => 'My <cool> TestPageContent', 'html' => '<p>My supercool <great> TestPageContent page</p>']);
400 $search = $this->asEditor()->get('/search?term=TestPageContent');
401 $search->assertSee('My <cool> <strong>TestPageContent</strong>', false);
402 $search->assertSee('My supercool <great> <strong>TestPageContent</strong> page', false);
405 public function test_words_adjacent_to_lines_breaks_can_be_matched_with_normal_terms()
407 $page = $this->entities->newPage(['name' => 'TermA', 'html' => '
408 <p>TermA<br>TermB<br>TermC</p>
411 $search = $this->asEditor()->get('/search?term=' . urlencode('TermB TermC'));
413 $search->assertSee($page->getUrl(), false);
416 public function test_backslashes_can_be_searched_upon()
418 $page = $this->entities->newPage(['name' => 'TermA', 'html' => '
419 <p>More info is at the path \\\\cat\\dog\\badger</p>
421 $page->tags()->save(new Tag(['name' => '\\Category', 'value' => '\\animals\\fluffy']));
423 $search = $this->asEditor()->get('/search?term=' . urlencode('\\\\cat\\dog'));
424 $search->assertSee($page->getUrl(), false);
426 $search = $this->asEditor()->get('/search?term=' . urlencode('"\\dog\\\\"'));
427 $search->assertSee($page->getUrl(), false);
429 $search = $this->asEditor()->get('/search?term=' . urlencode('"\\badger\\\\"'));
430 $search->assertDontSee($page->getUrl(), false);
432 $search = $this->asEditor()->get('/search?term=' . urlencode('[\\Categorylike%\\fluffy]'));
433 $search->assertSee($page->getUrl(), false);
436 public function test_searches_with_terms_without_controls_includes_them_in_extras()
438 $resp = $this->asEditor()->get('/search?term=' . urlencode('test {updated_by:dan} {created_by:dan} -{viewed_by_me} -[a=b] -"dog" {is_template} {sort_by:last_commented}'));
439 $this->withHtml($resp)->assertFieldHasValue('extras', '{updated_by:dan} {created_by:dan} {is_template} {sort_by:last_commented} -"dog" -[a=b] -{viewed_by_me}');
442 public function test_negated_searches_dont_show_in_inputs()
444 $resp = $this->asEditor()->get('/search?term=' . urlencode('-{created_by:me} -[a=b] -"dog"'));
445 $this->withHtml($resp)->assertElementNotExists('input[name="tags[]"][value="a=b"]');
446 $this->withHtml($resp)->assertElementNotExists('input[name="exact[]"][value="dog"]');
447 $this->withHtml($resp)->assertElementNotExists('input[name="filters[created_by]"][value="me"][checked="checked"]');
450 public function test_searches_with_user_filters_using_me_adds_them_into_advanced_search_form()
452 $resp = $this->asEditor()->get('/search?term=' . urlencode('test {updated_by:me} {created_by:me}'));
453 $this->withHtml($resp)->assertElementExists('form input[name="filters[updated_by]"][value="me"][checked="checked"]');
454 $this->withHtml($resp)->assertElementExists('form input[name="filters[created_by]"][value="me"][checked="checked"]');
457 public function test_search_suggestion_endpoint()
459 $this->entities->newPage(['name' => 'My suggestion page', 'html' => '<p>My supercool suggestion page</p>']);
461 // Test specific search
462 $resp = $this->asEditor()->get('/search/suggest?term="supercool+suggestion"');
463 $resp->assertSee('My suggestion page');
464 $resp->assertDontSee('My supercool suggestion page');
465 $resp->assertDontSee('No items available');
466 $this->withHtml($resp)->assertElementCount('a', 1);
469 $resp = $this->asEditor()->get('/search/suggest?term=et');
470 $this->withHtml($resp)->assertElementCount('a', 5);
473 $resp = $this->asEditor()->get('/search/suggest?term=spaghettisaurusrex');
474 $this->withHtml($resp)->assertElementCount('a', 0);
475 $resp->assertSee('No items available');