mirror of
https://github.com/BookStackApp/BookStack.git
synced 2024-12-13 14:33:37 +08:00
Search Index: Fixed SQL error when indexing large pages
Due to hitting statement placeholder limits (typically 65k) when inserting index terms for single page. Added test to cover. Also added skipped tests for tests we don't always want to run. For #5322
This commit is contained in:
parent
5632fef621
commit
509af2463d
|
@ -30,7 +30,7 @@ class SearchIndex
|
||||||
{
|
{
|
||||||
$this->deleteEntityTerms($entity);
|
$this->deleteEntityTerms($entity);
|
||||||
$terms = $this->entityToTermDataArray($entity);
|
$terms = $this->entityToTermDataArray($entity);
|
||||||
SearchTerm::query()->insert($terms);
|
$this->insertTerms($terms);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -46,10 +46,7 @@ class SearchIndex
|
||||||
array_push($terms, ...$entityTerms);
|
array_push($terms, ...$entityTerms);
|
||||||
}
|
}
|
||||||
|
|
||||||
$chunkedTerms = array_chunk($terms, 500);
|
$this->insertTerms($terms);
|
||||||
foreach ($chunkedTerms as $termChunk) {
|
|
||||||
SearchTerm::query()->insert($termChunk);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -99,6 +96,19 @@ class SearchIndex
|
||||||
$entity->searchTerms()->delete();
|
$entity->searchTerms()->delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Insert the given terms into the database.
|
||||||
|
* Chunks through the given terms to remain within database limits.
|
||||||
|
* @param array[] $terms
|
||||||
|
*/
|
||||||
|
protected function insertTerms(array $terms): void
|
||||||
|
{
|
||||||
|
$chunkedTerms = array_chunk($terms, 500);
|
||||||
|
foreach ($chunkedTerms as $termChunk) {
|
||||||
|
SearchTerm::query()->insert($termChunk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a scored term array from the given text, where the keys are the terms
|
* Create a scored term array from the given text, where the keys are the terms
|
||||||
* and the values are their scores.
|
* and the values are their scores.
|
||||||
|
|
|
@ -6,6 +6,7 @@ use BookStack\Activity\Models\Tag;
|
||||||
use BookStack\Entities\Models\Book;
|
use BookStack\Entities\Models\Book;
|
||||||
use BookStack\Entities\Models\Bookshelf;
|
use BookStack\Entities\Models\Bookshelf;
|
||||||
use BookStack\Entities\Models\Chapter;
|
use BookStack\Entities\Models\Chapter;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
use Tests\TestCase;
|
use Tests\TestCase;
|
||||||
|
|
||||||
class EntitySearchTest extends TestCase
|
class EntitySearchTest extends TestCase
|
||||||
|
@ -477,6 +478,25 @@ class EntitySearchTest extends TestCase
|
||||||
$this->assertEquals(2, $scoreByTerm->get('TermG'));
|
$this->assertEquals(2, $scoreByTerm->get('TermG'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function test_indexing_works_as_expected_for_page_with_lots_of_terms()
|
||||||
|
{
|
||||||
|
$this->markTestSkipped('Time consuming test');
|
||||||
|
|
||||||
|
$count = 100000;
|
||||||
|
$text = '';
|
||||||
|
$chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_#';
|
||||||
|
for ($i = 0; $i < $count; $i++) {
|
||||||
|
$text .= substr(str_shuffle($chars), 0, 5) . ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
$page = $this->entities->newPage(['name' => 'Test page A', 'html' => '<p>' . $text . '</p>']);
|
||||||
|
|
||||||
|
$termCount = $page->searchTerms()->count();
|
||||||
|
|
||||||
|
// Expect at least 90% unique rate
|
||||||
|
$this->assertGreaterThan($count * 0.9, $termCount);
|
||||||
|
}
|
||||||
|
|
||||||
public function test_name_and_content_terms_are_merged_to_single_score()
|
public function test_name_and_content_terms_are_merged_to_single_score()
|
||||||
{
|
{
|
||||||
$page = $this->entities->newPage(['name' => 'TermA', 'html' => '
|
$page = $this->entities->newPage(['name' => 'TermA', 'html' => '
|
||||||
|
|
|
@ -27,8 +27,10 @@ class LanguageTest extends TestCase
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not part of standard phpunit test runs since we sometimes expect non-added langs.
|
// Not part of standard phpunit test runs since we sometimes expect non-added langs.
|
||||||
public function do_test_locales_all_have_language_dropdown_entry()
|
public function test_locales_all_have_language_dropdown_entry()
|
||||||
{
|
{
|
||||||
|
$this->markTestSkipped('Only used when checking language inclusion');
|
||||||
|
|
||||||
$dropdownLocales = array_keys(trans('settings.language_select', [], 'en'));
|
$dropdownLocales = array_keys(trans('settings.language_select', [], 'en'));
|
||||||
sort($dropdownLocales);
|
sort($dropdownLocales);
|
||||||
sort($this->langs);
|
sort($this->langs);
|
||||||
|
|
Loading…
Reference in New Issue
Block a user