Skip to content
赞助商赞助商
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待
虚位以待

Laravel Scout

介绍

Laravel Scout 提供了一种简单的基于驱动的解决方案,用于为您的 Eloquent 模型 添加全文搜索。使用模型观察者,Scout 将自动保持您的搜索索引与 Eloquent 记录同步。

目前,Scout 附带 AlgoliaMeilisearchTypesense 和 MySQL / PostgreSQL (database) 驱动程序。此外,Scout 还包括一个“集合”驱动程序,旨在用于本地开发,不需要任何外部依赖或第三方服务。此外,编写自定义驱动程序非常简单,您可以自由扩展 Scout 以实现自己的搜索实现。

安装

首先,通过 Composer 包管理器安装 Scout:

shell
composer require laravel/scout

安装 Scout 后,您应该使用 vendor:publish Artisan 命令发布 Scout 配置文件。此命令将把 scout.php 配置文件发布到您应用程序的 config 目录:

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

最后,将 Laravel\Scout\Searchable 特性添加到您希望使其可搜索的模型中。此特性将注册一个模型观察者,自动保持模型与您的搜索驱动程序同步:

php
<?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

php
'queue' => true,

即使 queue 选项设置为 false,也要记住,某些 Scout 驱动程序(如 Algolia 和 Meilisearch)始终异步索引记录。这意味着,即使索引操作在您的 Laravel 应用程序中已完成,搜索引擎本身可能不会立即反映新记录和更新记录。

要指定 Scout 作业使用的连接和队列,您可以将 queue 配置选项定义为数组:

php
'queue' => [
    'connection' => 'redis',
    'queue' => 'scout'
],

当然,如果您自定义了 Scout 作业使用的连接和队列,您应该运行一个队列工作程序来处理该连接和队列上的作业:

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

驱动程序先决条件

Algolia

使用 Algolia 驱动程序时,您应该在 config/scout.php 配置文件中配置您的 Algolia idsecret 凭据。一旦配置了凭据,您还需要通过 Composer 包管理器安装 Algolia PHP SDK:

shell
composer require algolia/algoliasearch-client-php

Meilisearch

Meilisearch 是一个快速且开源的搜索引擎。如果您不确定如何在本地计算机上安装 Meilisearch,您可以使用 Laravel Sail,这是 Laravel 官方支持的 Docker 开发环境。

使用 Meilisearch 驱动程序时,您需要通过 Composer 包管理器安装 Meilisearch PHP SDK:

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

然后,在您应用程序的 .env 文件中设置 SCOUT_DRIVER 环境变量以及您的 Meilisearch hostkey 凭据:

ini
SCOUT_DRIVER=meilisearch
MEILISEARCH_HOST=http://127.0.0.1:7700
MEILISEARCH_KEY=masterKey

有关 Meilisearch 的更多信息,请查阅 Meilisearch 文档

此外,您应该确保安装与您的 Meilisearch 二进制版本兼容的 meilisearch/meilisearch-php 版本,具体请查看 Meilisearch 关于二进制兼容性的文档

WARNING

在升级使用 Meilisearch 的应用程序上的 Scout 时,您应始终 查看任何其他破坏性更改 到 Meilisearch 服务本身。

Typesense

Typesense 是一个快速的开源搜索引擎,支持关键字搜索、语义搜索、地理搜索和向量搜索。

您可以 自托管 Typesense 或使用 Typesense Cloud

要开始使用 Typesense 和 Scout,请通过 Composer 包管理器安装 Typesense PHP SDK:

shell
composer require typesense/typesense-php

然后,在您应用程序的 .env 文件中设置 SCOUT_DRIVER 环境变量以及您的 Typesense 主机和 API 密钥凭据:

ini
SCOUT_DRIVER=typesense
TYPESENSE_API_KEY=masterKey
TYPESENSE_HOST=localhost

如果您使用 Laravel Sail,您可能需要调整 TYPESENSE_HOST 环境变量以匹配 Docker 容器名称。您还可以选择性地指定安装的端口、路径和协议:

ini
TYPESENSE_PORT=8108
TYPESENSE_PATH=
TYPESENSE_PROTOCOL=http

有关您应用程序的 Typesense 集合的其他设置和模式定义,请查阅您应用程序的 config/scout.php 配置文件。有关 Typesense 的更多信息,请查阅 Typesense 文档

为 Typesense 准备存储数据

