From c7b4c9333233af25dfdec25f10b4a3f2afbf90ca Mon Sep 17 00:00:00 2001 From: Jakub Wolniewicz Date: Mon, 12 Jun 2023 10:33:21 +0200 Subject: [PATCH 01/12] Update index alias on queue --- .../IndexManagement/Job/UpdateIndexAlias.php | 28 +++++++++++++++++++ src/Infrastructure/Console/ElasticUpdate.php | 8 +++++- 2 files changed, 35 insertions(+), 1 deletion(-) create mode 100755 src/Domain/IndexManagement/Job/UpdateIndexAlias.php mode change 100644 => 100755 src/Infrastructure/Console/ElasticUpdate.php diff --git a/src/Domain/IndexManagement/Job/UpdateIndexAlias.php b/src/Domain/IndexManagement/Job/UpdateIndexAlias.php new file mode 100755 index 00000000..3444320b --- /dev/null +++ b/src/Domain/IndexManagement/Job/UpdateIndexAlias.php @@ -0,0 +1,28 @@ +findForIndex($this->index); + $indexAdapter->pointToAlias($indexConfiguration); + } +} \ No newline at end of file diff --git a/src/Infrastructure/Console/ElasticUpdate.php b/src/Infrastructure/Console/ElasticUpdate.php old mode 100644 new mode 100755 index 2b8ac17e..1cff4df4 --- a/src/Infrastructure/Console/ElasticUpdate.php +++ b/src/Infrastructure/Console/ElasticUpdate.php @@ -10,6 +10,7 @@ use JeroenG\Explorer\Domain\IndexManagement\AliasedIndexConfiguration; use JeroenG\Explorer\Domain\IndexManagement\IndexConfigurationInterface; use JeroenG\Explorer\Domain\IndexManagement\IndexConfigurationRepositoryInterface; +use JeroenG\Explorer\Domain\IndexManagement\Job\UpdateIndexAlias; final class ElasticUpdate extends Command { @@ -51,6 +52,11 @@ private function updateIndex( } } - $indexAdapter->pointToAlias($indexConfiguration); + $modelClassName = $indexConfiguration->getModel(); + $model = new $modelClassName(); + + dispatch(new UpdateIndexAlias($indexConfiguration->getName())) + ->onConnection($model->syncWithSearchUsing()) + ->onQueue($model->syncWithSearchUsingQueue()); } } From 4230a8381e69c1429c8881fd61633e1332a7be49 Mon Sep 17 00:00:00 2001 From: Vincent Hagen Date: Wed, 14 Jun 2023 23:21:08 +0200 Subject: [PATCH 02/12] Finalize UpdateIndexAlias job after index update - Add tests for UpdateIndexAlias - DI the dispatcher - use static constructor for job creation --- phpstan-baseline.neon | 5 -- src/Infrastructure/Console/ElasticUpdate.php | 22 +++---- .../IndexManagement/Job/UpdateIndexAlias.php | 19 ++++-- tests/Support/Models/SyncableModel.php | 23 +++++++ .../Job/UpdateIndexAliasTest.php | 60 +++++++++++++++++++ 5 files changed, 109 insertions(+), 20 deletions(-) rename src/{Domain => Infrastructure}/IndexManagement/Job/UpdateIndexAlias.php (53%) create mode 100644 tests/Support/Models/SyncableModel.php create mode 100644 tests/Unit/IndexManagement/Job/UpdateIndexAliasTest.php diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 324b9a21..ff74d5f9 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -10,11 +10,6 @@ parameters: count: 1 path: src/Infrastructure/Console/ElasticSearch.php - - - message: "#^Argument of an invalid type JeroenG\\\\Explorer\\\\Domain\\\\IndexManagement\\\\IndexConfigurationInterface supplied for foreach, only iterables are supported\\.$#" - count: 1 - path: src/Infrastructure/Console/ElasticUpdate.php - - message: "#^Call to an undefined method JeroenG\\\\Explorer\\\\Domain\\\\IndexManagement\\\\IndexConfigurationInterface\\:\\:getAliasConfiguration\\(\\)\\.$#" count: 1 diff --git a/src/Infrastructure/Console/ElasticUpdate.php b/src/Infrastructure/Console/ElasticUpdate.php index 1cff4df4..4df8032d 100755 --- a/src/Infrastructure/Console/ElasticUpdate.php +++ b/src/Infrastructure/Console/ElasticUpdate.php @@ -4,13 +4,14 @@ namespace JeroenG\Explorer\Infrastructure\Console; +use Illuminate\Bus\Dispatcher; use Illuminate\Console\Command; use Illuminate\Support\Facades\Artisan; use JeroenG\Explorer\Application\IndexAdapterInterface; use JeroenG\Explorer\Domain\IndexManagement\AliasedIndexConfiguration; use JeroenG\Explorer\Domain\IndexManagement\IndexConfigurationInterface; use JeroenG\Explorer\Domain\IndexManagement\IndexConfigurationRepositoryInterface; -use JeroenG\Explorer\Domain\IndexManagement\Job\UpdateIndexAlias; +use JeroenG\Explorer\Infrastructure\IndexManagement\Job\UpdateIndexAlias; final class ElasticUpdate extends Command { @@ -20,16 +21,17 @@ final class ElasticUpdate extends Command public function handle( IndexAdapterInterface $indexAdapter, - IndexConfigurationRepositoryInterface $indexConfigurationRepository + IndexConfigurationRepositoryInterface $indexConfigurationRepository, + Dispatcher $dispatcher ): int { $index = $this->argument('index'); - /** @var IndexConfigurationInterface $allConfigs */ + /** @var IndexConfigurationInterface[] $allConfigs */ $allConfigs = is_null($index) ? $indexConfigurationRepository->getConfigurations() : [$indexConfigurationRepository->findForIndex($index)]; foreach ($allConfigs as $config) { - $this->updateIndex($config, $indexAdapter); + $this->updateIndex($config, $indexAdapter, $dispatcher); } return 0; @@ -37,7 +39,8 @@ public function handle( private function updateIndex( IndexConfigurationInterface $indexConfiguration, - IndexAdapterInterface $indexAdapter + IndexAdapterInterface $indexAdapter, + Dispatcher $dispatcher, ): void { if ($indexConfiguration instanceof AliasedIndexConfiguration) { $indexAdapter->createNewWriteIndex($indexConfiguration); @@ -52,11 +55,8 @@ private function updateIndex( } } - $modelClassName = $indexConfiguration->getModel(); - $model = new $modelClassName(); - - dispatch(new UpdateIndexAlias($indexConfiguration->getName())) - ->onConnection($model->syncWithSearchUsing()) - ->onQueue($model->syncWithSearchUsingQueue()); + $dispatcher->dispatch( + UpdateIndexAlias::createFor($indexConfiguration) + ); } } diff --git a/src/Domain/IndexManagement/Job/UpdateIndexAlias.php b/src/Infrastructure/IndexManagement/Job/UpdateIndexAlias.php similarity index 53% rename from src/Domain/IndexManagement/Job/UpdateIndexAlias.php rename to src/Infrastructure/IndexManagement/Job/UpdateIndexAlias.php index 3444320b..e3ab73ef 100755 --- a/src/Domain/IndexManagement/Job/UpdateIndexAlias.php +++ b/src/Infrastructure/IndexManagement/Job/UpdateIndexAlias.php @@ -1,6 +1,6 @@ -getModel(); + $model = new $modelClassName(); + + return (new self($indexConfiguration->getName())) + ->onQueue($model->syncWithSearchUsingQueue()) + ->onConnection($model->syncWithSearchUsing()); + } + public function handle( IndexAdapterInterface $indexAdapter, IndexConfigurationRepositoryInterface $indexConfigurationRepository diff --git a/tests/Support/Models/SyncableModel.php b/tests/Support/Models/SyncableModel.php new file mode 100644 index 00000000..d682b70a --- /dev/null +++ b/tests/Support/Models/SyncableModel.php @@ -0,0 +1,23 @@ +queue); + Assert::assertSame(':connection:', $subject->connection); + Assert::assertSame(':index:', $subject->index); + + } + + public function testHandleCallsPointToIndex(): void + { + $adapter = Mockery::mock(IndexAdapterInterface::class); + $repository = Mockery::mock(IndexConfigurationRepositoryInterface::class); + + $config = DirectIndexConfiguration::create( + name: ':index:', + properties: [], + settings: [], + model: SyncableModel::class, + ); + + $repository + ->expects('findForIndex') + ->with(':index:') + ->andReturn($config); + + $adapter + ->expects('pointToAlias') + ->with($config); + + UpdateIndexAlias::createFor($config) + ->handle($adapter, $repository); + + } +} From 294b6787c1a7a8a09e59196bcab4dadb2a7cca59 Mon Sep 17 00:00:00 2001 From: Ar0010r Date: Tue, 20 Jun 2023 15:21:44 +0300 Subject: [PATCH 03/12] allow custom order by --- src/Infrastructure/Scout/ScoutSearchCommandBuilder.php | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php b/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php index 5611450f..0c5ff872 100644 --- a/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php +++ b/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php @@ -12,6 +12,7 @@ use JeroenG\Explorer\Domain\Syntax\Compound\QueryType; use JeroenG\Explorer\Domain\Syntax\MultiMatch; use JeroenG\Explorer\Domain\Syntax\Sort; +use JeroenG\Explorer\Domain\Syntax\SyntaxInterface; use JeroenG\Explorer\Domain\Syntax\Term; use JeroenG\Explorer\Domain\Syntax\Terms; use Laravel\Scout\Builder; @@ -33,7 +34,7 @@ class ScoutSearchCommandBuilder implements SearchCommandInterface private ?string $minimumShouldMatch = null; - /** @var Sort[] */ + /** @var SyntaxInterface[] */ private array $sort = []; private array $aggregations = []; @@ -207,7 +208,7 @@ public function setLimit(?int $limit): void public function setSort(array $sort): void { - Assert::allIsInstanceOf($sort, Sort::class); + Assert::allIsInstanceOf($sort, SyntaxInterface::class); $this->sort = $sort; } @@ -307,6 +308,8 @@ public function getOffset(): ?int /** @return Sort[] */ private static function getSorts(Builder $builder): array { - return array_map(static fn ($order) => new Sort($order['column'], $order['direction']), $builder->orders); + return array_map(static function($order) { + return $order instanceof SyntaxInterface ? $order : new Sort($order['column'], $order['direction']); + }, $builder->orders); } } From 765ce8e2100dd99d9aeeea2315ec3839b9657c0b Mon Sep 17 00:00:00 2001 From: Ar0010r Date: Tue, 20 Jun 2023 21:11:24 +0300 Subject: [PATCH 04/12] fix test_it_only_accepts_sort_classes test and rename it to test_it_only_accepts_syntax_interface_classes --- tests/Unit/ScoutSearchCommandBuilderTest.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/Unit/ScoutSearchCommandBuilderTest.php b/tests/Unit/ScoutSearchCommandBuilderTest.php index 3e78f3da..41164c5e 100644 --- a/tests/Unit/ScoutSearchCommandBuilderTest.php +++ b/tests/Unit/ScoutSearchCommandBuilderTest.php @@ -140,12 +140,12 @@ public function test_it_can_set_the_sort_order(): void $command->setSort([new Sort('id', 'invalid')]); } - public function test_it_only_accepts_sort_classes(): void + public function test_it_only_accepts_syntax_interface_classes(): void { $command = new ScoutSearchCommandBuilder(); $this->expectException(InvalidArgumentException::class); - $this->expectExceptionMessage('Expected an instance of JeroenG\Explorer\Domain\Syntax\Sort. Got: string'); + $this->expectExceptionMessage('Expected an instance of JeroenG\Explorer\Domain\Syntax\SyntaxInterface. Got: string'); $command->setSort(['not' => 'a class']); } From 50c92b114aa948483bd605d74b09080253268f96 Mon Sep 17 00:00:00 2001 From: Ar0010r Date: Tue, 20 Jun 2023 21:12:32 +0300 Subject: [PATCH 05/12] fix phpdoc for getSorts() method --- src/Infrastructure/Scout/ScoutSearchCommandBuilder.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php b/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php index 0c5ff872..8c29493f 100644 --- a/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php +++ b/src/Infrastructure/Scout/ScoutSearchCommandBuilder.php @@ -305,7 +305,7 @@ public function getOffset(): ?int return $this->offset; } - /** @return Sort[] */ + /** @return SyntaxInterface[] */ private static function getSorts(Builder $builder): array { return array_map(static function($order) { From a46a0d5c5fea61bf4c5c48eec152ddb640d52a64 Mon Sep 17 00:00:00 2001 From: Jeroen Date: Fri, 30 Jun 2023 20:19:29 +0200 Subject: [PATCH 06/12] update changelog --- changelog.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/changelog.md b/changelog.md index 8d4c7847..10c6a669 100644 --- a/changelog.md +++ b/changelog.md @@ -6,6 +6,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## Unreleased +### Added +- Updating the index alias is now done through a (queueable) job. + ## [3.7.0] ### Added From 49f512d760b22ebf79d5bbc393349ae7c9afdd9a Mon Sep 17 00:00:00 2001 From: Jeroen Date: Fri, 30 Jun 2023 20:24:57 +0200 Subject: [PATCH 07/12] update docs --- docs/index-aliases.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/docs/index-aliases.md b/docs/index-aliases.md index 338d4f6e..f5d3c873 100644 --- a/docs/index-aliases.md +++ b/docs/index-aliases.md @@ -54,3 +54,11 @@ return [ Be aware that if you currently already have indices and would like to move to using aliases you will need to delete those indices before configuring the aliases. In Elasticsearch a given name can only be either an index or alias, not both and this cannot be changed on-the-fly. + +### Note on updating aliases +When you update a model, Laravel Scouts will update the index. +When you use index aliases, a new index is created and the alias is being pointed to the nex one. +What you don't want is for the alias to be pointing to the new index before Elasticsearch is done with indexing all documents. +To prevent this, the alias update is done in a job that is dispatched to the queue. +If there is no queue it will still be done in the background, but it will be done synchronously. +This could still be enough of a "delay" for Elasticsearch to finish indexing, so there is no immediate need to set up a queue. From ff34a4e3313bb74491909bdf7b744194a079682a Mon Sep 17 00:00:00 2001 From: Ar0010r Date: Fri, 28 Jul 2023 00:34:52 +0300 Subject: [PATCH 08/12] adjust test_it_can_get_the_sorting_from_the_scout_builder test --- tests/Unit/ScoutSearchCommandBuilderTest.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/Unit/ScoutSearchCommandBuilderTest.php b/tests/Unit/ScoutSearchCommandBuilderTest.php index 41164c5e..93d843e1 100644 --- a/tests/Unit/ScoutSearchCommandBuilderTest.php +++ b/tests/Unit/ScoutSearchCommandBuilderTest.php @@ -174,11 +174,11 @@ public function test_it_can_get_the_sorting_from_the_scout_builder(): void $builder->model = Mockery::mock(Model::class); $builder->index = self::TEST_INDEX; - $builder->orders = [[ 'column' => 'id', 'direction' => 'asc']]; + $builder->orders = [['column' => 'id', 'direction' => 'asc'], new Sort('name')]; $subject = ScoutSearchCommandBuilder::wrap($builder); - self::assertSame([['id' => 'asc']], $subject->getSort()); + self::assertSame([['id' => 'asc'], ['name' => 'asc']], $subject->getSort()); } public function test_it_can_get_the_fields_from_scout_builder(): void From 104ff706356e792db78f71275d4cb9c0a589d5fe Mon Sep 17 00:00:00 2001 From: Menno Braam Date: Tue, 12 Sep 2023 20:06:22 +0200 Subject: [PATCH 09/12] Add support for getting aggregations on nested aggregation results --- src/Application/Results.php | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/src/Application/Results.php b/src/Application/Results.php index 3eda4280..7d2d8b45 100644 --- a/src/Application/Results.php +++ b/src/Application/Results.php @@ -30,6 +30,15 @@ public function aggregations(): array $aggregations = []; foreach ($this->rawResults['aggregations'] as $name => $rawAggregation) { + if (array_key_exists('doc_count', $rawAggregation)) { + foreach ($rawAggregation as $nestedAggregationName => $rawNestedAggregation) { + if (isset($rawNestedAggregation['buckets'])) { + $aggregations[] = new AggregationResult($nestedAggregationName, $rawNestedAggregation['buckets']); + } + } + continue; + } + $aggregations[] = new AggregationResult($name, $rawAggregation['buckets']); } From 7fc6e30f3e23b52078d3a47bfe5587b848c5c6ed Mon Sep 17 00:00:00 2001 From: Menno Braam Date: Wed, 13 Sep 2023 10:51:54 +0200 Subject: [PATCH 10/12] Add test coverage for nested aggregation results --- tests/Unit/FinderTest.php | 77 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 77 insertions(+) diff --git a/tests/Unit/FinderTest.php b/tests/Unit/FinderTest.php index c7cec018..9f3a23bf 100644 --- a/tests/Unit/FinderTest.php +++ b/tests/Unit/FinderTest.php @@ -18,6 +18,7 @@ use JeroenG\Explorer\Infrastructure\Scout\ScoutSearchCommandBuilder; use Mockery; use Mockery\Adapter\Phpunit\MockeryTestCase; +use JeroenG\Explorer\Domain\Aggregations\NestedAggregation; class FinderTest extends MockeryTestCase { @@ -362,6 +363,82 @@ public function test_it_adds_aggregates(): void self::assertEquals('myKey', $specificAggregationValue['key']); } + public function test_it_adds_nested_aggregations(): void + { + $client = Mockery::mock(Client::class); + $client->expects('search') + ->with([ + 'index' => self::TEST_INDEX, + 'body' => [ + 'query' => [ + 'bool' => [ + 'must' => [], + 'should' => [], + 'filter' => [], + ], + ], + 'aggs' => [ + 'nestedAggregation' => [ + 'nested' => [ + 'path' => 'nestedAggregation', + ], + 'aggs' => [ + 'someField' => [ + 'terms' => [ + 'field' => 'nestedAggregation.someField', + 'size' => 10, + ], + ], + ], + ], + ], + ], + ]) + ->andReturn([ + 'hits' => [ + 'total' => ['value' => 1], + 'hits' => [$this->hit()], + ], + 'aggregations' => [ + 'nestedAggregation' => [ + 'doc_count' => 42, + 'someField' => [ + 'doc_count_error_upper_bound' => 0, + 'sum_other_doc_count' => 0, + 'buckets' => [ + ['key' => 'someKey', 'doc_count' => 6,] + ], + ], + ], + ], + ]); + + $query = Query::with(new BoolQuery()); + $nestedAggregation = new NestedAggregation('nestedAggregation'); + $nestedAggregation->add('someField', new TermsAggregation('nestedAggregation.someField')); + $query->addAggregation('nestedAggregation',$nestedAggregation); + $builder = new SearchCommand(self::TEST_INDEX, $query); + $builder->setIndex(self::TEST_INDEX); + + $subject = new Finder($client, $builder); + $results = $subject->find(); + + self::assertCount(1, $results->aggregations()); + + $nestedAggregation = $results->aggregations()[0]; + + self::assertInstanceOf(AggregationResult::class, $nestedAggregation); + self::assertEquals('someField', $nestedAggregation->name()); + self::assertCount(1, $nestedAggregation->values()); + + $nestedAggregationValue = $nestedAggregation->values()[0]; + + + self::assertEquals(6, $nestedAggregationValue['doc_count']); + self::assertEquals('someKey', $nestedAggregationValue['key']); + } + + private function hit(int $id = 1, float $score = 1.0): array { return [ From d29b2d406ba50c63bb2ad7476a8fb91517cf4627 Mon Sep 17 00:00:00 2001 From: Menno Braam Date: Wed, 13 Sep 2023 11:03:37 +0200 Subject: [PATCH 11/12] Codestyle --- tests/Unit/FinderTest.php | 2 -- 1 file changed, 2 deletions(-) diff --git a/tests/Unit/FinderTest.php b/tests/Unit/FinderTest.php index 9f3a23bf..c12ca50f 100644 --- a/tests/Unit/FinderTest.php +++ b/tests/Unit/FinderTest.php @@ -433,12 +433,10 @@ public function test_it_adds_nested_aggregations(): void $nestedAggregationValue = $nestedAggregation->values()[0]; - self::assertEquals(6, $nestedAggregationValue['doc_count']); self::assertEquals('someKey', $nestedAggregationValue['key']); } - private function hit(int $id = 1, float $score = 1.0): array { return [ From c3c497f17058e3022138e3c92c7ce90177b45606 Mon Sep 17 00:00:00 2001 From: Menno Braam Date: Wed, 13 Sep 2023 12:42:50 +0200 Subject: [PATCH 12/12] Catch the escaped mutation --- tests/Unit/FinderTest.php | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/tests/Unit/FinderTest.php b/tests/Unit/FinderTest.php index c12ca50f..c0f96ee2 100644 --- a/tests/Unit/FinderTest.php +++ b/tests/Unit/FinderTest.php @@ -391,6 +391,7 @@ public function test_it_adds_nested_aggregations(): void ], ], ], + 'anotherAggregation' => ['terms' => ['field' => 'anotherField', 'size' => 10]] ], ], ]) @@ -410,10 +411,16 @@ public function test_it_adds_nested_aggregations(): void ], ], ], + 'specificAggregation' => [ + 'buckets' => [ + ['key' => 'myKey', 'doc_count' => 42] + ] + ], ], ]); $query = Query::with(new BoolQuery()); + $query->addAggregation('anotherAggregation', new TermsAggregation('anotherField')); $nestedAggregation = new NestedAggregation('nestedAggregation'); $nestedAggregation->add('someField', new TermsAggregation('nestedAggregation.someField')); $query->addAggregation('nestedAggregation',$nestedAggregation); @@ -423,7 +430,7 @@ public function test_it_adds_nested_aggregations(): void $subject = new Finder($client, $builder); $results = $subject->find(); - self::assertCount(1, $results->aggregations()); + self::assertCount(2, $results->aggregations()); $nestedAggregation = $results->aggregations()[0];