Laravel Scout
介绍
Laravel Scout 提供了一种简单的基于驱动的解决方案,用于为您的 Eloquent 模型 添加全文搜索。使用模型观察者,Scout 将自动保持您的搜索索引与 Eloquent 记录同步。
目前,Scout 附带 Algolia、Meilisearch、Typesense 和 MySQL / PostgreSQL (database
) 驱动程序。此外,Scout 还包括一个“集合”驱动程序,旨在用于本地开发,不需要任何外部依赖或第三方服务。此外,编写自定义驱动程序非常简单,您可以自由扩展 Scout 以实现自己的搜索实现。
安装
首先,通过 Composer 包管理器安装 Scout:
composer require laravel/scout
安装 Scout 后,您应该使用 vendor:publish
Artisan 命令发布 Scout 配置文件。此命令将把 scout.php
配置文件发布到您应用程序的 config
目录:
php artisan vendor:publish --provider="Laravel\Scout\ScoutServiceProvider"
最后,将 Laravel\Scout\Searchable
特性添加到您希望使其可搜索的模型中。此特性将注册一个模型观察者,自动保持模型与您的搜索驱动程序同步:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Post extends Model
{
use Searchable;
}
队列
虽然使用 Scout 并不是严格要求的,但您应该强烈考虑在使用该库之前配置一个 队列驱动程序。运行队列工作程序将允许 Scout 将所有操作排队,这些操作将您的模型信息同步到搜索索引,从而为您应用程序的 Web 界面提供更好的响应时间。
一旦您配置了队列驱动程序,请将 config/scout.php
配置文件中的 queue
选项的值设置为 true
:
'queue' => true,
即使 queue
选项设置为 false
,也要记住,某些 Scout 驱动程序(如 Algolia 和 Meilisearch)始终异步索引记录。这意味着,即使索引操作在您的 Laravel 应用程序中已完成,搜索引擎本身可能不会立即反映新记录和更新记录。
要指定 Scout 作业使用的连接和队列,您可以将 queue
配置选项定义为数组:
'queue' => [
'connection' => 'redis',
'queue' => 'scout'
],
当然,如果您自定义了 Scout 作业使用的连接和队列,您应该运行一个队列工作程序来处理该连接和队列上的作业:
php artisan queue:work redis --queue=scout
驱动程序先决条件
Algolia
使用 Algolia 驱动程序时,您应该在 config/scout.php
配置文件中配置您的 Algolia id
和 secret
凭据。一旦配置了凭据,您还需要通过 Composer 包管理器安装 Algolia PHP SDK:
composer require algolia/algoliasearch-client-php
Meilisearch
Meilisearch 是一个快速且开源的搜索引擎。如果您不确定如何在本地计算机上安装 Meilisearch,您可以使用 Laravel Sail,这是 Laravel 官方支持的 Docker 开发环境。
使用 Meilisearch 驱动程序时,您需要通过 Composer 包管理器安装 Meilisearch PHP SDK:
composer require meilisearch/meilisearch-php http-interop/http-factory-guzzle
然后,在您应用程序的 .env
文件中设置 SCOUT_DRIVER
环境变量以及您的 Meilisearch host
和 key
凭据:
SCOUT_DRIVER=meilisearch
MEILISEARCH_HOST=http://127.0.0.1:7700
MEILISEARCH_KEY=masterKey
有关 Meilisearch 的更多信息,请查阅 Meilisearch 文档。
此外,您应该确保安装与您的 Meilisearch 二进制版本兼容的 meilisearch/meilisearch-php
版本,具体请查看 Meilisearch 关于二进制兼容性的文档。
在升级使用 Meilisearch 的应用程序上的 Scout 时,您应始终 查看任何其他破坏性更改 到 Meilisearch 服务本身。
Typesense
Typesense 是一个快速的开源搜索引擎,支持关键字搜索、语义搜索、地理搜索和向量搜索。
您可以 自托管 Typesense 或使用 Typesense Cloud。
要开始使用 Typesense 和 Scout,请通过 Composer 包管理器安装 Typesense PHP SDK:
composer require typesense/typesense-php
然后,在您应用程序的 .env
文件中设置 SCOUT_DRIVER
环境变量以及您的 Typesense 主机和 API 密钥凭据:
SCOUT_DRIVER=typesense
TYPESENSE_API_KEY=masterKey
TYPESENSE_HOST=localhost
如果您使用 Laravel Sail,您可能需要调整 TYPESENSE_HOST
环境变量以匹配 Docker 容器名称。您还可以选择性地指定安装的端口、路径和协议:
TYPESENSE_PORT=8108
TYPESENSE_PATH=
TYPESENSE_PROTOCOL=http
有关您应用程序的 Typesense 集合的其他设置和模式定义,请查阅您应用程序的 config/scout.php
配置文件。有关 Typesense 的更多信息,请查阅 Typesense 文档。
为 Typesense 准备存储数据
使用 Typesense 时,您的可搜索模型必须定义一个 toSearchableArray
方法,该方法将模型的主键转换为字符串,并将创建日期转换为 UNIX 时间戳:
/**
* 获取模型的可索引数据数组。
*
* @return array<string, mixed>
*/
public function toSearchableArray()
{
return array_merge($this->toArray(),[
'id' => (string) $this->id,
'created_at' => $this->created_at->timestamp,
]);
}
您还应该在应用程序的 config/scout.php
文件中定义您的 Typesense 集合模式。集合模式描述了每个可通过 Typesense 搜索的字段的数据类型。有关所有可用模式选项的更多信息,请查阅 Typesense 文档。
如果您需要在定义后更改 Typesense 集合的模式,您可以运行 scout:flush
和 scout:import
,这将删除所有现有的索引数据并重新创建模式。或者,您可以使用 Typesense 的 API 修改集合的模式,而无需删除任何索引数据。
如果您的可搜索模型是软可删除的,您应该在应用程序的 config/scout.php
配置文件中定义一个 __soft_deleted
字段在模型对应的 Typesense 模式中:
User::class => [
'collection-schema' => [
'fields' => [
// ...
[
'name' => '__soft_deleted',
'type' => 'int32',
'optional' => true,
],
],
],
],
动态搜索参数
Typesense 允许您在通过 options
方法执行搜索操作时动态修改您的 搜索参数:
use App\Models\Todo;
Todo::search('杂货')->options([
'query_by' => 'title, description'
])->get();
配置
配置模型索引
每个 Eloquent 模型与给定的搜索“索引”同步,该索引包含该模型的所有可搜索记录。换句话说,您可以将每个索引视为 MySQL 表。默认情况下,每个模型将被持久化到与模型的典型“表”名称匹配的索引中。通常,这是模型名称的复数形式;但是,您可以通过覆盖模型上的 searchableAs
方法来自定义模型的索引:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Post extends Model
{
use Searchable;
/**
* 获取与模型关联的索引的名称。
*/
public function searchableAs(): string
{
return 'posts_index';
}
}
配置可搜索数据
默认情况下,给定模型的整个 toArray
形式将被持久化到其搜索索引中。如果您希望自定义同步到搜索索引的数据,您可以覆盖模型上的 toSearchableArray
方法:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class Post extends Model
{
use Searchable;
/**
* 获取模型的可索引数据数组。
*
* @return array<string, mixed>
*/
public function toSearchableArray(): array
{
$array = $this->toArray();
// 自定义数据数组...
return $array;
}
}
某些搜索引擎(例如 Meilisearch)将仅对正确类型的数据执行过滤操作(>
、<
等)。因此,在使用这些搜索引擎并自定义可搜索数据时,您应该确保将数值转换为其正确类型:
public function toSearchableArray()
{
return [
'id' => (int) $this->id,
'name' => $this->name,
'price' => (float) $this->price,
];
}
配置可过滤数据和索引设置(Meilisearch)
与 Scout 的其他驱动程序不同,Meilisearch 要求您预先定义索引搜索设置,例如可过滤属性、可排序属性和 其他支持的设置字段。
可过滤属性是您计划在调用 Scout 的 where
方法时进行过滤的任何属性,而可排序属性是您计划在调用 Scout 的 orderBy
方法时进行排序的任何属性。要定义您的索引设置,请调整您应用程序的 scout
配置文件中 meilisearch
配置条目的 index-settings
部分:
use App\Models\User;
use App\Models\Flight;
'meilisearch' => [
'host' => env('MEILISEARCH_HOST', 'http://localhost:7700'),
'key' => env('MEILISEARCH_KEY', null),
'index-settings' => [
User::class => [
'filterableAttributes'=> ['id', 'name', 'email'],
'sortableAttributes' => ['created_at'],
// 其他设置字段...
],
Flight::class => [
'filterableAttributes'=> ['id', 'destination'],
'sortableAttributes' => ['updated_at'],
],
],
],
如果给定索引的模型是软可删除的,并且包含在 index-settings
数组中,Scout 将自动支持在该索引上对软删除模型进行过滤。如果您没有其他可过滤或可排序的属性要为软可删除模型索引定义,您可以简单地为该模型添加一个空条目到 index-settings
数组中:
'index-settings' => [
Flight::class => []
],
配置完应用程序的索引设置后,您必须调用 scout:sync-index-settings
Artisan 命令。此命令将通知 Meilisearch 您当前配置的索引设置。为了方便起见,您可能希望将此命令作为部署过程的一部分:
php artisan scout:sync-index-settings
配置模型 ID
默认情况下,Scout 将使用模型的主键作为存储在搜索索引中的模型唯一 ID / 键。如果您需要自定义此行为,可以在模型上覆盖 getScoutKey
和 getScoutKeyName
方法:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Searchable;
class User extends Model
{
use Searchable;
/**
* 获取用于索引模型的值。
*/
public function getScoutKey(): mixed
{
return $this->email;
}
/**
* 获取用于索引模型的键名。
*/
public function getScoutKeyName(): mixed
{
return 'email';
}
}
为模型配置搜索引擎
在搜索时,Scout 通常会使用您应用程序的 scout
配置文件中指定的默认搜索引擎。但是,可以通过覆盖模型上的 searchableUsing
方法来更改特定模型的搜索引擎:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use Laravel\Scout\Engines\Engine;
use Laravel\Scout\EngineManager;
use Laravel\Scout\Searchable;
class User extends Model
{
use Searchable;
/**
* 获取用于索引模型的引擎。
*/
public function searchableUsing(): Engine
{
return app(EngineManager::class)->engine('meilisearch');
}
}
识别用户
Scout 还允许您在使用 Algolia 时自动识别用户。将经过身份验证的用户与搜索操作关联可能在查看 Algolia 仪表板中的搜索分析时很有帮助。您可以通过在应用程序的 .env
文件中将 SCOUT_IDENTIFY
环境变量定义为 true
来启用用户识别:
SCOUT_IDENTIFY=true
启用此功能还将把请求的 IP 地址和经过身份验证的用户的主标识符传递给 Algolia,因此这些数据与用户进行的任何搜索请求相关联。
数据库 / 集合引擎
数据库引擎
数据库引擎当前支持 MySQL 和 PostgreSQL。
如果您的应用程序与小型到中型数据库交互或工作负载较轻,您可能会发现使用 Scout 的“数据库”引擎更方便。数据库引擎将在现有数据库中使用“where like”子句和全文索引来过滤结果,以确定与您的查询相应的搜索结果。
要使用数据库引擎,您只需将 SCOUT_DRIVER
环境变量的值设置为 database
,或在应用程序的 scout
配置文件中直接指定 database
驱动程序:
SCOUT_DRIVER=database
一旦您将数据库引擎指定为首选驱动程序,您必须 配置可搜索数据。然后,您可以开始对模型执行 搜索查询。使用数据库引擎时,不需要进行搜索引擎索引,例如需要为 Algolia、Meilisearch 或 Typesense 索引播种的索引。
自定义数据库搜索策略
默认情况下,数据库引擎将对您已 配置为可搜索 的每个模型属性执行“where like”查询。然而,在某些情况下,这可能会导致性能不佳。因此,数据库引擎的搜索策略可以配置,以便某些指定的列使用全文搜索查询,或者仅使用“where like”约束来搜索字符串的前缀(example%
),而不是在整个字符串中搜索(%example%
)。
要定义此行为,您可以将 PHP 属性分配给模型的 toSearchableArray
方法。未分配额外搜索策略行为的任何列将继续使用默认的“where like”策略:
use Laravel\Scout\Attributes\SearchUsingFullText;
use Laravel\Scout\Attributes\SearchUsingPrefix;
/**
* 获取模型的可索引数据数组。
*
* @return array<string, mixed>
*/
#[SearchUsingPrefix(['id', 'email'])]
#[SearchUsingFullText(['bio'])]
public function toSearchableArray(): array
{
return [
'id' => $this->id,
'name' => $this->name,
'email' => $this->email,
'bio' => $this->bio,
];
}
在指定某列应使用全文查询约束之前,请确保该列已分配 全文索引。
集合引擎
虽然您可以在本地开发期间使用 Algolia、Meilisearch 或 Typesense 搜索引擎,但您可能会发现使用“集合”引擎更方便。集合引擎将使用“where”子句和集合过滤来确定与您的查询相应的搜索结果。当使用此引擎时,不需要“索引”您的可搜索模型,因为它们将直接从本地数据库中检索。
要使用集合引擎,您只需将 SCOUT_DRIVER
环境变量的值设置为 collection
,或在应用程序的 scout
配置文件中直接指定 collection
驱动程序:
SCOUT_DRIVER=collection
一旦您将集合驱动程序指定为首选驱动程序,您可以开始对模型执行 搜索查询。使用集合引擎时,不需要进行搜索引擎索引,例如需要为 Algolia、Meilisearch 或 Typesense 索引播种的索引。
与数据库引擎的区别
乍一看,“数据库”和“集合”引擎非常相似。它们都直接与您的数据库交互以检索搜索结果。然而,集合引擎不使用全文索引或 LIKE
子句来查找匹配记录。相反,它会提取所有可能的记录,并使用 Laravel 的 Str::is
辅助函数来确定搜索字符串是否存在于模型属性值中。
集合引擎是最便携的搜索引擎,因为它适用于 Laravel 支持的所有关系数据库(包括 SQLite 和 SQL Server);但是,它的效率低于 Scout 的数据库引擎。
索引
批量导入
如果您在现有项目中安装 Scout,您可能已经有需要导入到索引中的数据库记录。Scout 提供了一个 scout:import
Artisan 命令,您可以使用它将所有现有记录导入到搜索索引中:
php artisan scout:import "App\Models\Post"
flush
命令可用于从搜索索引中删除模型的所有记录:
php artisan scout:flush "App\Models\Post"
修改导入查询
如果您希望修改用于检索所有模型以进行批量导入的查询,您可以在模型上定义一个 makeAllSearchableUsing
方法。这是添加任何必要的急切关系加载的好地方,以便在导入模型之前:
use Illuminate\Database\Eloquent\Builder;
/**
* 修改用于检索模型的查询,以使所有模型可搜索。
*/
protected function makeAllSearchableUsing(Builder $query): Builder
{
return $query->with('author');
}
当使用队列批量导入模型时,makeAllSearchableUsing
方法可能不适用。处理作业时,关系不会 恢复。
添加记录
一旦您将 Laravel\Scout\Searchable
特性添加到模型,您只需 save
或 create
模型实例,它将自动添加到您的搜索索引中。如果您已配置 Scout 使用队列,此操作将在后台由队列工作程序执行:
use App\Models\Order;
$order = new Order;
// ...
$order->save();
通过查询添加记录
如果您希望通过 Eloquent 查询将一组模型添加到搜索索引中,您可以将 searchable
方法链接到 Eloquent 查询。searchable
方法将 分块结果 的查询并将记录添加到搜索索引中。同样,如果您已配置 Scout 使用队列,所有块将在后台由队列工作程序导入:
use App\Models\Order;
Order::where('price', '>', 100)->searchable();
您还可以在 Eloquent 关系实例上调用 searchable
方法:
$user->orders()->searchable();
或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 searchable
方法,将模型实例添加到其对应的索引中:
$orders->searchable();
searchable
方法可以视为“upsert”操作。换句话说,如果模型记录已经在您的索引中,它将被更新。如果它在搜索索引中不存在,它将被添加到索引中。
更新记录
要更新可搜索模型,您只需更新模型实例的属性并将模型 save
到数据库。Scout 将自动将更改持久化到您的搜索索引:
use App\Models\Order;
$order = Order::find(1);
// 更新订单...
$order->save();
您还可以在 Eloquent 查询实例上调用 searchable
方法以更新一组模型。如果模型在您的搜索索引中不存在,它们将被创建:
Order::where('price', '>', 100)->searchable();
如果您希望更新关系中所有模型的搜索索引记录,可以在关系实例上调用 searchable
:
$user->orders()->searchable();
或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 searchable
方法,以更新其对应索引中的模型实例:
$orders->searchable();
在导入之前修改记录
有时,您可能需要在使集合模型可搜索之前准备它。例如,您可能希望急切加载一个关系,以便可以有效地将关系数据添加到搜索索引中。为此,请在相应模型上定义一个 makeSearchableUsing
方法:
use Illuminate\Database\Eloquent\Collection;
/**
* 修改正在使可搜索的模型集合。
*/
public function makeSearchableUsing(Collection $models): Collection
{
return $models->load('author');
}
删除记录
要从索引中删除记录,您只需从数据库中 delete
模型。这可以在使用 软删除 模型时完成:
use App\Models\Order;
$order = Order::find(1);
$order->delete();
如果您不想在删除记录之前检索模型,您可以在 Eloquent 查询实例上使用 unsearchable
方法:
Order::where('price', '>', 100)->unsearchable();
如果您希望删除关系中所有模型的搜索索引记录,可以在关系实例上调用 unsearchable
:
$user->orders()->unsearchable();
或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 unsearchable
方法,以从其对应索引中删除模型实例:
$orders->unsearchable();
要从其对应索引中删除所有模型记录,您可以调用 removeAllFromSearch
方法:
Order::removeAllFromSearch();
暂停索引
有时,您可能需要对模型执行一批 Eloquent 操作,而不将模型数据同步到搜索索引。您可以使用 withoutSyncingToSearch
方法来实现此目的。此方法接受一个单一的闭包,该闭包将立即执行。在闭包内发生的任何模型操作都不会同步到模型的索引:
use App\Models\Order;
Order::withoutSyncingToSearch(function () {
// 执行模型操作...
});
有条件可搜索的模型实例
有时,您可能只希望在某些条件下使模型可搜索。例如,假设您有一个 App\Models\Post
模型,它可能处于“草稿”和“已发布”两种状态之一。您可能只希望允许“已发布”的帖子可搜索。为此,您可以在模型上定义一个 shouldBeSearchable
方法:
/**
* 确定模型是否应该可搜索。
*/
public function shouldBeSearchable(): bool
{
return $this->isPublished();
}
shouldBeSearchable
方法仅在通过 save
和 create
方法、查询或关系操作模型时应用。直接使用 searchable
方法使模型或集合可搜索将覆盖 shouldBeSearchable
方法的结果。
shouldBeSearchable
方法在使用 Scout 的“数据库”引擎时不适用,因为所有可搜索数据始终存储在数据库中。要在使用数据库引擎时实现类似行为,您应该使用 where 子句。
搜索
您可以使用 search
方法开始搜索模型。搜索方法接受一个字符串,该字符串将用于搜索您的模型。然后,您应该在搜索查询上链接 get
方法,以检索与给定搜索查询匹配的 Eloquent 模型:
use App\Models\Order;
$orders = Order::search('星际迷航')->get();
由于 Scout 搜索返回的是 Eloquent 模型的集合,您甚至可以直接从路由或控制器返回结果,它们将自动转换为 JSON:
use App\Models\Order;
use Illuminate\Http\Request;
Route::get('/search', function (Request $request) {
return Order::search($request->search)->get();
});
如果您希望在转换为 Eloquent 模型之前获取原始搜索结果,可以使用 raw
方法:
$orders = Order::search('星际迷航')->raw();
自定义索引
搜索查询通常将在模型的 searchableAs
方法指定的索引上执行。但是,您可以使用 within
方法指定应搜索的自定义索引:
$orders = Order::search('星际迷航')
->within('tv_shows_popularity_desc')
->get();
条件子句
Scout 允许您向搜索查询添加简单的“where”子句。目前,这些子句仅支持基本的数字相等检查,主要用于通过所有者 ID 对搜索查询进行范围限制:
use App\Models\Order;
$orders = Order::search('星际迷航')->where('user_id', 1)->get();
此外,whereIn
方法可用于验证给定列的值是否包含在给定数组中:
$orders = Order::search('星际迷航')->whereIn(
'status', ['open', 'paid']
)->get();
whereNotIn
方法验证给定列的值是否不包含在给定数组中:
$orders = Order::search('星际迷航')->whereNotIn(
'status', ['closed']
)->get();
由于搜索索引不是关系数据库,因此当前不支持更高级的“where”子句。
如果您的应用程序使用 Meilisearch,您必须在使用 Scout 的“where”子句之前配置应用程序的 可过滤属性。
分页
除了检索模型集合外,您还可以使用 paginate
方法对搜索结果进行分页。此方法将返回一个 Illuminate\Pagination\LengthAwarePaginator
实例,就像您对传统 Eloquent 查询进行分页一样 分页:
use App\Models\Order;
$orders = Order::search('星际迷航')->paginate();
您可以通过将数量作为第一个参数传递给 paginate
方法来指定每页检索多少模型:
$orders = Order::search('星际迷航')->paginate(15);
检索结果后,您可以使用 Blade 显示结果并渲染页面链接,就像您对传统 Eloquent 查询进行分页一样:
<div class="container">
@foreach ($orders as $order)
{{ $order->price }}
@endforeach
</div>
{{ $orders->links() }}
当然,如果您希望将分页结果作为 JSON 检索,您可以直接从路由或控制器返回分页器实例:
use App\Models\Order;
use Illuminate\Http\Request;
Route::get('/orders', function (Request $request) {
return Order::search($request->input('query'))->paginate(15);
});
由于搜索引擎不知道您的 Eloquent 模型的全局范围定义,因此在使用 Scout 分页的应用程序中,您不应使用全局范围。或者,您应该在通过 Scout 搜索时重新创建全局范围的约束。
软删除
如果您的索引模型 软删除,并且您需要搜索您的软删除模型,请将 config/scout.php
配置文件中的 soft_delete
选项设置为 true
:
'soft_delete' => true,
当此配置选项为 true
时,Scout 将不会从搜索索引中删除软删除模型。相反,它将在索引记录上设置一个隐藏的 __soft_deleted
属性。然后,您可以使用 withTrashed
或 onlyTrashed
方法在搜索时检索软删除记录:
use App\Models\Order;
// 在检索结果时包含已删除记录...
$orders = Order::search('星际迷航')->withTrashed()->get();
// 仅在检索结果时包含已删除记录...
$orders = Order::search('星际迷航')->onlyTrashed()->get();
当使用 forceDelete
永久删除软删除模型时,Scout 将自动将其从搜索索引中删除。
自定义引擎搜索
如果您需要对引擎的搜索行为进行高级自定义,您可以将闭包作为第二个参数传递给 search
方法。例如,您可以使用此回调在将搜索查询传递给 Algolia 之前向搜索选项添加地理位置数据:
use Algolia\AlgoliaSearch\SearchIndex;
use App\Models\Order;
Order::search(
'星际迷航',
function (SearchIndex $algolia, string $query, array $options) {
$options['body']['query']['bool']['filter']['geo_distance'] = [
'distance' => '1000km',
'location' => ['lat' => 36, 'lon' => 111],
];
return $algolia->search($query, $options);
}
)->get();
自定义 Eloquent 结果查询
在 Scout 从应用程序的搜索引擎检索到匹配的 Eloquent 模型列表后,Eloquent 用于通过其主键检索所有匹配的模型。您可以通过调用 query
方法自定义此查询。query
方法接受一个闭包,该闭包将接收 Eloquent 查询构建器实例作为参数:
use App\Models\Order;
use Illuminate\Database\Eloquent\Builder;
$orders = Order::search('星际迷航')
->query(fn (Builder $query) => $query->with('invoices'))
->get();
由于此回调在相关模型已经从应用程序的搜索引擎中检索后调用,因此 query
方法不应用于“过滤”结果。相反,您应该使用 Scout 条件子句。
自定义引擎
编写引擎
如果内置的 Scout 搜索引擎不符合您的需求,您可以编写自己的自定义引擎并将其注册到 Scout。您的引擎应扩展 Laravel\Scout\Engines\Engine
抽象类。此抽象类包含您的自定义引擎必须实现的八个方法:
use Laravel\Scout\Builder;
abstract public function update($models);
abstract public function delete($models);
abstract public function search(Builder $builder);
abstract public function paginate(Builder $builder, $perPage, $page);
abstract public function mapIds($results);
abstract public function map(Builder $builder, $results, $model);
abstract public function getTotalCount($results);
abstract public function flush($model);
您可能会发现查看 Laravel\Scout\Engines\AlgoliaEngine
类中这些方法的实现很有帮助。此类将为您提供学习如何在自己的引擎中实现每个方法的良好起点。
注册引擎
编写自定义引擎后,您可以使用 Scout 引擎管理器的 extend
方法将其注册到 Scout。您可以从 Laravel 服务容器中解析 Scout 的引擎管理器。您应该在 App\Providers\AppServiceProvider
类的 boot
方法或您应用程序使用的任何其他服务提供者中调用 extend
方法:
use App\ScoutExtensions\MySqlSearchEngine;
use Laravel\Scout\EngineManager;
/**
* 启动任何应用程序服务。
*/
public function boot(): void
{
resolve(EngineManager::class)->extend('mysql', function () {
return new MySqlSearchEngine;
});
}
一旦您的引擎被注册,您可以在应用程序的 config/scout.php
配置文件中将其指定为默认 Scout driver
:
'driver' => 'mysql',