使用 Typesense 时,您的可搜索模型必须定义一个 toSearchableArray 方法,该方法将模型的主键转换为字符串,并将创建日期转换为 UNIX 时间戳:

php
/**
 * 获取模型的可索引数据数组。
 *
 * @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:flushscout:import,这将删除所有现有的索引数据并重新创建模式。或者,您可以使用 Typesense 的 API 修改集合的模式,而无需删除任何索引数据。

如果您的可搜索模型是软可删除的,您应该在应用程序的 config/scout.php 配置文件中定义一个 __soft_deleted 字段在模型对应的 Typesense 模式中:

php
User::class => [
    'collection-schema' => [
        'fields' => [
            // ...
            [
                'name' => '__soft_deleted',
                'type' => 'int32',
                'optional' => true,
            ],
        ],
    ],
],

动态搜索参数

Typesense 允许您在通过 options 方法执行搜索操作时动态修改您的 搜索参数

php
use App\Models\Todo;

Todo::search('杂货')->options([
    'query_by' => 'title, description'
])->get();

配置

配置模型索引

每个 Eloquent 模型与给定的搜索“索引”同步,该索引包含该模型的所有可搜索记录。换句话说,您可以将每个索引视为 MySQL 表。默认情况下,每个模型将被持久化到与模型的典型“表”名称匹配的索引中。通常,这是模型名称的复数形式;但是,您可以通过覆盖模型上的 searchableAs 方法来自定义模型的索引:

php
<?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
<?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)将仅对正确类型的数据执行过滤操作(>< 等)。因此,在使用这些搜索引擎并自定义可搜索数据时,您应该确保将数值转换为其正确类型:

php
public function toSearchableArray()
{
    return [
        'id' => (int) $this->id,
        'name' => $this->name,
        'price' => (float) $this->price,
    ];
}

配置索引设置(Algolia)

有时,你可能希望在 Algolia 索引上配置其他设置。虽然可以通过 Algolia UI 进行管理,但有时直接在应用程序的 config/scout.php 配置文件中管理索引配置的期望状态更加高效。

这种方法允许你通过应用程序的自动化部署流程部署这些设置,避免手动配置,并确保多个环境之间的一致性。你可以配置可筛选属性(filterable attributes)、排名(ranking)、分面(faceting),或 任何其他支持的设置

要开始配置,请在应用程序的 config/scout.php 配置文件中为每个索引添加设置:

php
use App\Models\User;
use App\Models\Flight;

'algolia' => [
    'id' => env('ALGOLIA_APP_ID', ''),
    'secret' => env('ALGOLIA_SECRET', ''),
    'index-settings' => [
        User::class => [
            'searchableAttributes' => ['id', 'name', 'email'],
            'attributesForFaceting'=> ['filterOnly(email)'],
            // 其他设置字段...
        ],
        Flight::class => [
            'searchableAttributes'=> ['id', 'destination'],
        ],
    ],
],

如果某个索引的底层模型支持软删除(soft deletable),并且已包含在 index-settings 数组中,Scout 将自动在该索引上启用软删除模型的分面支持。如果该模型索引没有其他需要定义的分面属性,你可以为该模型在 index-settings 数组中添加一个空条目:

php
'index-settings' => [
    Flight::class => []
],

配置完应用程序的索引设置后,你必须执行 scout:sync-index-settings Artisan 命令。此命令将通知 Algolia 你的当前索引设置。为了方便起见,你可能希望将此命令纳入部署流程:

shell
php artisan scout:sync-index-settings

配置可过滤数据和索引设置(Meilisearch)

与 Scout 的其他驱动程序不同,Meilisearch 要求您预先定义索引搜索设置,例如可过滤属性、可排序属性和 其他支持的设置字段

可过滤属性是您计划在调用 Scout 的 where 方法时进行过滤的任何属性,而可排序属性是您计划在调用 Scout 的 orderBy 方法时进行排序的任何属性。要定义您的索引设置,请调整您应用程序的 scout 配置文件中 meilisearch 配置条目的 index-settings 部分:

php
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 数组中:

php
'index-settings' => [
    Flight::class => []
],

配置完应用程序的索引设置后,您必须调用 scout:sync-index-settings Artisan 命令。此命令将通知 Meilisearch 您当前配置的索引设置。为了方便起见,您可能希望将此命令作为部署过程的一部分:

shell
php artisan scout:sync-index-settings

配置模型 ID

默认情况下,Scout 将使用模型的主键作为存储在搜索索引中的模型唯一 ID / 键。如果您需要自定义此行为,可以在模型上覆盖 getScoutKeygetScoutKeyName 方法:

php
<?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
<?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 来启用用户识别:

ini
SCOUT_IDENTIFY=true

启用此功能还将把请求的 IP 地址和经过身份验证的用户的主标识符传递给 Algolia,因此这些数据与用户进行的任何搜索请求相关联。

数据库 / 集合引擎

数据库引擎

WARNING

数据库引擎当前支持 MySQL 和 PostgreSQL。

如果您的应用程序与小型到中型数据库交互或工作负载较轻,您可能会发现使用 Scout 的“数据库”引擎更方便。数据库引擎将在现有数据库中使用“where like”子句和全文索引来过滤结果,以确定与您的查询相应的搜索结果。

要使用数据库引擎,您只需将 SCOUT_DRIVER 环境变量的值设置为 database,或在应用程序的 scout 配置文件中直接指定 database 驱动程序:

ini
SCOUT_DRIVER=database

一旦您将数据库引擎指定为首选驱动程序,您必须 配置可搜索数据。然后,您可以开始对模型执行 搜索查询。使用数据库引擎时,不需要进行搜索引擎索引,例如需要为 Algolia、Meilisearch 或 Typesense 索引播种的索引。

自定义数据库搜索策略

默认情况下,数据库引擎将对您已 配置为可搜索 的每个模型属性执行“where like”查询。然而,在某些情况下,这可能会导致性能不佳。因此,数据库引擎的搜索策略可以配置,以便某些指定的列使用全文搜索查询,或者仅使用“where like”约束来搜索字符串的前缀(example%),而不是在整个字符串中搜索(%example%)。

要定义此行为,您可以将 PHP 属性分配给模型的 toSearchableArray 方法。未分配额外搜索策略行为的任何列将继续使用默认的“where like”策略:

php
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,
    ];
}

WARNING

在指定某列应使用全文查询约束之前,请确保该列已分配 全文索引

集合引擎

虽然您可以在本地开发期间使用 Algolia、Meilisearch 或 Typesense 搜索引擎,但您可能会发现使用“集合”引擎更方便。集合引擎将使用“where”子句和集合过滤来确定与您的查询相应的搜索结果。当使用此引擎时,不需要“索引”您的可搜索模型,因为它们将直接从本地数据库中检索。

要使用集合引擎,您只需将 SCOUT_DRIVER 环境变量的值设置为 collection,或在应用程序的 scout 配置文件中直接指定 collection 驱动程序:

ini
SCOUT_DRIVER=collection

一旦您将集合驱动程序指定为首选驱动程序,您可以开始对模型执行 搜索查询。使用集合引擎时,不需要进行搜索引擎索引,例如需要为 Algolia、Meilisearch 或 Typesense 索引播种的索引。

与数据库引擎的区别

乍一看,“数据库”和“集合”引擎非常相似。它们都直接与您的数据库交互以检索搜索结果。然而,集合引擎不使用全文索引或 LIKE 子句来查找匹配记录。相反,它会提取所有可能的记录,并使用 Laravel 的 Str::is 辅助函数来确定搜索字符串是否存在于模型属性值中。

集合引擎是最便携的搜索引擎,因为它适用于 Laravel 支持的所有关系数据库(包括 SQLite 和 SQL Server);但是,它的效率低于 Scout 的数据库引擎。

索引

批量导入

如果您在现有项目中安装 Scout,您可能已经有需要导入到索引中的数据库记录。Scout 提供了一个 scout:import Artisan 命令,您可以使用它将所有现有记录导入到搜索索引中:

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

flush 命令可用于从搜索索引中删除模型的所有记录:

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

修改导入查询

如果您希望修改用于检索所有模型以进行批量导入的查询,您可以在模型上定义一个 makeAllSearchableUsing 方法。这是添加任何必要的急切关系加载的好地方,以便在导入模型之前:

php
use Illuminate\Database\Eloquent\Builder;

/**
 * 修改用于检索模型的查询,以使所有模型可搜索。
 */
