コンテンツへスキップ

Laravel Scout

イントロダクション

Laravel Scoutは、Eloquentモデルに全文検索を追加するための、シンプルでドライバベースのソリューションを提供します。モデルオブザーバを使用し、Scoutは自動的に検索インデックスとEloquentレコードを同期させます。

現在、ScoutはAlgoliaMeilisearchTypesense、MySQL / PostgreSQL(database)ドライバを同梱しています。さらに、Scoutはローカル開発での使用を目的とし、外部の依存関係やサードパーティのサービスを必要としない「コレクション」ドライバを含んでいます。さらに、カスタムドライバの作成は簡単で、独自の検索実装でScoutを自由に拡張できます。

インストール

まず、ComposerパッケージマネージャでScoutをインストールしてください。

1composer require laravel/scout

Scoutをインストールした後、vendor:publish Artisanコマンドを使用してScout設定ファイルを公開する必要があります。このコマンドは、scout.php設定ファイルをアプリケーションのconfigディレクトリに公開します。

1php artisan vendor:publish --provider="Laravel\Scout\ScoutServiceProvider"

最後に、検索可能にしたいモデルにLaravel\Scout\Searchableトレイトを追加します。このトレイトは、モデルを検索ドライバと自動的に同期させるモデルオブザーバを登録します。

1<?php
2 
3namespace App\Models;
4 
5use Illuminate\Database\Eloquent\Model;
6use Laravel\Scout\Searchable;
7 
8class Post extends Model
9{
10 use Searchable;
11}

キュー

Scoutを使用するために厳密には必須ではありませんが、ライブラリを使用する前にキュードライバを設定することを強く推奨します。キューワーカを実行すると、Scoutはモデル情報を検索インデックスに同期するすべての操作をキューに入れることができ、アプリケーションのWebインターフェイスの応答時間が大幅に向上します。

キュードライバを設定したら、config/scout.php設定ファイルでqueueオプションの値をtrueに設定します。

1'queue' => true,

queueオプションをfalseに設定した場合でも、AlgoliaやMeilisearchのような一部のScoutドライバは常にレコードを非同期にインデックスすることを覚えておくことが重要です。つまり、Laravelアプリケーション内でインデックス操作が完了したとしても、検索エンジン自体が新しいレコードや更新されたレコードをすぐに反映しない場合があります。

Scoutジョブが利用する接続とキューを指定するには、queue設定オプションを配列として定義します。

1'queue' => [
2 'connection' => 'redis',
3 'queue' => 'scout'
4],

もちろん、Scoutジョブが利用する接続とキューをカスタマイズした場合は、その接続とキューでジョブを処理するためにキューワーカを実行する必要があります。

1php artisan queue:work redis --queue=scout

ドライバの前提条件

Algolia

Algoliaドライバを使用する場合は、config/scout.php設定ファイルでAlgoliaのidsecretの認証情報を設定する必要があります。認証情報を設定したら、Composerパッケージマネージャを介してAlgolia PHP SDKもインストールする必要があります。

1composer require algolia/algoliasearch-client-php

Meilisearch

Meilisearchは、驚くほど高速なオープンソースの検索エンジンです。ローカルマシンにMeilisearchをインストールする方法がわからない場合は、Laravelの公式にサポートしているDocker開発環境であるLaravel Sailを使用できます。

Meilisearchドライバを使用する場合は、Composerパッケージマネージャを介してMeilisearch PHP SDKをインストールする必要があります。

1composer require meilisearch/meilisearch-php http-interop/http-factory-guzzle

次に、SCOUT_DRIVER環境変数を設定し、アプリケーションの.envファイル内でMeilisearchのhostkeyの認証情報を設定します。

1SCOUT_DRIVER=meilisearch
2MEILISEARCH_HOST=http://127.0.0.1:7700
3MEILISEARCH_KEY=masterKey

Meilisearchに関するより詳しい情報は、Meilisearchのドキュメントを参照してください。

さらに、バイナリの互換性に関するMeilisearchのドキュメントを確認し、Meilisearchバイナリのバージョンと互換性のあるバージョンのmeilisearch/meilisearch-phpをインストールしていることを確認する必要があります。

Meilisearchを利用するアプリケーションでScoutをアップグレードする場合は、Meilisearchサービス自体の追加の破壊的変更を常に確認する必要があります。

Typesense

