Skip to content

Laravel Scout

介绍

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

目前,Scout 附带一个 Algolia 驱动程序;然而,编写自定义驱动程序很简单,您可以自由扩展 Scout 以实现自己的搜索实现。

安装

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

php
composer require laravel/scout

接下来,您应该将 ScoutServiceProvider 添加到 config/app.php 配置文件的 providers 数组中:

php
Laravel\Scout\ScoutServiceProvider::class,

在注册 Scout 服务提供者之后,您应该使用 vendor:publish Artisan 命令发布 Scout 配置。此命令将 scout.php 配置文件发布到您的 config 目录:

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

最后,将 Laravel\Scout\Searchable trait 添加到您希望进行搜索的模型中。此 trait 将注册一个模型观察者,以保持模型与您的搜索驱动程序同步:

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;
}

队列

虽然使用 Scout 并不严格要求,但在使用该库之前,您应该强烈考虑配置一个 队列驱动程序。运行队列工作程序将允许 Scout 将所有操作排队,以将您的模型信息同步到您的搜索索引,从而为您的应用程序的 Web 界面提供更好的响应时间。

一旦您配置了队列驱动程序,请将 config/scout.php 配置文件中的 queue 选项的值设置为 true

php
'queue' => true,

驱动程序先决条件

Algolia

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

php
composer require algolia/algoliasearch-client-php

配置

配置模型索引

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

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;

    /**
     * 获取模型的索引名称。
     *
     * @return string
     */
    public function searchableAs()
    {
        return 'posts_index';
    }
}

配置可搜索数据

默认情况下,给定模型的整个 toArray 形式将被持久化到其搜索索引中。如果您想自定义同步到搜索索引的数据,可以覆盖模型上的 toSearchableArray 方法:

php
<?php

namespace App;

use Laravel\Scout\Searchable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Searchable;

    /**
     * 获取模型的可索引数据数组。
     *
     * @return array
     */
    public function toSearchableArray()
    {
        $array = $this->toArray();

        // 自定义数组...

        return $array;
    }
}

索引

批量导入

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

php
php artisan scout:import "App\Post"

添加记录

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

php
$order = new App\Order;

// ...

$order->save();

通过查询添加

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

php
// 通过 Eloquent 查询添加...
App\Order::where('price', '>', 100)->searchable();

// 您还可以通过关系添加记录...
$user->orders()->searchable();

// 您还可以通过集合添加记录...
$orders->searchable();

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

更新记录

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

php
$order = App\Order::find(1);

// 更新订单...

$order->save();

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

php
// 通过 Eloquent 查询更新...
App\Order::where('price', '>', 100)->searchable();

// 您还可以通过关系更新...
$user->orders()->searchable();

// 您还可以通过集合更新...
$orders->searchable();

删除记录

要从索引中删除记录,只需从数据库中 delete 模型即可。这种删除形式甚至与 软删除 模型兼容:

php
$order = App\Order::find(1);

$order->delete();

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

php
// 通过 Eloquent 查询删除...
App\Order::where('price', '>', 100)->unsearchable();

// 您还可以通过关系删除...
$user->orders()->unsearchable();

// 您还可以通过集合删除...
$orders->unsearchable();

暂停索引

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

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

搜索

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

php
$orders = App\Order::search('Star Trek')->get();

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

php
use Illuminate\Http\Request;

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

Where 子句

Scout 允许您向搜索查询添加简单的“where”子句。目前,这些子句仅支持基本的数字相等检查,主要用于通过租户 ID 限制搜索查询。由于搜索索引不是关系数据库,目前不支持更高级的“where”子句:

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

分页

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

php
$orders = App\Order::search('Star Trek')->paginate();

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

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

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

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

{{ $orders->links() }}

自定义引擎

编写引擎

如果内置的 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 map($results, $model);

您可能会发现查看 Laravel\Scout\Engines\AlgoliaEngine 类上这些方法的实现很有帮助。此类将为您提供一个良好的起点,以了解如何在自己的引擎中实现这些方法。

注册引擎

一旦您编写了自定义引擎,您可以使用 Scout 引擎管理器的 extend 方法将其注册到 Scout。您应该从 AppServiceProviderboot 方法或应用程序使用的任何其他服务提供者中调用 extend 方法。例如,如果您编写了一个 MySqlSearchEngine,您可以这样注册它:

php
use Laravel\Scout\EngineManager;

/**
 * 启动任何应用程序服务。
 *
 * @return void
 */
public function boot()
{
    resolve(EngineManager::class)->extend('mysql', function () {
        return new MySqlSearchEngine;
    });
}

一旦您的引擎注册完毕,您可以在 config/scout.php 配置文件中将其指定为默认的 Scout driver

php
'driver' => 'mysql',