Tutorial: Data stream retention

Tutorial: Data stream retention

In this tutorial, we are going to go over the data stream lifecycle retention; we will define it, go over how it can be configured and how it can gets applied. Keep in mind, the following options apply only to data streams that are managed by the data stream lifecycle.

  1. What is data stream retention?
  2. How to configure retention?
  3. How is the effective retention calculated?
  4. How is the effective retention applied?

You can verify if a data steam is managed by the data stream lifecycle via the get data stream lifecycle API:

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

The result should look like this:

  1. {
  2. "data_streams": [
  3. {
  4. "name": "my-data-stream",
  5. "lifecycle": {
  6. "enabled": true
  7. }
  8. }
  9. ]
  10. }

The name of your data stream.

Ensure that the lifecycle is enabled, meaning this should be true.

What is data stream retention?

We define retention as the least amount of time the data of a data stream are going to be kept in Elasticsearch. After this time period has passed, Elasticsearch is allowed to remove these data to free up space and/or manage costs.

Retention does not define the period that the data will be removed, but the minimum time period they will be kept.

We define 4 different types of retention:

  • The data stream retention, or data_retention, which is the retention configured on the data stream level. It can be set via an index template for future data streams or via the PUT data stream lifecycle API for an existing data stream. When the data stream retention is not set, it implies that the data need to be kept forever.
  • The global default retention, let’s call it default_retention, which is a retention configured via the cluster setting data_streams.lifecycle.retention.default and will be applied to all data streams managed by data stream lifecycle that do not have data_retention configured. Effectively, it ensures that there will be no data streams keeping their data forever. This can be set via the update cluster settings API.
  • The global max retention, let’s call it max_retention, which is a retention configured via the cluster setting data_streams.lifecycle.retention.max and will be applied to all data streams managed by data stream lifecycle. Effectively, it ensures that there will be no data streams whose retention will exceed this time period. This can be set via the update cluster settings API.
  • The effective retention, or effective_retention, which is the retention applied at a data stream on a given moment. Effective retention cannot be set, it is derived by taking into account all the configured retention listed above and is calculated as it is described here.

Global default and max retention do not apply to data streams internal to elastic. Internal data streams are recognised either by having the system flag set to true or if their name is prefixed with a dot (.).

How to configure retention?

  • By setting the data_retention on the data stream level. This retention can be configured in two ways:

     — For new data streams, it can be defined in the index template that would be applied during the data stream’s creation. You can use the create index template API, for example:

    1. resp = client.indices.put_index_template(
    2. name="template",
    3. index_patterns=[
    4. "my-data-stream*"
    5. ],
    6. data_stream={},
    7. priority=500,
    8. template={
    9. "lifecycle": {
    10. "data_retention": "7d"
    11. }
    12. },
    13. meta={
    14. "description": "Template with data stream lifecycle"
    15. },
    16. )
    17. print(resp)
    1. const response = await client.indices.putIndexTemplate({
    2. name: "template",
    3. index_patterns: ["my-data-stream*"],
    4. data_stream: {},
    5. priority: 500,
    6. template: {
    7. lifecycle: {
    8. data_retention: "7d",
    9. },
    10. },
    11. _meta: {
    12. description: "Template with data stream lifecycle",
    13. },
    14. });
    15. console.log(response);
    1. PUT _index_template/template
    2. {
    3. "index_patterns": ["my-data-stream*"],
    4. "data_stream": { },
    5. "priority": 500,
    6. "template": {
    7. "lifecycle": {
    8. "data_retention": "7d"
    9. }
    10. },
    11. "_meta": {
    12. "description": "Template with data stream lifecycle"
    13. }
    14. }

     — For an existing data stream, it can be set via the PUT lifecycle API.

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

    The retention period of this data stream is set to 30 days.

  • By setting the global retention via the data_streams.lifecycle.retention.default and/or data_streams.lifecycle.retention.max that are set on a cluster level. You can be set via the update cluster settings API. For example:

    1. resp = client.cluster.put_settings(
    2. persistent={
    3. "data_streams.lifecycle.retention.default": "7d",
    4. "data_streams.lifecycle.retention.max": "90d"
    5. },
    6. )
    7. print(resp)
    1. const response = await client.cluster.putSettings({
    2. persistent: {
    3. "data_streams.lifecycle.retention.default": "7d",
    4. "data_streams.lifecycle.retention.max": "90d",
    5. },
    6. });
    7. console.log(response);
    1. PUT /_cluster/settings
    2. {
    3. "persistent" : {
    4. "data_streams.lifecycle.retention.default" : "7d",
    5. "data_streams.lifecycle.retention.max" : "90d"
    6. }
    7. }

How is the effective retention calculated?

The effective is calculated in the following way:

  • The effective_retention is the default_retention, when default_retention is defined and the data stream does not have data_retention.
  • The effective_retention is the data_retention, when data_retention is defined and if max_retention is defined, it is less than the max_retention.
  • The effective_retention is the max_retention, when max_retention is defined, and the data stream has either no data_retention or its data_retention is greater than the max_retention.

The above is demonstrated in the examples below:

default_retentionmax_retentiondata_retentioneffective_retentionRetention determined by

Not set

Not set

Not set

Infinite

N/A

Not relevant

12 months

30 days

30 days

data_retention

Not relevant

Not set

30 days

30 days

data_retention

30 days

12 months

Not set

30 days

default_retention

30 days

30 days

Not set

30 days

default_retention

Not relevant

30 days

12 months

30 days

max_retention

Not set

30 days

Not set

30 days

max_retention

Considering our example, if we retrieve the lifecycle of my-data-stream:

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

We see that it will remain the same with what the user configured:

  1. {
  2. "global_retention" : {
  3. "max_retention" : "90d",
  4. "default_retention" : "7d"
  5. },
  6. "data_streams": [
  7. {
  8. "name": "my-data-stream",
  9. "lifecycle": {
  10. "enabled": true,
  11. "data_retention": "30d",
  12. "effective_retention": "30d",
  13. "retention_determined_by": "data_stream_configuration"
  14. }
  15. }
  16. ]
  17. }

The maximum retention configured in the cluster.

The default retention configured in the cluster.

The requested retention for this data stream.

The retention that is applied by the data stream lifecycle on this data stream.

The configuration that determined the effective retention. In this case it’s the data_configuration because it is less than the max_retention.

How is the effective retention applied?

Retention is applied to the remaining backing indices of a data stream as the last step of a data stream lifecycle run. Data stream lifecycle will retrieve the backing indices whose generation_time is longer than the effective retention period and delete them. The generation_time is only applicable to rolled over backing indices and it is either the time since the backing index got rolled over, or the time optionally configured in the index.lifecycle.origination_date setting.

We use the generation_time instead of the creation time because this ensures that all data in the backing index have passed the retention period. As a result, the retention period is not the exact time data get deleted, but the minimum time data will be stored.