Typesenseは、超高速のオープンソース検索エンジンで、キーワード検索、セマンティック検索、地理検索、ベクトル検索をサポートしています。

Typesenseはセルフホストするか、Typesense Cloudを使用できます。

ScoutでTypesenseを使い始めるには、ComposerパッケージマネージャでTypesense PHP SDKをインストールします。

1composer require typesense/typesense-php

次に、`SCOUT_DRIVER`環境変数を設定し、アプリケーションの.envファイル内でTypesenseのホストとAPIキーの認証情報を設定します。

1SCOUT_DRIVER=typesense
2TYPESENSE_API_KEY=masterKey
3TYPESENSE_HOST=localhost

Laravel Sailを使用している場合は、TYPESENSE_HOST環境変数をDockerコンテナ名に合わせて調整する必要があるかもしれません。オプションで、インストールのポート、パス、プロトコルを指定することもできます。

1TYPESENSE_PORT=8108
2TYPESENSE_PATH=
3TYPESENSE_PROTOCOL=http

Typesenseコレクションの追加設定やスキーマ定義は、アプリケーションのconfig/scout.php設定ファイル内にあります。Typesenseに関するより詳しい情報は、Typesenseのドキュメントを参照してください。

Typesenseでの保存のためのデータ準備

Typesenseを利用する場合、検索可能なモデルは、モデルのプライマリキーを文字列に、作成日時をUNIXタイムスタンプにキャストするtoSearchableArrayメソッドを定義する必要があります。

1/**
2 * Get the indexable data array for the model.
3 *
4 * @return array<string, mixed>
5 */
6public function toSearchableArray()
7{
8 return array_merge($this->toArray(),[
9 'id' => (string) $this->id,
10 'created_at' => $this->created_at->timestamp,
11 ]);
12}

また、アプリケーションのconfig/scout.phpファイルでTypesenseコレクションのスキーマを定義する必要があります。コレクションスキーマは、Typesenseで検索可能な各フィールドのデータ型を記述します。利用可能なすべてのスキーマオプションについての詳細は、Typesenseのドキュメントを参照してください。

定義後にTypesenseコレクションのスキーマを変更する必要がある場合は、scout:flushscout:importを実行して、既存のインデックスデータをすべて削除してスキーマを再作成するか、TypesenseのAPIを使用してインデックスデータを削除せずにコレクションのスキーマを変更できます。

検索可能なモデルがソフトデリート可能な場合、アプリケーションのconfig/scout.php設定ファイル内にある、モデルに対応するTypesenseスキーマで__soft_deletedフィールドを定義する必要があります。

1User::class => [
2 'collection-schema' => [
3 'fields' => [
4 // ...
5 [
6 'name' => '__soft_deleted',
7 'type' => 'int32',
8 'optional' => true,
9 ],
10 ],
11 ],
12],

動的検索パラメータ

Typesenseでは、optionsメソッドを介して検索操作を実行する際に、検索パラメータを動的に変更できます。

1use App\Models\Todo;
2 
3Todo::search('Groceries')->options([
4 'query_by' => 'title, description'
5])->get();

設定

モデルインデックスの設定

各Eloquentモデルは、そのモデルのすべての検索可能なレコードを含む、特定の検索「インデックス」と同期されます。言い換えれば、各インデックスはMySQLのテーブルのようなものだと考えることができます。デフォルトでは、各モデルはそのモデルの典型的な「テーブル」名に一致するインデックスに永続化されます。通常、これはモデル名の複数形ですが、モデルのsearchableAsメソッドをオーバーライドすることで、モデルのインデックスを自由にカスタマイズできます。

1<?php
2 
3namespace App\Models;
4 
5use Illuminate\Database\Eloquent\Model;
6use Laravel\Scout\Searchable;
7 
8class Post extends Model
9{
10 use Searchable;
11 
12 /**
13 * Get the name of the index associated with the model.
14 */
15 public function searchableAs(): string
16 {
17 return 'posts_index';
18 }
19}

検索対象データの設定

デフォルトでは、特定のモデルのtoArray形式全体が検索インデックスに永続化されます。検索インデックスに同期されるデータをカスタマイズしたい場合は、モデルのtoSearchableArrayメソッドをオーバーライドします。

