Restore a snapshot

Restore a snapshot

This guide shows you how to restore a snapshot. Snapshots are a convenient way to store a copy of your data outside of a cluster. You can restore a snapshot to recover indices and data streams after deletion or a hardware failure. You can also use snapshots to transfer data between clusters.

In this guide, you’ll learn how to:

  • Get a list of available snapshots
  • Restore an index or data stream from a snapshot
  • Restore a feature state
  • Restore an entire cluster
  • Monitor the restore operation
  • Cancel an ongoing restore

This guide also provides tips for restoring to another cluster and troubleshooting common restore errors.

Prerequisites

  • To use Kibana’s Snapshot and Restore feature, you must have the following permissions:

  • You can only restore a snapshot to a running cluster with an elected master node. The snapshot’s repository must be registered and available to the cluster.

  • The snapshot and cluster versions must be compatible. See Snapshot compatibility.
  • To restore a snapshot, the cluster’s global metadata must be writable. Ensure there aren’t any cluster blocks that prevent writes. The restore operation ignores index blocks.
  • Before you restore a data stream, ensure the cluster contains a matching index template with data stream enabled. To check, use Kibana’s Index Management feature or the get index template API:

    1. resp = client.indices.get_index_template(
    2. name="*",
    3. filter_path="index_templates.name,index_templates.index_template.index_patterns,index_templates.index_template.data_stream",
    4. )
    5. print(resp)
    1. response = client.indices.get_index_template(
    2. name: '*',
    3. filter_path: 'index_templates.name,index_templates.index_template.index_patterns,index_templates.index_template.data_stream'
    4. )
    5. puts response
    1. const response = await client.indices.getIndexTemplate({
    2. name: "*",
    3. filter_path:
    4. "index_templates.name,index_templates.index_template.index_patterns,index_templates.index_template.data_stream",
    5. });
    6. console.log(response);
    1. GET _index_template/*?filter_path=index_templates.name,index_templates.index_template.index_patterns,index_templates.index_template.data_stream

    If no such template exists, you can create one or restore a cluster state that contains one. Without a matching index template, a data stream can’t roll over or create backing indices.

  • If your snapshot contains data from App Search or Workplace Search, ensure you’ve restored the Enterprise Search encryption key before restoring the snapshot.

Considerations

When restoring data from a snapshot, keep the following in mind:

  • If you restore a data stream, you also restore its backing indices.
  • You can only restore an existing index if it’s closed and the index in the snapshot has the same number of primary shards.
  • You can’t restore an existing open index. This includes backing indices for a data stream.
  • The restore operation automatically opens restored indices, including backing indices.
  • You can restore only a specific backing index from a data stream. However, the restore operation doesn’t add the restored backing index to any existing data stream.

Get a list of available snapshots

To view a list of available snapshots in Kibana, go to the main menu and click Stack Management > Snapshot and Restore.

You can also use the get repository API and the get snapshot API to find snapshots that are available to restore. First, use the get repository API to fetch a list of registered snapshot repositories.

  1. resp = client.snapshot.get_repository()
  2. print(resp)
  1. response = client.snapshot.get_repository
  2. puts response
  1. const response = await client.snapshot.getRepository();
  2. console.log(response);
  1. GET _snapshot

Then use the get snapshot API to get a list of snapshots in a specific repository. This also returns each snapshot’s contents.

  1. resp = client.snapshot.get(
  2. repository="my_repository",
  3. snapshot="*",
  4. verbose=False,
  5. )
  6. print(resp)
  1. response = client.snapshot.get(
  2. repository: 'my_repository',
  3. snapshot: '*',
  4. verbose: false
  5. )
  6. puts response
  1. const response = await client.snapshot.get({
  2. repository: "my_repository",
  3. snapshot: "*",
  4. verbose: "false",
  5. });
  6. console.log(response);
  1. GET _snapshot/my_repository/*?verbose=false

Restore an index or data stream

You can restore a snapshot using Kibana’s Snapshot and Restore feature or the restore snapshot API.

By default, a restore request attempts to restore all regular indices and regular data streams in a snapshot. In most cases, you only need to restore a specific index or data stream from a snapshot. However, you can’t restore an existing open index.

If you’re restoring data to a pre-existing cluster, use one of the following methods to avoid conflicts with existing indices and data streams:

Delete and restore

The simplest way to avoid conflicts is to delete an existing index or data stream before restoring it. To prevent the accidental re-creation of the index or data stream, we recommend you temporarily stop all indexing until the restore operation is complete.

If the action.destructive_requires_name cluster setting is false, don’t use the delete index API to target the * or .* wildcard pattern. If you use Elasticsearch’s security features, this will delete system indices required for authentication. Instead, target the *,-.* wildcard pattern to exclude these system indices and other index names that begin with a dot (.).

  1. resp = client.indices.delete(
  2. index="my-index",
  3. )
  4. print(resp)
  5. resp1 = client.indices.delete_data_stream(
  6. name="logs-my_app-default",
  7. )
  8. print(resp1)
  1. response = client.indices.delete(
  2. index: 'my-index'
  3. )
  4. puts response
  5. response = client.indices.delete_data_stream(
  6. name: 'logs-my_app-default'
  7. )
  8. puts response
  1. const response = await client.indices.delete({
  2. index: "my-index",
  3. });
  4. console.log(response);
  5. const response1 = await client.indices.deleteDataStream({
  6. name: "logs-my_app-default",
  7. });
  8. console.log(response1);
  1. # Delete an index
  2. DELETE my-index
  3. # Delete a data stream
  4. DELETE _data_stream/logs-my_app-default

In the restore request, explicitly specify any indices and data streams to restore.

  1. resp = client.snapshot.restore(
  2. repository="my_repository",
  3. snapshot="my_snapshot_2099.05.06",
  4. indices="my-index,logs-my_app-default",
  5. )
  6. print(resp)
  1. response = client.snapshot.restore(
  2. repository: 'my_repository',
  3. snapshot: 'my_snapshot_2099.05.06',
  4. body: {
  5. indices: 'my-index,logs-my_app-default'
  6. }
  7. )
  8. puts response
  1. const response = await client.snapshot.restore({
  2. repository: "my_repository",
  3. snapshot: "my_snapshot_2099.05.06",
  4. indices: "my-index,logs-my_app-default",
  5. });
  6. console.log(response);
  1. POST _snapshot/my_repository/my_snapshot_2099.05.06/_restore
  2. {
  3. "indices": "my-index,logs-my_app-default"
  4. }

Rename on restore

If you want to avoid deleting existing data, you can instead rename the indices and data streams you restore. You typically use this method to compare existing data to historical data from a snapshot. For example, you can use this method to review documents after an accidental update or deletion.

Before you start, ensure the cluster has enough capacity for both the existing and restored data.

The following restore snapshot API request prepends restored- to the name of any restored index or data stream.

  1. resp = client.snapshot.restore(
  2. repository="my_repository",
  3. snapshot="my_snapshot_2099.05.06",
  4. indices="my-index,logs-my_app-default",
  5. rename_pattern="(.+)",
  6. rename_replacement="restored-$1",
  7. )
  8. print(resp)
  1. response = client.snapshot.restore(
  2. repository: 'my_repository',
  3. snapshot: 'my_snapshot_2099.05.06',
  4. body: {
  5. indices: 'my-index,logs-my_app-default',
  6. rename_pattern: '(.+)',
  7. rename_replacement: 'restored-$1'
  8. }
  9. )
  10. puts response
  1. const response = await client.snapshot.restore({
  2. repository: "my_repository",
  3. snapshot: "my_snapshot_2099.05.06",
  4. indices: "my-index,logs-my_app-default",
  5. rename_pattern: "(.+)",
  6. rename_replacement: "restored-$1",
  7. });
  8. console.log(response);
  1. POST _snapshot/my_repository/my_snapshot_2099.05.06/_restore
  2. {
  3. "indices": "my-index,logs-my_app-default",
  4. "rename_pattern": "(.+)",
  5. "rename_replacement": "restored-$1"
  6. }

If the rename options produce two or more indices or data streams with the same name, the restore operation fails.

If you rename a data stream, its backing indices are also renamed. For example, if you rename the logs-my_app-default data stream to restored-logs-my_app-default, the backing index .ds-logs-my_app-default-2099.03.09-000005 is renamed to .ds-restored-logs-my_app-default-2099.03.09-000005.

When the restore operation is complete, you can compare the original and restored data. If you no longer need an original index or data stream, you can delete it and use a reindex to rename the restored one.

  1. resp = client.indices.delete(
  2. index="my-index",
  3. )
  4. print(resp)
  5. resp1 = client.reindex(
  6. source={
  7. "index": "restored-my-index"
  8. },
  9. dest={
  10. "index": "my-index"
  11. },
  12. )
  13. print(resp1)
  14. resp2 = client.indices.delete_data_stream(
  15. name="logs-my_app-default",
  16. )
  17. print(resp2)
  18. resp3 = client.reindex(
  19. source={
  20. "index": "restored-logs-my_app-default"
  21. },
  22. dest={
  23. "index": "logs-my_app-default",
  24. "op_type": "create"
  25. },
  26. )
  27. print(resp3)
  1. response = client.indices.delete(
  2. index: 'my-index'
  3. )
  4. puts response
  5. response = client.reindex(
  6. body: {
  7. source: {
  8. index: 'restored-my-index'
  9. },
  10. dest: {
  11. index: 'my-index'
  12. }
  13. }
  14. )
  15. puts response
  16. response = client.indices.delete_data_stream(
  17. name: 'logs-my_app-default'
  18. )
  19. puts response
  20. response = client.reindex(
  21. body: {
  22. source: {
  23. index: 'restored-logs-my_app-default'
  24. },
  25. dest: {
  26. index: 'logs-my_app-default',
  27. op_type: 'create'
  28. }
  29. }
  30. )
  31. puts response
  1. const response = await client.indices.delete({
  2. index: "my-index",
  3. });
  4. console.log(response);
  5. const response1 = await client.reindex({
  6. source: {
  7. index: "restored-my-index",
  8. },
  9. dest: {
  10. index: "my-index",
  11. },
  12. });
  13. console.log(response1);
  14. const response2 = await client.indices.deleteDataStream({
  15. name: "logs-my_app-default",
  16. });
  17. console.log(response2);
  18. const response3 = await client.reindex({
  19. source: {
  20. index: "restored-logs-my_app-default",
  21. },
  22. dest: {
  23. index: "logs-my_app-default",
  24. op_type: "create",
  25. },
  26. });
  27. console.log(response3);
  1. # Delete the original index
  2. DELETE my-index
  3. # Reindex the restored index to rename it
  4. POST _reindex
  5. {
  6. "source": {
  7. "index": "restored-my-index"
  8. },
  9. "dest": {
  10. "index": "my-index"
  11. }
  12. }
  13. # Delete the original data stream
  14. DELETE _data_stream/logs-my_app-default
  15. # Reindex the restored data stream to rename it
  16. POST _reindex
  17. {
  18. "source": {
  19. "index": "restored-logs-my_app-default"
  20. },
  21. "dest": {
  22. "index": "logs-my_app-default",
  23. "op_type": "create"
  24. }
  25. }

Restore a feature state

You can restore a feature state to recover system indices, system data streams, and other configuration data for a feature from a snapshot.

If you restore a snapshot’s cluster state, the operation restores all feature states in the snapshot by default. Similarly, if you don’t restore a snapshot’s cluster state, the operation doesn’t restore any feature states by default. You can also choose to restore only specific feature states from a snapshot, regardless of the cluster state.

To view a snapshot’s feature states, use the get snapshot API.

  1. resp = client.snapshot.get(
  2. repository="my_repository",
  3. snapshot="my_snapshot_2099.05.06",
  4. )
  5. print(resp)
  1. response = client.snapshot.get(
  2. repository: 'my_repository',
  3. snapshot: 'my_snapshot_2099.05.06'
  4. )
  5. puts response
  1. const response = await client.snapshot.get({
  2. repository: "my_repository",
  3. snapshot: "my_snapshot_2099.05.06",
  4. });
  5. console.log(response);
  1. GET _snapshot/my_repository/my_snapshot_2099.05.06

The response’s feature_states property contains a list of features in the snapshot as well as each feature’s indices.

To restore a specific feature state from the snapshot, specify the feature_name from the response in the restore snapshot API’s feature_states parameter.

When you restore a feature state, Elasticsearch closes and overwrites the feature’s existing indices.

Restoring the security feature state overwrites system indices used for authentication. If you use Elasticsearch Service, ensure you have access to the Elasticsearch Service Console before restoring the security feature state. If you run Elasticsearch on your own hardware, create a superuser in the file realm to ensure you’ll still be able to access your cluster.

  1. resp = client.snapshot.restore(
  2. repository="my_repository",
  3. snapshot="my_snapshot_2099.05.06",
  4. feature_states=[
  5. "geoip"
  6. ],
  7. include_global_state=False,
  8. indices="-*",
  9. )
  10. print(resp)
  1. response = client.snapshot.restore(
  2. repository: 'my_repository',
  3. snapshot: 'my_snapshot_2099.05.06',
  4. body: {
  5. feature_states: [
  6. 'geoip'
  7. ],
  8. include_global_state: false,
  9. indices: '-*'
  10. }
  11. )
  12. puts response
  1. const response = await client.snapshot.restore({
  2. repository: "my_repository",
  3. snapshot: "my_snapshot_2099.05.06",
  4. feature_states: ["geoip"],
  5. include_global_state: false,
  6. indices: "-*",
  7. });
  8. console.log(response);
  1. POST _snapshot/my_repository/my_snapshot_2099.05.06/_restore
  2. {
  3. "feature_states": [ "geoip" ],
  4. "include_global_state": false,
  5. "indices": "-*"
  6. }

Exclude the cluster state from the restore operation.

Exclude the other indices and data streams in the snapshot from the restore operation.

Restore an entire cluster

In some cases, you need to restore an entire cluster from a snapshot, including the cluster state and all feature states. These cases should be rare, such as in the event of a catastrophic failure.

Restoring an entire cluster involves deleting important system indices, including those used for authentication. Consider whether you can restore specific indices or data streams instead.

If you’re restoring to a different cluster, see Restore to a different cluster before you start.

  1. If you backed up the cluster’s configuration files, you can restore them to each node. This step is optional and requires a full cluster restart.

    After you shut down a node, copy the backed-up configuration files over to the node’s $ES_PATH_CONF directory. Before restarting the node, ensure elasticsearch.yml contains the appropriate node roles, node name, and other node-specific settings.

    If you choose to perform this step, you must repeat this process on each node in the cluster.

  2. Temporarily stop indexing and turn off the following features:

    • GeoIP database downloader and ILM history store

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "ingest.geoip.downloader.enabled": False,
      4. "indices.lifecycle.history_index_enabled": False
      5. },
      6. )
      7. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'ingest.geoip.downloader.enabled' => false,
      5. 'indices.lifecycle.history_index_enabled' => false
      6. }
      7. }
      8. )
      9. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "ingest.geoip.downloader.enabled": false,
      4. "indices.lifecycle.history_index_enabled": false,
      5. },
      6. });
      7. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "ingest.geoip.downloader.enabled": false,
      5. "indices.lifecycle.history_index_enabled": false
      6. }
      7. }
    • ILM

      1. resp = client.ilm.stop()
      2. print(resp)
      1. response = client.ilm.stop
      2. puts response
      1. const response = await client.ilm.stop();
      2. console.log(response);
      1. POST _ilm/stop
    • Machine Learning

      1. resp = client.ml.set_upgrade_mode(
      2. enabled=True,
      3. )
      4. print(resp)
      1. response = client.ml.set_upgrade_mode(
      2. enabled: true
      3. )
      4. puts response
      1. const response = await client.ml.setUpgradeMode({
      2. enabled: "true",
      3. });
      4. console.log(response);
      1. POST _ml/set_upgrade_mode?enabled=true
    • Monitoring

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "xpack.monitoring.collection.enabled": False
      4. },
      5. )
      6. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'xpack.monitoring.collection.enabled' => false
      5. }
      6. }
      7. )
      8. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "xpack.monitoring.collection.enabled": false,
      4. },
      5. });
      6. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "xpack.monitoring.collection.enabled": false
      5. }
      6. }
    • Watcher

      1. resp = client.watcher.stop()
      2. print(resp)
      1. response = client.watcher.stop
      2. puts response
      1. const response = await client.watcher.stop();
      2. console.log(response);
      1. POST _watcher/_stop
    • Universal Profiling

      Check if Universal Profiling index template management is enabled:

      1. resp = client.cluster.get_settings(
      2. filter_path="**.xpack.profiling.templates.enabled",
      3. include_defaults=True,
      4. )
      5. print(resp)
      1. response = client.cluster.get_settings(
      2. filter_path: '**.xpack.profiling.templates.enabled',
      3. include_defaults: true
      4. )
      5. puts response
      1. const response = await client.cluster.getSettings({
      2. filter_path: "**.xpack.profiling.templates.enabled",
      3. include_defaults: "true",
      4. });
      5. console.log(response);
      1. GET /_cluster/settings?filter_path=**.xpack.profiling.templates.enabled&include_defaults=true

      If the value is true, disable Universal Profiling index template management:

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "xpack.profiling.templates.enabled": False
      4. },
      5. )
      6. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'xpack.profiling.templates.enabled' => false
      5. }
      6. }
      7. )
      8. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "xpack.profiling.templates.enabled": false,
      4. },
      5. });
      6. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "xpack.profiling.templates.enabled": false
      5. }
      6. }
  3. If you use Elasticsearch security features, log in to a node host, navigate to the Elasticsearch installation directory, and add a user with the superuser role to the file realm using the elasticsearch-users tool.

    For example, the following command creates a user named restore_user.

    1. ./bin/elasticsearch-users useradd restore_user -p my_password -r superuser

    Use this file realm user to authenticate requests until the restore operation is complete.

  4. Use the cluster update settings API to set action.destructive_requires_name to false. This lets you delete data streams and indices using wildcards.

    1. resp = client.cluster.put_settings(
    2. persistent={
    3. "action.destructive_requires_name": False
    4. },
    5. )
    6. print(resp)
    1. response = client.cluster.put_settings(
    2. body: {
    3. persistent: {
    4. 'action.destructive_requires_name' => false
    5. }
    6. }
    7. )
    8. puts response
    1. const response = await client.cluster.putSettings({
    2. persistent: {
    3. "action.destructive_requires_name": false,
    4. },
    5. });
    6. console.log(response);
    1. PUT _cluster/settings
    2. {
    3. "persistent": {
    4. "action.destructive_requires_name": false
    5. }
    6. }
  5. Delete all existing data streams on the cluster.

    1. resp = client.indices.delete_data_stream(
    2. name="*",
    3. expand_wildcards="all",
    4. )
    5. print(resp)
    1. response = client.indices.delete_data_stream(
    2. name: '*',
    3. expand_wildcards: 'all'
    4. )
    5. puts response
    1. const response = await client.indices.deleteDataStream({
    2. name: "*",
    3. expand_wildcards: "all",
    4. });
    5. console.log(response);
    1. DELETE _data_stream/*?expand_wildcards=all
  6. Delete all existing indices on the cluster.

    1. resp = client.indices.delete(
    2. index="*",
    3. expand_wildcards="all",
    4. )
    5. print(resp)
    1. response = client.indices.delete(
    2. index: '*',
    3. expand_wildcards: 'all'
    4. )
    5. puts response
    1. const response = await client.indices.delete({
    2. index: "*",
    3. expand_wildcards: "all",
    4. });
    5. console.log(response);
    1. DELETE *?expand_wildcards=all
  7. Restore the entire snapshot, including the cluster state. By default, restoring the cluster state also restores any feature states in the snapshot.

    1. resp = client.snapshot.restore(
    2. repository="my_repository",
    3. snapshot="my_snapshot_2099.05.06",
    4. indices="*",
    5. include_global_state=True,
    6. )
    7. print(resp)
    1. response = client.snapshot.restore(
    2. repository: 'my_repository',
    3. snapshot: 'my_snapshot_2099.05.06',
    4. body: {
    5. indices: '*',
    6. include_global_state: true
    7. }
    8. )
    9. puts response
    1. const response = await client.snapshot.restore({
    2. repository: "my_repository",
    3. snapshot: "my_snapshot_2099.05.06",
    4. indices: "*",
    5. include_global_state: true,
    6. });
    7. console.log(response);
    1. POST _snapshot/my_repository/my_snapshot_2099.05.06/_restore
    2. {
    3. "indices": "*",
    4. "include_global_state": true
    5. }
  8. When the restore operation is complete, resume indexing and restart any features you stopped:

    When the snapshot is restored, the license that was in use at the time the snapshot was taken will be restored as well. If your license has expired since the snapshot was taken, you will need to use the Update License API to install a current license.

    • GeoIP database downloader and ILM history store

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "ingest.geoip.downloader.enabled": True,
      4. "indices.lifecycle.history_index_enabled": True
      5. },
      6. )
      7. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'ingest.geoip.downloader.enabled' => true,
      5. 'indices.lifecycle.history_index_enabled' => true
      6. }
      7. }
      8. )
      9. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "ingest.geoip.downloader.enabled": true,
      4. "indices.lifecycle.history_index_enabled": true,
      5. },
      6. });
      7. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "ingest.geoip.downloader.enabled": true,
      5. "indices.lifecycle.history_index_enabled": true
      6. }
      7. }
    • ILM

      1. resp = client.ilm.start()
      2. print(resp)
      1. response = client.ilm.start
      2. puts response
      1. const response = await client.ilm.start();
      2. console.log(response);
      1. POST _ilm/start
    • Machine Learning

      1. resp = client.ml.set_upgrade_mode(
      2. enabled=False,
      3. )
      4. print(resp)
      1. response = client.ml.set_upgrade_mode(
      2. enabled: false
      3. )
      4. puts response
      1. const response = await client.ml.setUpgradeMode({
      2. enabled: "false",
      3. });
      4. console.log(response);
      1. POST _ml/set_upgrade_mode?enabled=false
    • Monitoring

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "xpack.monitoring.collection.enabled": True
      4. },
      5. )
      6. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'xpack.monitoring.collection.enabled' => true
      5. }
      6. }
      7. )
      8. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "xpack.monitoring.collection.enabled": true,
      4. },
      5. });
      6. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "xpack.monitoring.collection.enabled": true
      5. }
      6. }
    • Watcher

      1. resp = client.watcher.start()
      2. print(resp)
      1. response = client.watcher.start
      2. puts response
      1. const response = await client.watcher.start();
      2. console.log(response);
      1. POST _watcher/_start
    • Universal Profiling

      If the value was true initially, enable Universal Profiling index template management again, otherwise skip this step:

      1. resp = client.cluster.put_settings(
      2. persistent={
      3. "xpack.profiling.templates.enabled": True
      4. },
      5. )
      6. print(resp)
      1. response = client.cluster.put_settings(
      2. body: {
      3. persistent: {
      4. 'xpack.profiling.templates.enabled' => true
      5. }
      6. }
      7. )
      8. puts response
      1. const response = await client.cluster.putSettings({
      2. persistent: {
      3. "xpack.profiling.templates.enabled": true,
      4. },
      5. });
      6. console.log(response);
      1. PUT _cluster/settings
      2. {
      3. "persistent": {
      4. "xpack.profiling.templates.enabled": true
      5. }
      6. }
  9. If wanted, reset the action.destructive_requires_name cluster setting.

    1. resp = client.cluster.put_settings(
    2. persistent={
    3. "action.destructive_requires_name": None
    4. },
    5. )
    6. print(resp)
    1. response = client.cluster.put_settings(
    2. body: {
    3. persistent: {
    4. 'action.destructive_requires_name' => nil
    5. }
    6. }
    7. )
    8. puts response
    1. const response = await client.cluster.putSettings({
    2. persistent: {
    3. "action.destructive_requires_name": null,
    4. },
    5. });
    6. console.log(response);
    1. PUT _cluster/settings
    2. {
    3. "persistent": {
    4. "action.destructive_requires_name": null
    5. }
    6. }

Monitor a restore

The restore operation uses the shard recovery process to restore an index’s primary shards from a snapshot. While the restore operation recovers primary shards, the cluster will have a yellow health status.

After all primary shards are recovered, the replication process creates and distributes replicas across eligible data nodes. When replication is complete, the cluster health status typically becomes green.

Once you start a restore in Kibana, you’re navigated to the Restore Status page. You can use this page to track the current state for each shard in the snapshot.

You can also monitor snapshot recover using Elasticsearch APIs. To monitor the cluster health status, use the cluster health API.

  1. $response = $client->cluster()->health();
  1. resp = client.cluster.health()
  2. print(resp)
  1. response = client.cluster.health
  2. puts response
  1. res, err := es.Cluster.Health()
  2. fmt.Println(res, err)
  1. const response = await client.cluster.health();
  2. console.log(response);
  1. GET _cluster/health

To get detailed information about ongoing shard recoveries, use the index recovery API.

  1. resp = client.indices.recovery(
  2. index="my-index",
  3. )
  4. print(resp)
  1. response = client.indices.recovery(
  2. index: 'my-index'
  3. )
  4. puts response
  1. const response = await client.indices.recovery({
  2. index: "my-index",
  3. });
  4. console.log(response);
  1. GET my-index/_recovery

To view any unassigned shards, use the cat shards API.

  1. resp = client.cat.shards(
  2. v=True,
  3. h="index,shard,prirep,state,node,unassigned.reason",
  4. s="state",
  5. )
  6. print(resp)
  1. response = client.cat.shards(
  2. v: true,
  3. h: 'index,shard,prirep,state,node,unassigned.reason',
  4. s: 'state'
  5. )
  6. puts response
  1. const response = await client.cat.shards({
  2. v: "true",
  3. h: "index,shard,prirep,state,node,unassigned.reason",
  4. s: "state",
  5. });
  6. console.log(response);
  1. GET _cat/shards?v=true&h=index,shard,prirep,state,node,unassigned.reason&s=state

Unassigned shards have a state of UNASSIGNED. The prirep value is p for primary shards and r for replicas. The unassigned.reason describes why the shard remains unassigned.

To get a more in-depth explanation of an unassigned shard’s allocation status, use the cluster allocation explanation API.

  1. resp = client.cluster.allocation_explain(
  2. index="my-index",
  3. shard=0,
  4. primary=False,
  5. current_node="my-node",
  6. )
  7. print(resp)
  1. response = client.cluster.allocation_explain(
  2. body: {
  3. index: 'my-index',
  4. shard: 0,
  5. primary: false,
  6. current_node: 'my-node'
  7. }
  8. )
  9. puts response
  1. const response = await client.cluster.allocationExplain({
  2. index: "my-index",
  3. shard: 0,
  4. primary: false,
  5. current_node: "my-node",
  6. });
  7. console.log(response);
  1. GET _cluster/allocation/explain
  2. {
  3. "index": "my-index",
  4. "shard": 0,
  5. "primary": false,
  6. "current_node": "my-node"
  7. }

Cancel a restore

You can delete an index or data stream to cancel its ongoing restore. This also deletes any existing data in the cluster for the index or data stream. Deleting an index or data stream doesn’t affect the snapshot or its data.

  1. resp = client.indices.delete(
  2. index="my-index",
  3. )
  4. print(resp)
  5. resp1 = client.indices.delete_data_stream(
  6. name="logs-my_app-default",
  7. )
  8. print(resp1)
  1. response = client.indices.delete(
  2. index: 'my-index'
  3. )
  4. puts response
  5. response = client.indices.delete_data_stream(
  6. name: 'logs-my_app-default'
  7. )
  8. puts response
  1. const response = await client.indices.delete({
  2. index: "my-index",
  3. });
  4. console.log(response);
  5. const response1 = await client.indices.deleteDataStream({
  6. name: "logs-my_app-default",
  7. });
  8. console.log(response1);
  1. # Delete an index
  2. DELETE my-index
  3. # Delete a data stream
  4. DELETE _data_stream/logs-my_app-default

Restore to a different cluster

Elasticsearch Service can help you restore snapshots from other deployments. See Work with snapshots.

Snapshots aren’t tied to a particular cluster or a cluster name. You can create a snapshot in one cluster and restore it in another compatible cluster. Any data stream or index you restore from a snapshot must also be compatible with the current cluster’s version. The topology of the clusters doesn’t need to match.

To restore a snapshot, its repository must be registered and available to the new cluster. If the original cluster still has write access to the repository, register the repository as read-only. This prevents multiple clusters from writing to the repository at the same time and corrupting the repository’s contents. It also prevents Elasticsearch from caching the repository’s contents, which means that changes made by other clusters will become visible straight away.

Before you start a restore operation, ensure the new cluster has enough capacity for any data streams or indices you want to restore. If the new cluster has a smaller capacity, you can:

  • Add nodes or upgrade your hardware to increase capacity.
  • Restore fewer indices and data streams.
  • Reduce the number of replicas for restored indices.

    For example, the following restore snapshot API request uses the index_settings option to set index.number_of_replicas to 1.

    1. resp = client.snapshot.restore(
    2. repository="my_repository",
    3. snapshot="my_snapshot_2099.05.06",
    4. indices="my-index,logs-my_app-default",
    5. index_settings={
    6. "index.number_of_replicas": 1
    7. },
    8. )
    9. print(resp)
    1. response = client.snapshot.restore(
    2. repository: 'my_repository',
    3. snapshot: 'my_snapshot_2099.05.06',
    4. body: {
    5. indices: 'my-index,logs-my_app-default',
    6. index_settings: {
    7. 'index.number_of_replicas' => 1
    8. }
    9. }
    10. )
    11. puts response
    1. const response = await client.snapshot.restore({
    2. repository: "my_repository",
    3. snapshot: "my_snapshot_2099.05.06",
    4. indices: "my-index,logs-my_app-default",
    5. index_settings: {
    6. "index.number_of_replicas": 1,
    7. },
    8. });
    9. console.log(response);
    1. POST _snapshot/my_repository/my_snapshot_2099.05.06/_restore
    2. {
    3. "indices": "my-index,logs-my_app-default",
    4. "index_settings": {
    5. "index.number_of_replicas": 1
    6. }
    7. }

If indices or backing indices in the original cluster were assigned to particular nodes using shard allocation filtering, the same rules will be enforced in the new cluster. If the new cluster does not contain nodes with appropriate attributes that a restored index can be allocated on, the index will not be successfully restored unless these index allocation settings are changed during the restore operation.

The restore operation also checks that restored persistent settings are compatible with the current cluster to avoid accidentally restoring incompatible settings. If you need to restore a snapshot with incompatible persistent settings, try restoring it without the global cluster state.

Troubleshoot restore errors

Here’s how to resolve common errors returned by restore requests.

Cannot restore index [<index>] because an open index with same name already exists in the cluster

You can’t restore an open index that already exists. To resolve this error, try one of the methods in Restore an index or data stream.

Cannot restore index [<index>] with [x] shards from a snapshot of index [<snapshot-index>] with [y] shards

You can only restore an existing index if it’s closed and the index in the snapshot has the same number of primary shards. This error indicates the index in the snapshot has a different number of primary shards.

To resolve this error, try one of the methods in Restore an index or data stream.