protected function makeAllSearchableUsing(Builder $query): Builder
{
    return $query->with('author');
}

WARNING

当使用队列批量导入模型时,makeAllSearchableUsing 方法可能不适用。处理作业时,关系不会 恢复

添加记录

一旦您将 Laravel\Scout\Searchable 特性添加到模型,您只需 savecreate 模型实例,它将自动添加到您的搜索索引中。如果您已配置 Scout 使用队列,此操作将在后台由队列工作程序执行:

php
use App\Models\Order;

$order = new Order;

// ...

$order->save();

通过查询添加记录

如果您希望通过 Eloquent 查询将一组模型添加到搜索索引中,您可以将 searchable 方法链接到 Eloquent 查询。searchable 方法将 分块结果 的查询并将记录添加到搜索索引中。同样,如果您已配置 Scout 使用队列,所有块将在后台由队列工作程序导入:

php
use App\Models\Order;

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

您还可以在 Eloquent 关系实例上调用 searchable 方法:

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

或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 searchable 方法,将模型实例添加到其对应的索引中:

php
$orders->searchable();

NOTE

searchable 方法可以视为“upsert”操作。换句话说,如果模型记录已经在您的索引中,它将被更新。如果它在搜索索引中不存在,它将被添加到索引中。

更新记录

要更新可搜索模型,您只需更新模型实例的属性并将模型 save 到数据库。Scout 将自动将更改持久化到您的搜索索引:

php
use App\Models\Order;

$order = Order::find(1);

// 更新订单...

$order->save();

您还可以在 Eloquent 查询实例上调用 searchable 方法以更新一组模型。如果模型在您的搜索索引中不存在,它们将被创建:

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

如果您希望更新关系中所有模型的搜索索引记录,可以在关系实例上调用 searchable

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

或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 searchable 方法,以更新其对应索引中的模型实例:

php
$orders->searchable();

在导入之前修改记录

有时,您可能需要在使集合模型可搜索之前准备它。例如,您可能希望急切加载一个关系,以便可以有效地将关系数据添加到搜索索引中。为此,请在相应模型上定义一个 makeSearchableUsing 方法:

php
use Illuminate\Database\Eloquent\Collection;

/**
 * 修改正在使可搜索的模型集合。
 */
public function makeSearchableUsing(Collection $models): Collection
{
    return $models->load('author');
}

删除记录

要从索引中删除记录,您只需从数据库中 delete 模型。这可以在使用 软删除 模型时完成:

php
use App\Models\Order;

$order = Order::find(1);

$order->delete();

如果您不想在删除记录之前检索模型,您可以在 Eloquent 查询实例上使用 unsearchable 方法:

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

如果您希望删除关系中所有模型的搜索索引记录,可以在关系实例上调用 unsearchable

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

或者,如果您已经在内存中有一组 Eloquent 模型,您可以在集合实例上调用 unsearchable 方法,以从其对应索引中删除模型实例:

php
$orders->unsearchable();

要从其对应索引中删除所有模型记录,您可以调用 removeAllFromSearch 方法:

php
Order::removeAllFromSearch();

暂停索引

有时,您可能需要对模型执行一批 Eloquent 操作,而不将模型数据同步到搜索索引。您可以使用 withoutSyncingToSearch 方法来实现此目的。此方法接受一个单一的闭包,该闭包将立即执行。在闭包内发生的任何模型操作都不会同步到模型的索引:

php
use App\Models\Order;

Order::withoutSyncingToSearch(function () {
    // 执行模型操作...
});

有条件可搜索的模型实例

有时,您可能只希望在某些条件下使模型可搜索。例如,假设您有一个 App\Models\Post 模型,它可能处于“草稿”和“已发布”两种状态之一。您可能只希望允许“已发布”的帖子可搜索。为此,您可以在模型上定义一个 shouldBeSearchable 方法:

php
/**
 * 确定模型是否应该可搜索。
 */
public function shouldBeSearchable(): bool
{
    return $this->isPublished();
}

shouldBeSearchable 方法仅在通过 savecreate 方法、查询或关系操作模型时应用。直接使用 searchable 方法使模型或集合可搜索将覆盖 shouldBeSearchable 方法的结果。

WARNING

shouldBeSearchable 方法在使用 Scout 的“数据库”引擎时不适用,因为所有可搜索数据始终存储在数据库中。要在使用数据库引擎时实现类似行为,您应该使用 where 子句

搜索

您可以使用 search 方法开始搜索模型。搜索方法接受一个字符串,该字符串将用于搜索您的模型。然后,您应该在搜索查询上链接 get 方法,以检索与给定搜索查询匹配的 Eloquent 模型:

php
use App\Models\Order;

$orders = Order::search('星际迷航')->get();

由于 Scout 搜索返回的是 Eloquent 模型的集合,您甚至可以直接从路由或控制器返回结果,它们将自动转换为 JSON:

php
use App\Models\Order;
use Illuminate\Http\Request;

Route::get('/search', function (Request $request) {
    return Order::search($request->search)->get();
});

如果您希望在转换为 Eloquent 模型之前获取原始搜索结果,可以使用 raw 方法:

php
$orders = Order::search('星际迷航')->raw();

自定义索引

搜索查询通常将在模型的 searchableAs 方法指定的索引上执行。但是,您可以使用 within 方法指定应搜索的自定义索引:

php
$orders = Order::search('星际迷航')
    ->within('tv_shows_popularity_desc')
    ->get();

条件子句

Scout 允许您向搜索查询添加简单的“where”子句。目前,这些子句仅支持基本的数字相等检查,主要用于通过所有者 ID 对搜索查询进行范围限制:

php
use App\Models\Order;

$orders = Order::search('星际迷航')->where('user_id', 1)->get();

此外,whereIn 方法可用于验证给定列的值是否包含在给定数组中:

php
$orders = Order::search('星际迷航')->whereIn(
    'status', ['open', 'paid']
)->get();

whereNotIn 方法验证给定列的值是否不包含在给定数组中:

php
$orders = Order::search('星际迷航')->whereNotIn(
    'status', ['closed']
)->get();

由于搜索索引不是关系数据库,因此当前不支持更高级的“where”子句。

WARNING

如果您的应用程序使用 Meilisearch,您必须在使用 Scout 的“where”子句之前配置应用程序的 可过滤属性

分页

除了检索模型集合外,您还可以使用 paginate 方法对搜索结果进行分页。此方法将返回一个 Illuminate\Pagination\LengthAwarePaginator 实例,就像您对传统 Eloquent 查询进行分页一样 分页

php
use App\Models\Order;

$orders = Order::search('星际迷航')->paginate();

您可以通过将数量作为第一个参数传递给 paginate 方法来指定每页检索多少模型:

php
$orders = Order::search('星际迷航')->paginate(15);

检索结果后,您可以使用 Blade 显示结果并渲染页面链接,就像您对传统 Eloquent 查询进行分页一样:

html
<div class="container">
    @foreach ($orders as $order)
        {{ $order->price }}
    @endforeach
</div>

{{ $orders->links() }}

当然,如果您希望将分页结果作为 JSON 检索,您可以直接从路由或控制器返回分页器实例:

php
use App\Models\Order;
use Illuminate\Http\Request;

Route::get('/orders', function (Request $request) {
    return Order::search($request->input('query'))->paginate(15);
});

WARNING

由于搜索引擎不知道您的 Eloquent 模型的全局范围定义,因此在使用 Scout 分页的应用程序中,您不应使用全局范围。或者,您应该在通过 Scout 搜索时重新创建全局范围的约束。

软删除

如果您的索引模型 软删除,并且您需要搜索您的软删除模型,请将 config/scout.php 配置文件中的 soft_delete 选项设置为 true

php
'soft_delete' => true,

当此配置选项为 true 时,Scout 将不会从搜索索引中删除软删除模型。相反,它将在索引记录上设置一个隐藏的 __soft_deleted 属性。然后,您可以使用 withTrashedonlyTrashed 方法在搜索时检索软删除记录:

php
use App\Models\Order;

// 在检索结果时包含已删除记录...
$orders = Order::search('星际迷航')->withTrashed()->get();

// 仅在检索结果时包含已删除记录...
$orders = Order::search('星际迷航')->onlyTrashed()->get();

NOTE

当使用 forceDelete 永久删除软删除模型时,Scout 将自动将其从搜索索引中删除。

自定义引擎搜索

如果您需要对引擎的搜索行为进行高级自定义,您可以将闭包作为第二个参数传递给 search 方法。例如,您可以使用此回调在将搜索查询传递给 Algolia 之前向搜索选项添加地理位置数据:

php
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 查询构建器实例作为参数:

php
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 抽象类。此抽象类包含您的自定义引擎必须实现的八个方法:

php
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 方法:

php
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

php
'driver' => 'mysql',