1<?php
2 
3namespace App\Models;
4 
5use Illuminate\Database\Eloquent\Model;
6use Laravel\Scout\Searchable;
7 
8class Post extends Model
9{
10 use Searchable;
11 
12 /**
13 * Get the indexable data array for the model.
14 *
15 * @return array<string, mixed>
16 */
17 public function toSearchableArray(): array
18 {
19 $array = $this->toArray();
20 
21 // Customize the data array...
22 
23 return $array;
24 }
25}

Meilisearchなどの一部の検索エンジンは、正しい型のデータに対してのみフィルタ操作(`>`、`<`など)を実行します。そのため、これらの検索エンジンを使用して検索可能なデータをカスタマイズする場合、数値が正しい型にキャストされていることを確認する必要があります。

1public function toSearchableArray()
2{
3 return [
4 'id' => (int) $this->id,
5 'name' => $this->name,
6 'price' => (float) $this->price,
7 ];
8}

インデックス設定(Algolia)

Algoliaインデックスの追加設定を行いたい場合があります。これらの設定はAlgolia UIを介して管理できますが、アプリケーションのconfig/scout.php設定ファイルからインデックス設定の望ましい状態を直接管理する方が効率的な場合があります。

このアプローチにより、アプリケーションの自動デプロイパイプラインを通じてこれらの設定をデプロイでき、手動での設定を避け、複数の環境間での一貫性を確保できます。フィルタ可能な属性、ランキング、ファセット、またはその他のサポートされている設定を設定できます。

まず、アプリケーションのconfig/scout.php設定ファイルで各インデックスの設定を追加します。

1use App\Models\User;
2use App\Models\Flight;
3 
4'algolia' => [
5 'id' => env('ALGOLIA_APP_ID', ''),
6 'secret' => env('ALGOLIA_SECRET', ''),
7 'index-settings' => [
8 User::class => [
9 'searchableAttributes' => ['id', 'name', 'email'],
10 'attributesForFaceting'=> ['filterOnly(email)'],
11 // Other settings fields...
12 ],
13 Flight::class => [
14 'searchableAttributes'=> ['id', 'destination'],
15 ],
16 ],
17],

特定のインデックスの基になるモデルがソフトデリート可能で、`index-settings`配列に含まれている場合、Scoutはそのインデックスでソフトデリートされたモデルのファセットのサポートを自動的に含めます。ソフトデリート可能なモデルインデックスに他に定義するファセット属性がない場合は、そのモデルの`index-settings`配列に空のエントリを追加するだけで済みます。

1'index-settings' => [
2 Flight::class => []
3],

アプリケーションのインデックス設定を構成した後、scout:sync-index-settings Artisanコマンドを実行する必要があります。このコマンドは、現在構成されているインデックス設定をAlgoliaに通知します。便宜上、このコマンドをデプロイプロセスの一部にすることをお勧めします。

1php artisan scout:sync-index-settings

フィルタ可能データとインデックス設定の構成(Meilisearch)

Scoutの他のドライバとは異なり、Meilisearchでは、フィルタ可能な属性、ソート可能な属性、およびその他のサポートされている設定フィールドなど、インデックスの検索設定を事前に定義する必要があります。

フィルタ可能な属性は、Scoutの`where`メソッドを呼び出すときにフィルタリングする予定の属性であり、ソート可能な属性は、Scoutの`orderBy`メソッドを呼び出すときにソートする予定の属性です。インデックス設定を定義するには、アプリケーションの`scout`設定ファイル内の`meilisearch`設定エントリの`index-settings`部分を調整します。

1use App\Models\User;
2use App\Models\Flight;
3 
4'meilisearch' => [
5 'host' => env('MEILISEARCH_HOST', 'https://:7700'),
6 'key' => env('MEILISEARCH_KEY', null),
7 'index-settings' => [
8 User::class => [
9 'filterableAttributes'=> ['id', 'name', 'email'],
10 'sortableAttributes' => ['created_at'],
11 // Other settings fields...
12 ],
13 Flight::class => [
14 'filterableAttributes'=> ['id', 'destination'],
15 'sortableAttributes' => ['updated_at'],
16 ],
17 ],
18],

特定のインデックスの基になるモデルがソフトデリート可能で、`index-settings`配列に含まれている場合、Scoutはそのインデックスでソフトデリートされたモデルのフィルタリングのサポートを自動的に含めます。ソフトデリート可能なモデルインデックスに他に定義するフィルタ可能またはソート可能な属性がない場合は、そのモデルの`index-settings`配列に空のエントリを追加するだけで済みます。

