Tutorial: Migrate ILM managed data stream to data stream lifecycle

Tutorial: Migrate ILM managed data stream to data stream lifecycle

In this tutorial we’ll look at migrating an existing data stream from Index Lifecycle Management (ILM) to data stream lifecycle. The existing ILM managed backing indices will continue to be managed by ILM until they age out and get deleted by ILM; however, the new backing indices will be managed by data stream lifecycle. This way, a data stream is gradually migrated away from being managed by ILM to being managed by data stream lifecycle. As we’ll see, ILM and data stream lifecycle can co-manage a data stream; however, an index can only be managed by one system at a time.

TL;DR

To migrate a data stream from ILM to data stream lifecycle we’ll have to execute two steps:

  1. Update the index template that’s backing the data stream to set prefer_ilm to false, and to configure data stream lifecycle.
  2. Configure the data stream lifecycle for the existing data stream using the lifecycle API.

For more details see the migrate to data stream lifecycle section.

Setup ILM managed data stream

Let’s first create a data stream with two backing indices managed by ILM. We first create an ILM policy:

  1. resp = client.ilm.put_lifecycle(
  2. name="pre-dsl-ilm-policy",
  3. policy={
  4. "phases": {
  5. "hot": {
  6. "actions": {
  7. "rollover": {
  8. "max_primary_shard_size": "50gb"
  9. }
  10. }
  11. },
  12. "delete": {
  13. "min_age": "7d",
  14. "actions": {
  15. "delete": {}
  16. }
  17. }
  18. }
  19. },
  20. )
  21. print(resp)
  1. response = client.ilm.put_lifecycle(
  2. policy: 'pre-dsl-ilm-policy',
  3. body: {
  4. policy: {
  5. phases: {
  6. hot: {
  7. actions: {
  8. rollover: {
  9. max_primary_shard_size: '50gb'
  10. }
  11. }
  12. },
  13. delete: {
  14. min_age: '7d',
  15. actions: {
  16. delete: {}
  17. }
  18. }
  19. }
  20. }
  21. }
  22. )
  23. puts response
  1. const response = await client.ilm.putLifecycle({
  2. name: "pre-dsl-ilm-policy",
  3. policy: {
  4. phases: {
  5. hot: {
  6. actions: {
  7. rollover: {
  8. max_primary_shard_size: "50gb",
  9. },
  10. },
  11. },
  12. delete: {
  13. min_age: "7d",
  14. actions: {
  15. delete: {},
  16. },
  17. },
  18. },
  19. },
  20. });
  21. console.log(response);
  1. PUT _ilm/policy/pre-dsl-ilm-policy
  2. {
  3. "policy": {
  4. "phases": {
  5. "hot": {
  6. "actions": {
  7. "rollover": {
  8. "max_primary_shard_size": "50gb"
  9. }
  10. }
  11. },
  12. "delete": {
  13. "min_age": "7d",
  14. "actions": {
  15. "delete": {}
  16. }
  17. }
  18. }
  19. }
  20. }

And let’s create an index template that’ll back the data stream and configures ILM:

  1. resp = client.indices.put_index_template(
  2. name="dsl-data-stream-template",
  3. index_patterns=[
  4. "dsl-data-stream*"
  5. ],
  6. data_stream={},
  7. priority=500,
  8. template={
  9. "settings": {
  10. "index.lifecycle.name": "pre-dsl-ilm-policy"
  11. }
  12. },
  13. )
  14. print(resp)
  1. response = client.indices.put_index_template(
  2. name: 'dsl-data-stream-template',
  3. body: {
  4. index_patterns: [
  5. 'dsl-data-stream*'
  6. ],
  7. data_stream: {},
  8. priority: 500,
  9. template: {
  10. settings: {
  11. 'index.lifecycle.name' => 'pre-dsl-ilm-policy'
  12. }
  13. }
  14. }
  15. )
  16. puts response
  1. const response = await client.indices.putIndexTemplate({
  2. name: "dsl-data-stream-template",
  3. index_patterns: ["dsl-data-stream*"],
  4. data_stream: {},
  5. priority: 500,
  6. template: {
  7. settings: {
  8. "index.lifecycle.name": "pre-dsl-ilm-policy",
  9. },
  10. },
  11. });
  12. console.log(response);
  1. PUT _index_template/dsl-data-stream-template
  2. {
  3. "index_patterns": ["dsl-data-stream*"],
  4. "data_stream": { },
  5. "priority": 500,
  6. "template": {
  7. "settings": {
  8. "index.lifecycle.name": "pre-dsl-ilm-policy"
  9. }
  10. }
  11. }

