A library to enable you to safely rotate indexes with no downtime to end users.
In many situations, Elasticsearch is used as an ephemeral datastore used to take structured or relational data and make it fast to search on that data. Often this is achieved via scheduled jobs that read data from a permanent datastore (such as MySQL or Postgres) and translate it into an Elasticsearch index.
In many cases, rebuilding an index requires a clean slate so that the entire index is rebuilt. How do you do this without interrupting end users searching on that index? The answer is a rotating index.
Here the user's search is fully disrupted when the index is first removed, and only partially available while the index is being rebuilt. While the index is being rebuilt, users get incomplete data.
Here the user's search is never disrupted because we construct a new index and after it is built/settled, we change the what index to search by the client.
composer require zumba/elasticsearch-index-rotate
<?php
$client = new \Elasticsearch\Client();
$indexRotator = new \Zumba\ElasticsearchRotator\IndexRotator($client, 'pizza_shops');
$client->search([
'index' => $indexRotator->getPrimaryIndex(), // Get the current primary!
'type' => 'shop',
'body' => [] //...
]);
<?php
$client = new \Elasticsearch\Client();
$indexRotator = new IndexRotator($client, 'pizza_shops');
// Build your index here
$newlyBuiltIndexName = 'my_new_built_index_name';
$indexRotator->copyPrimaryIndexToSecondary();
$indexRotator->setPrimaryIndex($newlyBuiltIndexName);
// optionally remove the old index right now
$indexRotator->deleteSecondaryIndices();
<?php
use \Elasticsearch\Client;
use \Zumba\ElastsearchRotator\IndexRotator;
class MySearchIndex {
const INDEX_PREFIX = 'pizza_shops';
public function __constructor(\Elasticsearch\Client $client) {
$this->client = $client;
}
public function search($params) {
$indexRotator = new IndexRotator($this->client, static::INDEX_PREFIX);
return $client->search([
'index' => $indexRotator->getPrimaryIndex(), // Get the current primary!
'type' => 'shop',
'body' => $params
]);
}
public function rebuildIndex() {
$indexRotator = new IndexRotator($client, static::INDEX_PREFIX);
$newlyBuiltIndexName = $this->buildIndex($client);
$indexRotator->copyPrimaryIndexToSecondary();
$indexRotator->setPrimaryIndex($newlyBuiltIndexName);
// optionally remove the old index right now
$indexRotator->deleteSecondaryIndices();
}
private function buildIndex(\Elasticsearch\Client $client) {
$newIndex = static::INDEX_PREFIX . '_' . time();
// get data and build index for `$newIndex`
return $newIndex;
}
}