1'index-settings' => [
2 Flight::class => []
3],

アプリケーションのインデックス設定を構成した後、scout:sync-index-settings Artisanコマンドを実行する必要があります。このコマンドは、現在構成されているインデックス設定をMeilisearchに通知します。便宜上、このコマンドをデプロイプロセスの一部にすることをお勧めします。

1php artisan scout:sync-index-settings

モデルIDの設定

デフォルトでは、Scoutはモデルのプライマリキーを、検索インデックスに保存されるモデルの一意のID/キーとして使用します。この動作をカスタマイズする必要がある場合は、モデルの`getScoutKey`および`getScoutKeyName`メソッドをオーバーライドできます。

1<?php
2 
3namespace App\Models;
4 
5use Illuminate\Database\Eloquent\Model;
6use Laravel\Scout\Searchable;
7 
8class User extends Model
9{
10 use Searchable;
11 
12 /**
13 * Get the value used to index the model.
14 */
15 public function getScoutKey(): mixed
16 {
17 return $this->email;
18 }
19 
20 /**
21 * Get the key name used to index the model.
22 */
23 public function getScoutKeyName(): mixed
24 {
25 return 'email';
26 }
27}

モデルごとの検索エンジンの設定

検索時、Scoutは通常、アプリケーションの`scout`設定ファイルで指定されたデフォルトの検索エンジンを使用します。ただし、特定のモデルの検索エンジンは、モデルの`searchableUsing`メソッドをオーバーライドすることで変更できます。

1<?php
2 
3namespace App\Models;
4 
5use Illuminate\Database\Eloquent\Model;
6use Laravel\Scout\Engines\Engine;
7use Laravel\Scout\EngineManager;
8use Laravel\Scout\Searchable;
9 
10class User extends Model
11{
12 use Searchable;
13 
14 /**
15 * Get the engine used to index the model.
16 */
17 public function searchableUsing(): Engine
18 {
19 return app(EngineManager::class)->engine('meilisearch');
20 }
21}

ユーザーの識別

Scoutでは、Algoliaを使用する際にユーザーを自動的に識別することもできます。認証済みユーザーを検索操作に関連付けることは、Algoliaのダッシュボードで検索分析を表示する際に役立ちます。アプリケーションの`.env`ファイルで`SCOUT_IDENTIFY`環境変数を`true`として定義することで、ユーザー識別を有効にできます。

1SCOUT_IDENTIFY=true

この機能を有効にすると、リクエストのIPアドレスと認証済みユーザーのプライマリ識別子もAlgoliaに渡され、このデータがユーザーによって行われた検索リクエストに関連付けられます。

データベース/コレクションエンジン

データベースエンジン

データベースエンジンは現在、MySQLとPostgreSQLをサポートしています。

アプリケーションが小規模から中規模のデータベースとやり取りする場合や、ワークロードが軽い場合は、Scoutの「データベース」エンジンから始めるのが便利かもしれません。データベースエンジンは、「where like」句と全文インデックスを使用して、既存のデータベースから結果をフィルタリングし、クエリに適用可能な検索結果を決定します。

データベースエンジンを使用するには、`SCOUT_DRIVER`環境変数の値を`database`に設定するか、アプリケーションの`scout`設定ファイルで`database`ドライバを直接指定します。

1SCOUT_DRIVER=database

データベースエンジンを優先ドライバとして指定したら、検索可能なデータを構成する必要があります。その後、モデルに対して検索クエリを実行し始めることができます。Algolia、Meilisearch、またはTypesenseのインデックスをシードするために必要なインデックス作成などの検索エンジンのインデックス作成は、データベースエンジンを使用する場合は不要です。

データベース検索戦略のカスタマイズ

デフォルトでは、データベースエンジンは検索可能として構成したすべてのモデル属性に対して「where like」クエリを実行します。ただし、状況によっては、これがパフォーマンスの低下につながる可能性があります。そのため、データベースエンジンの検索戦略を構成して、一部の指定されたカラムが全文検索クエリを利用したり、「where like」制約のみを使用して文字列のプレフィックス(`example%`)を検索したり、文字列全体(`%example%`)を検索するのではなくしたりすることができます。

この動作を定義するには、モデルの`toSearchableArray`メソッドにPHP属性を割り当てます。追加の検索戦略の動作が割り当てられていないカラムは、引き続きデフォルトの「where like」戦略を使用します。