We’ll now index a document targetting dsl-data-stream to create the data stream and we’ll also manually rollover the data stream to have another generation index created:

  1. resp = client.index(
  2. index="dsl-data-stream",
  3. document={
  4. "@timestamp": "2023-10-18T16:21:15.000Z",
  5. "message": "192.0.2.42 - - [06/May/2099:16:21:15 +0000] \"GET /images/bg.jpg HTTP/1.0\" 200 24736"
  6. },
  7. )
  8. print(resp)
  1. response = client.index(
  2. index: 'dsl-data-stream',
  3. body: {
  4. "@timestamp": '2023-10-18T16:21:15.000Z',
  5. message: '192.0.2.42 - - [06/May/2099:16:21:15 +0000] "GET /images/bg.jpg HTTP/1.0" 200 24736'
  6. }
  7. )
  8. puts response
  1. const response = await client.index({
  2. index: "dsl-data-stream",
  3. document: {
  4. "@timestamp": "2023-10-18T16:21:15.000Z",
  5. message:
  6. '192.0.2.42 - - [06/May/2099:16:21:15 +0000] "GET /images/bg.jpg HTTP/1.0" 200 24736',
  7. },
  8. });
  9. console.log(response);
  1. POST dsl-data-stream/_doc?
  2. {
  3. "@timestamp": "2023-10-18T16:21:15.000Z",
  4. "message": "192.0.2.42 - - [06/May/2099:16:21:15 +0000] \"GET /images/bg.jpg HTTP/1.0\" 200 24736"
  5. }
  1. resp = client.indices.rollover(
  2. alias="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.rollover(
  2. alias: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.rollover({
  2. alias: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. POST dsl-data-stream/_rollover

We’ll use the GET _data_stream API to inspect the state of the data stream:

  1. resp = client.indices.get_data_stream(
  2. name="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.get_data_stream(
  2. name: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.getDataStream({
  2. name: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. GET _data_stream/dsl-data-stream

Inspecting the response we’ll see that both backing indices are managed by ILM and that the next generation index will also be managed by ILM:

  1. {
  2. "data_streams": [
  3. {
  4. "name": "dsl-data-stream",
  5. "timestamp_field": {
  6. "name": "@timestamp"
  7. },
  8. "indices": [
  9. {
  10. "index_name": ".ds-dsl-data-stream-2023.10.19-000001",
  11. "index_uuid": "xCEhwsp8Tey0-FLNFYVwSg",
  12. "prefer_ilm": true,
  13. "ilm_policy": "pre-dsl-ilm-policy",
  14. "managed_by": "Index Lifecycle Management"
  15. },
  16. {
  17. "index_name": ".ds-dsl-data-stream-2023.10.19-000002",
  18. "index_uuid": "PA_JquKGSiKcAKBA8DJ5gw",
  19. "prefer_ilm": true,
  20. "ilm_policy": "pre-dsl-ilm-policy",
  21. "managed_by": "Index Lifecycle Management"
  22. }
  23. ],
  24. "generation": 2,
  25. "status": "GREEN",
  26. "template": "dsl-data-stream-template",
  27. "next_generation_managed_by": "Index Lifecycle Management",
  28. "prefer_ilm": true,
  29. "ilm_policy": "pre-dsl-ilm-policy",
  30. "hidden": false,
  31. "system": false,
  32. "allow_custom_routing": false,
  33. "replicated": false,
  34. "rollover_on_write": false
  35. }
  36. ]
  37. }

The name of the backing index.

For each backing index we display the value of the prefer_ilm configuration which will indicate if ILM takes precedence over data stream lifecycle in case both systems are configured for an index.

The ILM policy configured for this index.

The system that manages this index (possible values are “Index Lifecycle Management”, “Data stream lifecycle”, or “Unmanaged”)

The system that will manage the next generation index (the new write index of this data stream, once the data stream is rolled over). The possible values are “Index Lifecycle Management”, “Data stream lifecycle”, or “Unmanaged”.

The prefer_ilm value configured in the index template that’s backing the data stream. This value will be configured for all the new backing indices. If it’s not configured in the index template the backing indices will receive the true default value (ILM takes precedence over data stream lifecycle by default as it’s currently richer in features).

The ILM policy configured in the index template that’s backing this data stream (which will be configured on all the new backing indices, as long as it exists in the index template).

Migrate data stream to data stream lifecycle

To migrate the dsl-data-stream to data stream lifecycle we’ll have to execute two steps:

  1. Update the index template that’s backing the data stream to set prefer_ilm to false, and to configure data stream lifecycle.
  2. Configure the data stream lifecycle for the existing dsl-data-stream using the lifecycle API.

The data stream lifecycle configuration that’s added to the index template, being a data stream configuration, will only apply to new data streams. Our data stream exists already, so even though we added a data stream lifecycle configuration in the index template it will not be applied to dsl-data-stream.

Let’s update the index template:

  1. resp = client.indices.put_index_template(
  2. name="dsl-data-stream-template",
  3. index_patterns=[
  4. "dsl-data-stream*"
  5. ],
  6. data_stream={},
  7. priority=500,
  8. template={
  9. "settings": {
  10. "index.lifecycle.name": "pre-dsl-ilm-policy",
  11. "index.lifecycle.prefer_ilm": False
  12. },
  13. "lifecycle": {
  14. "data_retention": "7d"
  15. }
  16. },
  17. )
  18. print(resp)
  1. response = client.indices.put_index_template(
  2. name: 'dsl-data-stream-template',
  3. body: {
  4. index_patterns: [
  5. 'dsl-data-stream*'
  6. ],
  7. data_stream: {},
  8. priority: 500,
  9. template: {
  10. settings: {
  11. 'index.lifecycle.name' => 'pre-dsl-ilm-policy',
  12. 'index.lifecycle.prefer_ilm' => false
  13. },
  14. lifecycle: {
  15. data_retention: '7d'
  16. }
  17. }
  18. }
  19. )
  20. puts response
  1. const response = await client.indices.putIndexTemplate({
  2. name: "dsl-data-stream-template",
  3. index_patterns: ["dsl-data-stream*"],
  4. data_stream: {},
  5. priority: 500,
  6. template: {
  7. settings: {
  8. "index.lifecycle.name": "pre-dsl-ilm-policy",
  9. "index.lifecycle.prefer_ilm": false,
  10. },
  11. lifecycle: {
  12. data_retention: "7d",
  13. },
  14. },
  15. });
  16. console.log(response);
  1. PUT _index_template/dsl-data-stream-template
  2. {
  3. "index_patterns": ["dsl-data-stream*"],
  4. "data_stream": { },
  5. "priority": 500,
  6. "template": {
  7. "settings": {
  8. "index.lifecycle.name": "pre-dsl-ilm-policy",
  9. "index.lifecycle.prefer_ilm": false
  10. },
  11. "lifecycle": {
  12. "data_retention": "7d"
  13. }
  14. }
  15. }

The prefer_ilm setting will now be configured on the new backing indices (created by rolling over the data stream) such that ILM does not take precedence over data stream lifecycle.

We’re configuring the data stream lifecycle so new data streams will be managed by data stream lifecycle.

We’ve now made sure that new data streams will be managed by data stream lifecycle.

Let’s update our existing dsl-data-stream and configure data stream lifecycle:

  1. resp = client.indices.put_data_lifecycle(
  2. name="dsl-data-stream",
  3. data_retention="7d",
  4. )
  5. print(resp)
  1. response = client.indices.put_data_lifecycle(
  2. name: 'dsl-data-stream',
  3. body: {
  4. data_retention: '7d'
  5. }
  6. )
  7. puts response
  1. const response = await client.indices.putDataLifecycle({
  2. name: "dsl-data-stream",
  3. data_retention: "7d",
  4. });
  5. console.log(response);
  1. PUT _data_stream/dsl-data-stream/_lifecycle
  2. {
  3. "data_retention": "7d"
  4. }

We can inspect the data stream to check that the next generation will indeed be managed by data stream lifecycle:

  1. resp = client.indices.get_data_stream(
  2. name="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.get_data_stream(
  2. name: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.getDataStream({
  2. name: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. GET _data_stream/dsl-data-stream
  1. {
  2. "data_streams": [
  3. {
  4. "name": "dsl-data-stream",
  5. "timestamp_field": {
  6. "name": "@timestamp"
  7. },
  8. "indices": [
  9. {
  10. "index_name": ".ds-dsl-data-stream-2023.10.19-000001",
  11. "index_uuid": "xCEhwsp8Tey0-FLNFYVwSg",
  12. "prefer_ilm": true,
  13. "ilm_policy": "pre-dsl-ilm-policy",
  14. "managed_by": "Index Lifecycle Management"
  15. },
  16. {
  17. "index_name": ".ds-dsl-data-stream-2023.10.19-000002",
  18. "index_uuid": "PA_JquKGSiKcAKBA8DJ5gw",
  19. "prefer_ilm": true,
  20. "ilm_policy": "pre-dsl-ilm-policy",
  21. "managed_by": "Index Lifecycle Management"
  22. }
  23. ],
  24. "generation": 2,
  25. "status": "GREEN",
  26. "template": "dsl-data-stream-template",
  27. "lifecycle": {
  28. "enabled": true,
  29. "data_retention": "7d",
  30. "effective_retention": "7d",
  31. "retention_determined_by": "data_stream_configuration"
  32. },
  33. "ilm_policy": "pre-dsl-ilm-policy",
  34. "next_generation_managed_by": "Data stream lifecycle",
  35. "prefer_ilm": false,
  36. "hidden": false,
  37. "system": false,
  38. "allow_custom_routing": false,
  39. "replicated": false,
  40. "rollover_on_write": false
  41. }
  42. ]
  43. }

The existing backing index will continue to be managed by ILM

The existing backing index will continue to be managed by ILM

The next generation index will be managed by Data stream lifecycle

The prefer_ilm setting value we configured in the index template is reflected and will be configured accordingly for new backing indices.

We’ll now rollover the data stream to see the new generation index being managed by data stream lifecycle:

  1. resp = client.indices.rollover(
  2. alias="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.rollover(
  2. alias: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.rollover({
  2. alias: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. POST dsl-data-stream/_rollover
  1. resp = client.indices.get_data_stream(
  2. name="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.get_data_stream(
  2. name: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.getDataStream({
  2. name: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. GET _data_stream/dsl-data-stream
  1. {
  2. "data_streams": [
  3. {
  4. "name": "dsl-data-stream",
  5. "timestamp_field": {
  6. "name": "@timestamp"
  7. },
  8. "indices": [
  9. {
  10. "index_name": ".ds-dsl-data-stream-2023.10.19-000001",
  11. "index_uuid": "xCEhwsp8Tey0-FLNFYVwSg",
  12. "prefer_ilm": true,
  13. "ilm_policy": "pre-dsl-ilm-policy",
  14. "managed_by": "Index Lifecycle Management"
  15. },
  16. {
  17. "index_name": ".ds-dsl-data-stream-2023.10.19-000002",
  18. "index_uuid": "PA_JquKGSiKcAKBA8DJ5gw",
  19. "prefer_ilm": true,
  20. "ilm_policy": "pre-dsl-ilm-policy",
  21. "managed_by": "Index Lifecycle Management"
  22. },
  23. {
  24. "index_name": ".ds-dsl-data-stream-2023.10.19-000003",
  25. "index_uuid": "PA_JquKGSiKcAKBA8abcd1",
  26. "prefer_ilm": false,
  27. "ilm_policy": "pre-dsl-ilm-policy",
  28. "managed_by": "Data stream lifecycle"
  29. }
  30. ],
  31. "generation": 3,
  32. "status": "GREEN",
  33. "template": "dsl-data-stream-template",
  34. "lifecycle": {
  35. "enabled": true,
  36. "data_retention": "7d",
  37. "effective_retention": "7d",
  38. "retention_determined_by": "data_stream_configuration"
  39. },
  40. "ilm_policy": "pre-dsl-ilm-policy",
  41. "next_generation_managed_by": "Data stream lifecycle",
  42. "prefer_ilm": false,
  43. "hidden": false,
  44. "system": false,
  45. "allow_custom_routing": false,
  46. "replicated": false,
  47. "rollover_on_write": false
  48. }
  49. ]
  50. }

The backing indices that existed before rollover will continue to be managed by ILM

The backing indices that existed before rollover will continue to be managed by ILM

The new write index received the false value for the prefer_ilm setting, as we configured in the index template

The new write index is managed by Data stream lifecycle

Migrate data stream back to ILM

We can easily change this data stream to be managed by ILM because we didn’t remove the ILM policy when we updated the index template.

We can achieve this in two ways:

  1. Delete the lifecycle from the data streams
  2. Disable data stream lifecycle by configuring the enabled flag to false.

Let’s implement option 2 and disable the data stream lifecycle:

  1. resp = client.indices.put_data_lifecycle(
  2. name="dsl-data-stream",
  3. data_retention="7d",
  4. enabled=False,
  5. )
  6. print(resp)
  1. response = client.indices.put_data_lifecycle(
  2. name: 'dsl-data-stream',
  3. body: {
  4. data_retention: '7d',
  5. enabled: false
  6. }
  7. )
  8. puts response
  1. const response = await client.indices.putDataLifecycle({
  2. name: "dsl-data-stream",
  3. data_retention: "7d",
  4. enabled: false,
  5. });
  6. console.log(response);
  1. PUT _data_stream/dsl-data-stream/_lifecycle
  2. {
  3. "data_retention": "7d",
  4. "enabled": false
  5. }

The enabled flag can be ommitted and defaults to true however, here we explicitly configure it to false Let’s check the state of the data stream:

  1. resp = client.indices.get_data_stream(
  2. name="dsl-data-stream",
  3. )
  4. print(resp)
  1. response = client.indices.get_data_stream(
  2. name: 'dsl-data-stream'
  3. )
  4. puts response
  1. const response = await client.indices.getDataStream({
  2. name: "dsl-data-stream",
  3. });
  4. console.log(response);
  1. GET _data_stream/dsl-data-stream
  1. {
  2. "data_streams": [
  3. {
  4. "name": "dsl-data-stream",
  5. "timestamp_field": {
  6. "name": "@timestamp"
  7. },
  8. "indices": [
  9. {
  10. "index_name": ".ds-dsl-data-stream-2023.10.19-000001",
  11. "index_uuid": "xCEhwsp8Tey0-FLNFYVwSg",
  12. "prefer_ilm": true,
  13. "ilm_policy": "pre-dsl-ilm-policy",
  14. "managed_by": "Index Lifecycle Management"
  15. },
  16. {
  17. "index_name": ".ds-dsl-data-stream-2023.10.19-000002",
  18. "index_uuid": "PA_JquKGSiKcAKBA8DJ5gw",
  19. "prefer_ilm": true,
  20. "ilm_policy": "pre-dsl-ilm-policy",
  21. "managed_by": "Index Lifecycle Management"
  22. },
  23. {
  24. "index_name": ".ds-dsl-data-stream-2023.10.19-000003",
  25. "index_uuid": "PA_JquKGSiKcAKBA8abcd1",
  26. "prefer_ilm": false,
  27. "ilm_policy": "pre-dsl-ilm-policy",
  28. "managed_by": "Index Lifecycle Management"
  29. }
  30. ],
  31. "generation": 3,
  32. "status": "GREEN",
  33. "template": "dsl-data-stream-template",
  34. "lifecycle": {
  35. "enabled": false,
  36. "data_retention": "7d"
  37. },
  38. "ilm_policy": "pre-dsl-ilm-policy",
  39. "next_generation_managed_by": "Index Lifecycle Management",
  40. "prefer_ilm": false,
  41. "hidden": false,
  42. "system": false,
  43. "allow_custom_routing": false,
  44. "replicated": false,
  45. "rollover_on_write": false
  46. }
  47. ]
  48. }

The write index is now managed by ILM

The lifecycle configured on the data stream is now disabled.

The next write index will be managed by ILM

Had we removed the ILM policy from the index template when we updated it, the write index of the data stream will now be Unmanaged because the index wouldn’t have the ILM policy configured to fallback onto.