Update Index Prefix
Index Prefix no longer auto concatenates with _
ES_INDEX_PREFIX=my_prefixES_INDEX_PREFIX=my_prefix_
Update Index Prefix
Index Prefix no longer auto concatenates with _
ES_INDEX_PREFIX=my_prefixES_INDEX_PREFIX=my_prefix_
Model id Field
$model->_id
is now $model->id
and is an alias for Elasticsearch’s _id
field. This is to maintain consistency with Laravel’s Eloquent ORM.
If you were using a distinct id
field alongside the default _id
field then this will no longer work. You will need to update your code to use a different field name for your previous id
field.
MAX_SIZE → $defaultLimit
const MAX_SIZE
has been replaced with protected $defaultLimit
for defining the default limit for search results.
use PDPhilip\Elasticsearch\Eloquent\Model;
class Product extends Model{ const MAX_SIZE = 10000; protected $defaultLimit = 10000; protected $connection = 'elasticsearch';
where() clause now runs a term query
The where()
clause now runs a term query, not a match as before.
If you still want to run a match query, then replace
where()
withwhereMatch()
where('name', 'John')whereMatch('name', 'John')
If exact matching is a better fit, then ensure the field has a
keyword
mapping.
$index->keyword('name'); //or$index->text('name', hasKeyword: true);
orderByRandom() → functionScore()
orderByRandom()
method has been removed. Use functionScore()
instead.
Product::where('orders', '>', 2)->orderByRandom('created_at',rand(1, 100)')->get();Product::functionScore('random_score', ['seed' => rand(1, 100), 'field' => 'created_at'], function ($query) { $query->where('orders', '>', 2);})->get();
Full text search options replaced
asFuzzy()
, setMinShouldMatch()
, setBoost()
search methods been removed. Use $options
instead.
Product::searchTerm('espreso tme') ->asFuzzy()->setMinShouldMatch(2)->setBoost(2) ->get();Product::searchTerm('espreso tme', function (SearchOptions $options) { $options->searchFuzzy(); $options->boost(2); $options->minimumShouldMatch(2);})->get();
Legacy search query builder methods removed
All {xx}->search()
methods have been removed. Use {multi_match}->get()
instead.
Book::term('Eric')->orTerm('Lean')->andTerm('Startup')->search();Book::searchTerm('Eric')->orSearchTerm('Lean')->searchTerm('Startup')->get();
Book::phrase('United States')->orPhrase('United Kingdom')->search();Book::searchPhrase('United States')->orSearchPhrase('United Kingdom')->get();
$results = Book::term('Eric')->fields(['title', 'author', 'description'])->search();Book::searchTerm('Eric',['title', 'author', 'description'])->get();
$results = Book::fuzzyTerm('quikc')->orFuzzyTerm('brwn')->andFuzzyTerm('foks')->search();Book::searchTerm('quikc',['fuzziness' => 'AUTO']) ->orSearchTerm('brwn',['fuzziness' => 'AUTO']) ->searchTerm('foks',['fuzziness' => 'AUTO']) ->get();
distinct() and groupBy() operate differently
The distinct() and groupBy() methods have been rebuilt and work differently.
distinct()
uses: Nested Term Aggs
groupBy()
uses: Composite Aggregation
Important: distinct()
now executes the aggregation:
UserLog::where('created_at', '>=', Carbon::now()->subDays(30)) ->distinct()->get('user_id');UserLog::where('created_at', '>=', Carbon::now()->subDays(30)) ->distinct('user_id');
Please review the Distinct & GroupBy documentation carefully for more information.
IndexBlueprint/AnalyzerBlueprint → Blueprint
IndexBlueprint
and AnalyzerBlueprint
have been removed and replaced with a single Blueprint
class
use PDPhilip\Elasticsearch\Schema\IndexBlueprint; use PDPhilip\Elasticsearch\Schema\AnalyzerBlueprint;use PDPhilip\Elasticsearch\Schema\Blueprint;
Schema::hasIndex → Schema::indexExists
Schema::hasIndex
has been removed. Use Schema::hasTable
or Schema::indexExists
instead.
Schema::hasIndex('index_name');Schema::hasTable('index_name'); //orSchema::indexExists('index_name');
Schema::dsl → Schema::indices()
Schema::dsl($method,$dsl)
has been removed. Use Schema::indices()->{method}
Schema::dsl('close', ['index' => 'my_index']);Schema::indices()->close(['index' => 'my_index']);
geo($field)
field property has been replaced with geoPoint($field)
;
$index->geo('last_login');$index->geoPoint('last_login');
->index($bool)
field property has been replaced with ->indexField($bool)
;
// Remove from index, can't search by this field but can still use for aggregations: $index->integer('age')->index(false);$index->integer('age')->indexField(false);
alias()
field type has been removed. Use aliasField()
instead.
$index->alias('comments', 'notes');$index->aliasField('comments', 'notes');
settings()
method has been replaced with withSetting()
$index->settings('number_of_shards', 3);$index->withSetting('number_of_shards', 3);
map()
method has been replaced with withMapping()
$index->map('date_detection', false);$index->withMapping('date_detection', false);
analyzer()
method has been replaced with addAnalyzer()
$settings->analyzer('my_custom_analyzer')$settings->addAnalyzer('my_custom_analyzer')
tokenizer()
method has been replaced with addTokenizer()
$settings->tokenizer('punctuation')$settings->addTokenizer('punctuation')
charFilter()
method has been replaced with addCharFilter()
$settings->charFilter('emoticons')$settings->addCharFilter('emoticons')
filter()
method has been replaced with addFilter()
//Custom Field Mapping $settings->filter('english_stop')$index->addFilter('english_stop')
Dynamic indices are now managed by the DynamicIndex
trait.
namespace App\Models;
use PDPhilip\Elasticsearch\Eloquent\DynamicIndex;use PDPhilip\Elasticsearch\Eloquent\Model
class PageHit extends Model{ use DynamicIndex; protected $connection = 'elasticsearch';
protected $index = 'page_hits_*'; // Dynamic index pattern
Replace setIndex()
with setSuffix()
in the context of saving a record.
$pageHit = new PageHit;$pageHit->page_id = 4;$pageHit->ip = $someIp;// Set the specific index for the new record $pageHit->setIndex('page_hits_' . date('Y-m-d')); $pageHit->setSuffix('_' . date('Y-m-d'));$pageHit->save();
getRecordIndex()
method has been replaced with getRecordSuffix()
.
$pageHit = PageHit::first(); $index = $pageHit->getRecordIndex(); $suffix = $pageHit->getRecordSuffix();
Replace setIndex()
with withSuffix()
in the context of querying indexes (within the given suffix).
$model = new PageHit; $model->setIndex('page_hits_2023-01-01'); $pageHits = $model->where('page_id', 3)->get();$pageHits = PageHit::withSuffix('_2023-01-01')->where('page_id', 3)->get();
Multiple Aggs are now returned as a flat array
Product::agg(['avg','sum'],'orders');
v4 | v5 |
---|---|
|
|
Matix aggs results differ
Product::matrix(['orders','status']);
v4 | v5 |
---|---|
|
|
withoutRefresh()
$product->saveWithoutRefresh()$product->withoutRefresh()->save()
Product::createWithoutRefresh([data])Product::withoutRefresh()->create([data])
'elasticsearch' => [ 'driver' => 'elasticsearch', 'auth_type' => env('ES_AUTH_TYPE', 'http'), //http or cloud 'hosts' => explode(',', env('ES_HOSTS', 'http://localhost:9200')), 'username' => env('ES_USERNAME', ''), 'password' => env('ES_PASSWORD', ''), 'cloud_id' => env('ES_CLOUD_ID', ''), 'api_id' => env('ES_API_ID', ''), 'api_key' => env('ES_API_KEY', ''), 'ssl_cert' => env('ES_SSL_CA', ''), 'ssl' => [ 'cert' => env('ES_SSL_CERT', ''), 'cert_password' => env('ES_SSL_CERT_PASSWORD', ''), 'key' => env('ES_SSL_KEY', ''), 'key_password' => env('ES_SSL_KEY_PASSWORD', ''), ], 'index_prefix' => env('ES_INDEX_PREFIX', false), 'options' => [ 'bypass_map_validation' => env('ES_OPT_BYPASS_MAP_VALIDATION', false), 'logging' => env('ES_OPT_LOGGING', false), 'ssl_verification' => env('ES_OPT_VERIFY_SSL', true), 'retires' => env('ES_OPT_RETRIES', null), 'meta_header' => env('ES_OPT_META_HEADERS', true), 'default_limit' => env('ES_OPT_DEFAULT_LIMIT', 1000), 'allow_id_sort' => env('ES_OPT_ID_SORTABLE', false), ],],