1use Laravel\Scout\Attributes\SearchUsingFullText;
2use Laravel\Scout\Attributes\SearchUsingPrefix;
3 
4/**
5 * Get the indexable data array for the model.
6 *
7 * @return array<string, mixed>
8 */
9#[SearchUsingPrefix(['id', 'email'])]
10#[SearchUsingFullText(['bio'])]
11public function toSearchableArray(): array
12{
13 return [
14 'id' => $this->id,
15 'name' => $this->name,
16 'email' => $this->email,
17 'bio' => $this->bio,
18 ];
19}

カラムが全文クエリ制約を使用するように指定する前に、そのカラムに全文インデックスが割り当てられていることを確認してください。

コレクションエンジン

ローカル開発中にAlgolia、Meilisearch、またはTypesense検索エンジンを自由に使用できますが、「コレクション」エンジンから始める方が便利な場合があります。コレクションエンジンは、「where」句とコレクションフィルタリングを既存のデータベースからの結果に使用して、クエリに適用可能な検索結果を決定します。このエンジンを使用する場合、検索可能なモデルを「インデックス」する必要はありません。それらは単にローカルデータベースから取得されるためです。

コレクションエンジンを使用するには、`SCOUT_DRIVER`環境変数の値を`collection`に設定するか、アプリケーションの`scout`設定ファイルで`collection`ドライバを直接指定します。

1SCOUT_DRIVER=collection

コレクションドライバを優先ドライバとして指定したら、モデルに対して検索クエリを実行し始めることができます。Algolia、Meilisearch、またはTypesenseのインデックスをシードするために必要なインデックス作成などの検索エンジンのインデックス作成は、コレクションエンジンを使用する場合は不要です。

データベースエンジンとの違い

一見すると、「データベース」エンジンと「コレクション」エンジンは非常に似ています。どちらもデータベースと直接やり取りして検索結果を取得します。しかし、コレクションエンジンは全文インデックスや`LIKE`句を使用して一致するレコードを見つけません。代わりに、すべての可能なレコードを取得し、Laravelの`Str::is`ヘルパーを使用して検索文字列がモデル属性値内に存在するかどうかを判断します。

コレクションエンジンは、Laravelがサポートするすべてのリレーショナルデータベース(SQLiteおよびSQL Serverを含む)で動作するため、最も移植性の高い検索エンジンです。ただし、Scoutのデータベースエンジンよりも効率は劣ります。

インデックス

バッチインポート

既存のプロジェクトにScoutをインストールする場合、インデックスにインポートする必要があるデータベースレコードが既にあるかもしれません。Scoutは、既存のすべてのレコードを検索インデックスにインポートするために使用できる`scout:import` Artisanコマンドを提供しています。

1php artisan scout:import "App\Models\Post"

`flush`コマンドを使用して、モデルのすべてのレコードを検索インデックスから削除できます。

1php artisan scout:flush "App\Models\Post"

インポートクエリの変更

バッチインポートのためにすべてのモデルを取得するために使用されるクエリを変更したい場合は、モデルに`makeAllSearchableUsing`メソッドを定義できます。これは、モデルをインポートする前に必要になる可能性のあるリレーションシップのイーガーローディングを追加するのに最適な場所です。

1use Illuminate\Database\Eloquent\Builder;
2 
3/**
4 * Modify the query used to retrieve models when making all of the models searchable.
5 */
6protected function makeAllSearchableUsing(Builder $query): Builder
7{
8 return $query->with('author');
9}

`makeAllSearchableUsing`メソッドは、キューを使用してモデルをバッチインポートする場合には適用できない場合があります。ジョブによってモデルコレクションが処理される際、リレーションシップは復元されません

レコードの追加

モデルに`Laravel\Scout\Searchable`トレイトを追加したら、モデルインスタンスを`save`または`create`するだけで、自動的に検索インデックスに追加されます。Scoutをキューを使用するように構成している場合、この操作はキューワーカによってバックグラウンドで実行されます。

1use App\Models\Order;
2 
3$order = new Order;
4 
5// ...
6 
7$order->save();

クエリによるレコードの追加

Eloquentクエリを介してモデルのコレクションを検索インデックスに追加したい場合は、Eloquentクエリに`searchable`メソッドをチェーンできます。`searchable`メソッドは、クエリの結果をチャンク化し、レコードを検索インデックスに追加します。繰り返しますが、Scoutをキューを使用するように構成している場合、すべてのチャンクはキューワーカによってバックグラウンドでインポートされます。

