2021-11-15 00:28:01 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Api;
|
|
|
|
|
2024-10-21 05:12:49 +08:00
|
|
|
use BookStack\Activity\Models\Tag;
|
2021-11-15 00:28:01 +08:00
|
|
|
use BookStack\Entities\Models\Book;
|
|
|
|
use BookStack\Entities\Models\Bookshelf;
|
|
|
|
use BookStack\Entities\Models\Chapter;
|
|
|
|
use BookStack\Entities\Models\Entity;
|
|
|
|
use BookStack\Entities\Models\Page;
|
|
|
|
use Tests\TestCase;
|
|
|
|
|
|
|
|
class SearchApiTest extends TestCase
|
|
|
|
{
|
|
|
|
use TestsApi;
|
|
|
|
|
|
|
|
protected $baseEndpoint = '/api/search';
|
|
|
|
|
|
|
|
public function test_all_endpoint_returns_search_filtered_results_with_query()
|
|
|
|
{
|
|
|
|
$this->actingAsApiEditor();
|
|
|
|
$uniqueTerm = 'MySuperUniqueTermForSearching';
|
|
|
|
|
|
|
|
/** @var Entity $entityClass */
|
|
|
|
foreach ([Page::class, Chapter::class, Book::class, Bookshelf::class] as $entityClass) {
|
|
|
|
/** @var Entity $first */
|
|
|
|
$first = $entityClass::query()->first();
|
|
|
|
$first->update(['name' => $uniqueTerm]);
|
|
|
|
$first->indexForSearch();
|
|
|
|
}
|
|
|
|
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=' . $uniqueTerm . '&count=5&page=1');
|
|
|
|
$resp->assertJsonCount(4, 'data');
|
|
|
|
$resp->assertJsonFragment(['name' => $uniqueTerm, 'type' => 'book']);
|
|
|
|
$resp->assertJsonFragment(['name' => $uniqueTerm, 'type' => 'chapter']);
|
|
|
|
$resp->assertJsonFragment(['name' => $uniqueTerm, 'type' => 'page']);
|
|
|
|
$resp->assertJsonFragment(['name' => $uniqueTerm, 'type' => 'bookshelf']);
|
|
|
|
}
|
|
|
|
|
2021-12-07 04:42:04 +08:00
|
|
|
public function test_all_endpoint_returns_entity_url()
|
|
|
|
{
|
2022-09-30 00:31:38 +08:00
|
|
|
$page = $this->entities->page();
|
2021-12-07 04:42:04 +08:00
|
|
|
$page->update(['name' => 'name with superuniquevalue within']);
|
|
|
|
$page->indexForSearch();
|
|
|
|
|
|
|
|
$resp = $this->actingAsApiAdmin()->getJson($this->baseEndpoint . '?query=superuniquevalue');
|
|
|
|
$resp->assertJsonFragment([
|
|
|
|
'type' => 'page',
|
2024-10-21 05:12:49 +08:00
|
|
|
'url' => $page->getUrl(),
|
2021-12-07 04:42:04 +08:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_all_endpoint_returns_items_with_preview_html()
|
|
|
|
{
|
2022-09-30 00:31:38 +08:00
|
|
|
$book = $this->entities->book();
|
2023-12-21 01:21:09 +08:00
|
|
|
$book->forceFill(['name' => 'name with superuniquevalue within', 'description' => 'Description with superuniquevalue within'])->save();
|
2021-12-07 04:42:04 +08:00
|
|
|
$book->indexForSearch();
|
|
|
|
|
|
|
|
$resp = $this->actingAsApiAdmin()->getJson($this->baseEndpoint . '?query=superuniquevalue');
|
|
|
|
$resp->assertJsonFragment([
|
2024-10-21 05:12:49 +08:00
|
|
|
'type' => 'book',
|
|
|
|
'url' => $book->getUrl(),
|
2021-12-07 04:42:04 +08:00
|
|
|
'preview_html' => [
|
2024-10-21 05:12:49 +08:00
|
|
|
'name' => 'name with <strong>superuniquevalue</strong> within',
|
2021-12-07 04:42:04 +08:00
|
|
|
'content' => 'Description with <strong>superuniquevalue</strong> within',
|
2021-12-15 21:49:20 +08:00
|
|
|
],
|
2021-12-07 04:42:04 +08:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2021-11-15 00:28:01 +08:00
|
|
|
public function test_all_endpoint_requires_query_parameter()
|
|
|
|
{
|
|
|
|
$resp = $this->actingAsApiEditor()->get($this->baseEndpoint);
|
|
|
|
$resp->assertStatus(422);
|
|
|
|
|
|
|
|
$resp = $this->actingAsApiEditor()->get($this->baseEndpoint . '?query=myqueryvalue');
|
|
|
|
$resp->assertOk();
|
|
|
|
}
|
2024-10-21 05:12:49 +08:00
|
|
|
|
|
|
|
public function test_all_endpoint_includes_book_and_chapter_titles_when_requested()
|
|
|
|
{
|
|
|
|
$this->actingAsApiEditor();
|
|
|
|
|
|
|
|
$book = $this->entities->book();
|
|
|
|
$chapter = $this->entities->chapter();
|
|
|
|
$page = $this->entities->newPage();
|
|
|
|
|
|
|
|
$book->name = 'My Test Book';
|
|
|
|
$book->save();
|
|
|
|
|
|
|
|
$chapter->name = 'My Test Chapter';
|
|
|
|
$chapter->book_id = $book->id;
|
|
|
|
$chapter->save();
|
|
|
|
|
|
|
|
$page->name = 'My Test Page With UniqueSearchTerm';
|
|
|
|
$page->book_id = $book->id;
|
|
|
|
$page->chapter_id = $chapter->id;
|
|
|
|
$page->save();
|
|
|
|
|
|
|
|
$page->indexForSearch();
|
|
|
|
|
|
|
|
// Test without include parameter
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=UniqueSearchTerm');
|
|
|
|
$resp->assertOk();
|
|
|
|
$resp->assertDontSee('book_title');
|
|
|
|
$resp->assertDontSee('chapter_title');
|
|
|
|
|
|
|
|
// Test with include parameter
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=UniqueSearchTerm&include=titles');
|
|
|
|
$resp->assertOk();
|
|
|
|
$resp->assertJsonFragment([
|
|
|
|
'name' => 'My Test Page With UniqueSearchTerm',
|
|
|
|
'book_title' => 'My Test Book',
|
|
|
|
'chapter_title' => 'My Test Chapter',
|
|
|
|
'type' => 'page'
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_all_endpoint_validates_include_parameter()
|
|
|
|
{
|
|
|
|
$this->actingAsApiEditor();
|
|
|
|
|
|
|
|
// Test invalid include value
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=test&include=invalid');
|
|
|
|
$resp->assertOk();
|
|
|
|
$resp->assertDontSee('book_title');
|
|
|
|
|
|
|
|
// Test SQL injection attempt
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=test&include=titles;DROP TABLE users');
|
|
|
|
$resp->assertStatus(422);
|
|
|
|
|
|
|
|
// Test multiple includes
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=test&include=titles,tags');
|
|
|
|
$resp->assertOk();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function test_all_endpoint_includes_tags_when_requested()
|
|
|
|
{
|
|
|
|
$this->actingAsApiEditor();
|
|
|
|
|
|
|
|
// Create a page and give it a unique name for search
|
|
|
|
$page = $this->entities->page();
|
|
|
|
$page->name = 'Page With UniqueSearchTerm';
|
|
|
|
$page->save();
|
|
|
|
|
|
|
|
// Save tags to the page using the existing saveTagsToEntity method
|
|
|
|
$tags = [
|
|
|
|
['name' => 'SampleTag', 'value' => 'SampleValue']
|
|
|
|
];
|
|
|
|
app(\BookStack\Activity\TagRepo::class)->saveTagsToEntity($page, $tags);
|
|
|
|
|
|
|
|
// Ensure the page is indexed for search
|
|
|
|
$page->indexForSearch();
|
|
|
|
|
|
|
|
// Test without the "tags" include
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=UniqueSearchTerm');
|
|
|
|
$resp->assertOk();
|
|
|
|
$resp->assertDontSee('tags');
|
|
|
|
|
|
|
|
// Test with the "tags" include
|
|
|
|
$resp = $this->getJson($this->baseEndpoint . '?query=UniqueSearchTerm&include=tags');
|
|
|
|
$resp->assertOk();
|
|
|
|
|
|
|
|
// Assert that tags are included in the response
|
|
|
|
$resp->assertJsonFragment([
|
|
|
|
'name' => 'SampleTag',
|
|
|
|
'value' => 'SampleValue',
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Optionally: check the structure to match the tag order as well
|
|
|
|
$resp->assertJsonStructure([
|
|
|
|
'data' => [
|
|
|
|
'*' => [
|
|
|
|
'tags' => [
|
|
|
|
'*' => [
|
|
|
|
'name',
|
|
|
|
'value',
|
|
|
|
'order',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
],
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-15 00:28:01 +08:00
|
|
|
}
|