mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-04-02 21:59:06 +08:00
Made search more efficient and tweaked weighting
Added per-entity weighting changes. Now Books score higher than chapters which score higher than pages. Reduced queries required on search by only searching once but at a higher count to see if there's another page.
This commit is contained in:
parent
3df7d828eb
commit
f101c1a622
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
class Book extends Entity
|
class Book extends Entity
|
||||||
{
|
{
|
||||||
|
public $searchFactor = 2;
|
||||||
|
|
||||||
protected $fillable = ['name', 'description', 'image_id'];
|
protected $fillable = ['name', 'description', 'image_id'];
|
||||||
|
|
||||||
|
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
class Chapter extends Entity
|
class Chapter extends Entity
|
||||||
{
|
{
|
||||||
|
public $searchFactor = 1.3;
|
||||||
|
|
||||||
protected $fillable = ['name', 'description', 'priority', 'book_id'];
|
protected $fillable = ['name', 'description', 'priority', 'book_id'];
|
||||||
|
|
||||||
protected $with = ['book'];
|
protected $with = ['book'];
|
||||||
|
@ -5,8 +5,16 @@ use Illuminate\Database\Eloquent\Relations\MorphMany;
|
|||||||
class Entity extends Ownable
|
class Entity extends Ownable
|
||||||
{
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string - Name of property where the main text content is found
|
||||||
|
*/
|
||||||
public $textField = 'description';
|
public $textField = 'description';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var float - Multiplier for search indexing.
|
||||||
|
*/
|
||||||
|
public $searchFactor = 1.0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compares this entity to another given entity.
|
* Compares this entity to another given entity.
|
||||||
* Matches by comparing class and id.
|
* Matches by comparing class and id.
|
||||||
@ -193,4 +201,5 @@ class Entity extends Ownable
|
|||||||
{
|
{
|
||||||
return '/';
|
return '/';
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -40,13 +40,12 @@ class SearchController extends Controller
|
|||||||
$nextPageLink = baseUrl('/search?term=' . urlencode($searchTerm) . '&page=' . ($page+1));
|
$nextPageLink = baseUrl('/search?term=' . urlencode($searchTerm) . '&page=' . ($page+1));
|
||||||
|
|
||||||
$results = $this->searchService->searchEntities($searchTerm, 'all', $page, 20);
|
$results = $this->searchService->searchEntities($searchTerm, 'all', $page, 20);
|
||||||
$hasNextPage = $this->searchService->searchEntities($searchTerm, 'all', $page+1, 20)['count'] > 0;
|
|
||||||
|
|
||||||
return view('search/all', [
|
return view('search/all', [
|
||||||
'entities' => $results['results'],
|
'entities' => $results['results'],
|
||||||
'totalResults' => $results['total'],
|
'totalResults' => $results['total'],
|
||||||
'searchTerm' => $searchTerm,
|
'searchTerm' => $searchTerm,
|
||||||
'hasNextPage' => $hasNextPage,
|
'hasNextPage' => $results['has_more'],
|
||||||
'nextPageLink' => $nextPageLink
|
'nextPageLink' => $nextPageLink
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
@ -72,7 +72,6 @@ class SearchService
|
|||||||
$terms = $this->parseSearchString($searchString);
|
$terms = $this->parseSearchString($searchString);
|
||||||
$entityTypes = array_keys($this->entities);
|
$entityTypes = array_keys($this->entities);
|
||||||
$entityTypesToSearch = $entityTypes;
|
$entityTypesToSearch = $entityTypes;
|
||||||
$results = collect();
|
|
||||||
|
|
||||||
if ($entityType !== 'all') {
|
if ($entityType !== 'all') {
|
||||||
$entityTypesToSearch = $entityType;
|
$entityTypesToSearch = $entityType;
|
||||||
@ -80,21 +79,23 @@ class SearchService
|
|||||||
$entityTypesToSearch = explode('|', $terms['filters']['type']);
|
$entityTypesToSearch = explode('|', $terms['filters']['type']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$results = collect();
|
||||||
$total = 0;
|
$total = 0;
|
||||||
|
|
||||||
foreach ($entityTypesToSearch as $entityType) {
|
foreach ($entityTypesToSearch as $entityType) {
|
||||||
if (!in_array($entityType, $entityTypes)) {
|
if (!in_array($entityType, $entityTypes)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$search = $this->searchEntityTable($terms, $entityType, $page, $count);
|
$search = $this->searchEntityTable($terms, $entityType, $page, $count + 1);
|
||||||
$total += $this->searchEntityTable($terms, $entityType, $page, $count, true);
|
$total += $this->searchEntityTable($terms, $entityType, $page, $count + 1, true);
|
||||||
$results = $results->merge($search);
|
$results = $results->merge($search);
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'total' => $total,
|
'total' => $total,
|
||||||
'count' => count($results),
|
'count' => count($results),
|
||||||
'results' => $results->sortByDesc('score')->values()
|
'has_more' => $results->count() > $count,
|
||||||
|
'results' => $results->sortByDesc('score')->slice(0, $count)->values()
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -322,8 +323,8 @@ class SearchService
|
|||||||
public function indexEntity(Entity $entity)
|
public function indexEntity(Entity $entity)
|
||||||
{
|
{
|
||||||
$this->deleteEntityTerms($entity);
|
$this->deleteEntityTerms($entity);
|
||||||
$nameTerms = $this->generateTermArrayFromText($entity->name, 5);
|
$nameTerms = $this->generateTermArrayFromText($entity->name, 5 * $entity->searchFactor);
|
||||||
$bodyTerms = $this->generateTermArrayFromText($entity->getText(), 1);
|
$bodyTerms = $this->generateTermArrayFromText($entity->getText(), 1 * $entity->searchFactor);
|
||||||
$terms = array_merge($nameTerms, $bodyTerms);
|
$terms = array_merge($nameTerms, $bodyTerms);
|
||||||
foreach ($terms as $index => $term) {
|
foreach ($terms as $index => $term) {
|
||||||
$terms[$index]['entity_type'] = $entity->getMorphClass();
|
$terms[$index]['entity_type'] = $entity->getMorphClass();
|
||||||
@ -340,8 +341,8 @@ class SearchService
|
|||||||
{
|
{
|
||||||
$terms = [];
|
$terms = [];
|
||||||
foreach ($entities as $entity) {
|
foreach ($entities as $entity) {
|
||||||
$nameTerms = $this->generateTermArrayFromText($entity->name, 5);
|
$nameTerms = $this->generateTermArrayFromText($entity->name, 5 * $entity->searchFactor);
|
||||||
$bodyTerms = $this->generateTermArrayFromText($entity->getText(), 1);
|
$bodyTerms = $this->generateTermArrayFromText($entity->getText(), 1 * $entity->searchFactor);
|
||||||
foreach (array_merge($nameTerms, $bodyTerms) as $term) {
|
foreach (array_merge($nameTerms, $bodyTerms) as $term) {
|
||||||
$term['entity_id'] = $entity->id;
|
$term['entity_id'] = $entity->id;
|
||||||
$term['entity_type'] = $entity->getMorphClass();
|
$term['entity_type'] = $entity->getMorphClass();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user