1use App\Models\Order;
2 
3Order::where('price', '>', 100)->searchable();

Eloquentリレーションシップインスタンスで`searchable`メソッドを呼び出すこともできます。

1$user->orders()->searchable();

または、メモリ内にEloquentモデルのコレクションが既にある場合は、コレクションインスタンスで`searchable`メソッドを呼び出して、モデルインスタンスを対応するインデックスに追加できます。

1$orders->searchable();

`searchable`メソッドは「upsert」操作と見なすことができます。つまり、モデルレコードが既にインデックスにある場合は更新され、検索インデックスに存在しない場合はインデックスに追加されます。

レコードの更新

検索可能なモデルを更新するには、モデルインスタンスのプロパティを更新し、モデルをデータベースに`save`するだけです。Scoutは自動的に変更を検索インデックスに永続化します。

1use App\Models\Order;
2 
3$order = Order::find(1);
4 
5// Update the order...
6 
7$order->save();

Eloquentクエリインスタンスで`searchable`メソッドを呼び出して、モデルのコレクションを更新することもできます。モデルが検索インデックスに存在しない場合は作成されます。

1Order::where('price', '>', 100)->searchable();

リレーションシップ内のすべてのモデルの検索インデックスレコードを更新したい場合は、リレーションシップインスタンスで`searchable`を呼び出すことができます。

1$user->orders()->searchable();

または、メモリ内にEloquentモデルのコレクションが既にある場合は、コレクションインスタンスで`searchable`メソッドを呼び出して、モデルインスタンスを対応するインデックスで更新できます。

1$orders->searchable();

インポート前のレコードの変更

検索可能にする前にモデルのコレクションを準備する必要がある場合があります。たとえば、リレーションシップデータを効率的に検索インデックスに追加できるように、リレーションシップをイーガーロードしたい場合があります。これを実現するには、対応するモデルで`makeSearchableUsing`メソッドを定義します。

1use Illuminate\Database\Eloquent\Collection;
2 
3/**
4 * Modify the collection of models being made searchable.
5 */
6public function makeSearchableUsing(Collection $models): Collection
7{
8 return $models->load('author');
9}

レコードの削除

インデックスからレコードを削除するには、データベースからモデルを`delete`するだけです。これは、ソフトデリートモデルを使用している場合でも実行できます。

1use App\Models\Order;
2 
3$order = Order::find(1);
4 
5$order->delete();

レコードを削除する前にモデルを取得したくない場合は、Eloquentクエリインスタンスで`unsearchable`メソッドを使用できます。

1Order::where('price', '>', 100)->unsearchable();

リレーションシップ内のすべてのモデルの検索インデックスレコードを削除したい場合は、リレーションシップインスタンスで`unsearchable`を呼び出すことができます。

1$user->orders()->unsearchable();

または、メモリ内にEloquentモデルのコレクションが既にある場合は、コレクションインスタンスで`unsearchable`メソッドを呼び出して、モデルインスタンスを対応するインデックスから削除できます。

1$orders->unsearchable();

モデルのすべてのレコードを対応するインデックスから削除するには、`removeAllFromSearch`メソッドを呼び出すことができます。

1Order::removeAllFromSearch();

インデックスの一時停止

モデルデータを検索インデックスに同期せずにモデルに対してEloquent操作のバッチを実行する必要がある場合があります。これは`withoutSyncingToSearch`メソッドを使用して行うことができます。このメソッドは、すぐに実行される単一のクロージャを受け入れます。クロージャ内で発生するモデル操作は、モデルのインデックスに同期されません。

1use App\Models\Order;
2 
3Order::withoutSyncingToSearch(function () {
4 // Perform model actions...
5});

検索対象モデルインスタンスの条件指定

特定の条件下でのみモデルを検索可能にする必要がある場合があります。たとえば、「下書き」と「公開済み」の2つの状態のいずれかである`App\Models\Post`モデルがあるとします。「公開済み」の投稿のみを検索可能にしたい場合があります。これを実現するには、モデルに`shouldBeSearchable`メソッドを定義します。

1/**
2 * Determine if the model should be searchable.
3 */
4public function shouldBeSearchable(): bool
5{
6 return $this->isPublished();
7}

`shouldBeSearchable`メソッドは、`save`および`create`メソッド、クエリ、またはリレーションシップを介してモデルを操作する場合にのみ適用されます。`searchable`メソッドを使用してモデルまたはコレクションを直接検索可能にすると、`shouldBeSearchable`メソッドの結果がオーバーライドされます。

すべての検索可能データが常にデータベースに保存されるため、Scoutの「データベース」エンジンを使用する場合、`shouldBeSearchable`メソッドは適用されません。データベースエンジンを使用する際に同様の動作を実現するには、代わりにwhere句を使用する必要があります。

検索

`search`メソッドを使用してモデルの検索を開始できます。searchメソッドは、モデルの検索に使用される単一の文字列を受け入れます。次に、検索クエリに`get`メソッドをチェーンして、特定の検索クエリに一致するEloquentモデルを取得する必要があります。

1use App\Models\Order;
2 
3$orders = Order::search('Star Trek')->get();

Scout検索はEloquentモデルのコレクションを返すため、ルートまたはコントローラから直接結果を返すこともでき、それらは自動的にJSONに変換されます。

1use App\Models\Order;
2use Illuminate\Http\Request;
3 
4Route::get('/search', function (Request $request) {
5 return Order::search($request->search)->get();
6});

Eloquentモデルに変換される前の生の検索結果を取得したい場合は、`raw`メソッドを使用できます。

1$orders = Order::search('Star Trek')->raw();

カスタムインデックス

検索クエリは通常、モデルの`searchableAs`メソッドで指定されたインデックスに対して実行されます。ただし、`within`メソッドを使用して、代わりに検索するカスタムインデックスを指定できます。

1$orders = Order::search('Star Trek')
2 ->within('tv_shows_popularity_desc')
3 ->get();

WHERE句

Scoutでは、検索クエリに単純な「where」句を追加できます。現在、これらの句は基本的な数値の等価性チェックのみをサポートしており、主に所有者IDによる検索クエリのスコープ設定に役立ちます。

1use App\Models\Order;
2 
3$orders = Order::search('Star Trek')->where('user_id', 1)->get();

さらに、`whereIn`メソッドを使用して、特定のカラムの値が指定された配列内に含まれていることを検証できます。

1$orders = Order::search('Star Trek')->whereIn(
2 'status', ['open', 'paid']
3)->get();

`whereNotIn`メソッドは、指定されたカラムの値が指定された配列に含まれていないことを検証します。

1$orders = Order::search('Star Trek')->whereNotIn(
2 'status', ['closed']
3)->get();

検索インデックスはリレーショナルデータベースではないため、より高度な「where」句は現在サポートされていません。

アプリケーションがMeilisearchを使用している場合は、Scoutの「where」句を利用する前に、アプリケーションのフィルタ可能属性を構成する必要があります。

ペジネーション

モデルのコレクションを取得するだけでなく、`paginate`メソッドを使用して検索結果をページネーションできます。このメソッドは、従来のEloquentクエリをページネーションした場合と同様に、`Illuminate\Pagination\LengthAwarePaginator`インスタンスを返します。

1use App\Models\Order;
2 
3$orders = Order::search('Star Trek')->paginate();

ページごとに取得するモデルの数を`paginate`メソッドの最初の引数として渡すことで指定できます。

1$orders = Order::search('Star Trek')->paginate(15);

結果を取得したら、従来のEloquentクエリをページネーションした場合と同様に、Bladeを使用して結果を表示し、ページリンクをレンダリングできます。

1<div class="container">
2 @foreach ($orders as $order)
3 {{ $order->price }}
4 @endforeach
5</div>
6 
7{{ $orders->links() }}

もちろん、ページネーションの結果をJSONとして取得したい場合は、ルートまたはコントローラからページネータインスタンスを直接返すことができます。

1use App\Models\Order;
2use Illuminate\Http\Request;
3 
4Route::get('/orders', function (Request $request) {
5 return Order::search($request->input('query'))->paginate(15);
6});

検索エンジンはEloquentモデルのグローバルスコープ定義を認識しないため、Scoutのページネーションを利用するアプリケーションでグローバルスコープを利用すべきではありません。または、Scoutを介して検索する際にグローバルスコープの制約を再作成する必要があります。

ソフトデリート

インデックス付きモデルがソフトデリートされており、ソフトデリートされたモデルを検索する必要がある場合は、`config/scout.php`設定ファイルの`soft_delete`オプションを`true`に設定します。

1'soft_delete' => true,

この設定オプションが`true`の場合、Scoutはソフトデリートされたモデルを検索インデックスから削除しません。代わりに、インデックス付きレコードに非表示の`__soft_deleted`属性を設定します。その後、`withTrashed`または`onlyTrashed`メソッドを使用して、検索時にソフトデリートされたレコードを取得できます。

1use App\Models\Order;
2 
3// Include trashed records when retrieving results...
4$orders = Order::search('Star Trek')->withTrashed()->get();
5 
6// Only include trashed records when retrieving results...
7$orders = Order::search('Star Trek')->onlyTrashed()->get();

ソフトデリートされたモデルが`forceDelete`を使用して完全に削除されると、Scoutはそれを検索インデックスから自動的に削除します。

エンジン検索のカスタマイズ

エンジンの検索動作を高度にカスタマイズする必要がある場合は、`search`メソッドの2番目の引数としてクロージャを渡すことができます。たとえば、このコールバックを使用して、検索クエリがAlgoliaに渡される前に、検索オプションに地理位置情報を追加できます。

1use Algolia\AlgoliaSearch\SearchIndex;
2use App\Models\Order;
3 
4Order::search(
5 'Star Trek',
6 function (SearchIndex $algolia, string $query, array $options) {
7 $options['body']['query']['bool']['filter']['geo_distance'] = [
8 'distance' => '1000km',
9 'location' => ['lat' => 36, 'lon' => 111],
10 ];
11 
12 return $algolia->search($query, $options);
13 }
14)->get();

Eloquent結果クエリのカスタマイズ

Scoutがアプリケーションの検索エンジンから一致するEloquentモデルのリストを取得した後、Eloquentはプライマリキーによってすべての一致するモデルを取得するために使用されます。`query`メソッドを呼び出すことで、このクエリをカスタマイズできます。`query`メソッドは、引数としてEloquentクエリビルダインスタンスを受け取るクロージャを受け入れます。

1use App\Models\Order;
2use Illuminate\Database\Eloquent\Builder;
3 
4$orders = Order::search('Star Trek')
5 ->query(fn (Builder $query) => $query->with('invoices'))
6 ->get();

このコールバックは、関連するモデルが既にアプリケーションの検索エンジンから取得された後に呼び出されるため、`query`メソッドは結果の「フィルタリング」には使用すべきではありません。代わりに、Scoutのwhere句を使用する必要があります。

カスタムエンジン

エンジンの作成

組み込みのScout検索エンジンのいずれかがニーズに合わない場合は、独自のカスタムエンジンを作成してScoutに登録できます。エンジンは`Laravel\Scout\Engines\Engine`抽象クラスを拡張する必要があります。この抽象クラスには、カスタムエンジンが実装する必要のある8つのメソッドが含まれています。

1use Laravel\Scout\Builder;
2 
3abstract public function update($models);
4abstract public function delete($models);
5abstract public function search(Builder $builder);
6abstract public function paginate(Builder $builder, $perPage, $page);
7abstract public function mapIds($results);
8abstract public function map(Builder $builder, $results, $model);
9abstract public function getTotalCount($results);
10abstract public function flush($model);

`Laravel\Scout\Engines\AlgoliaEngine`クラスでこれらのメソッドの実装を確認すると役立つ場合があります。このクラスは、独自のエンジンでこれらの各メソッドを実装する方法を学ぶための良い出発点を提供します。

エンジンの登録

カスタムエンジンを作成したら、Scoutエンジンマネージャの`extend`メソッドを使用してScoutに登録できます。Scoutのエンジンマネージャは、Laravelサービスコンテナから解決できます。`App\Providers\AppServiceProvider`クラスまたはアプリケーションで使用される他のサービスプロバイダの`boot`メソッドから`extend`メソッドを呼び出す必要があります。

1use App\ScoutExtensions\MySqlSearchEngine;
2use Laravel\Scout\EngineManager;
3 
4/**
5 * Bootstrap any application services.
6 */
7public function boot(): void
8{
9 resolve(EngineManager::class)->extend('mysql', function () {
10 return new MySqlSearchEngine;
11 });
12}

エンジンが登録されたら、アプリケーションの`config/scout.php`設定ファイルでデフォルトのScout`driver`として指定できます。

1'driver' => 'mysql',

Laravelは最も生産的な方法です
ソフトウェアを構築、デプロイ、監視します。