Slow Log

Slow Log

Search Slow Log

Shard level slow search log allows to log slow search (query and fetch phases) into a dedicated log file.

Thresholds can be set for both the query phase of the execution, and fetch phase, here is a sample:

  1. index.search.slowlog.threshold.query.warn: 10s
  2. index.search.slowlog.threshold.query.info: 5s
  3. index.search.slowlog.threshold.query.debug: 2s
  4. index.search.slowlog.threshold.query.trace: 500ms
  5. index.search.slowlog.threshold.fetch.warn: 1s
  6. index.search.slowlog.threshold.fetch.info: 800ms
  7. index.search.slowlog.threshold.fetch.debug: 500ms
  8. index.search.slowlog.threshold.fetch.trace: 200ms

All of the above settings are dynamic and can be set for each index using the update indices settings API. For example:

  1. resp = client.indices.put_settings(
  2. index="my-index-000001",
  3. settings={
  4. "index.search.slowlog.threshold.query.warn": "10s",
  5. "index.search.slowlog.threshold.query.info": "5s",
  6. "index.search.slowlog.threshold.query.debug": "2s",
  7. "index.search.slowlog.threshold.query.trace": "500ms",
  8. "index.search.slowlog.threshold.fetch.warn": "1s",
  9. "index.search.slowlog.threshold.fetch.info": "800ms",
  10. "index.search.slowlog.threshold.fetch.debug": "500ms",
  11. "index.search.slowlog.threshold.fetch.trace": "200ms"
  12. },
  13. )
  14. print(resp)
  1. response = client.indices.put_settings(
  2. index: 'my-index-000001',
  3. body: {
  4. 'index.search.slowlog.threshold.query.warn' => '10s',
  5. 'index.search.slowlog.threshold.query.info' => '5s',
  6. 'index.search.slowlog.threshold.query.debug' => '2s',
  7. 'index.search.slowlog.threshold.query.trace' => '500ms',
  8. 'index.search.slowlog.threshold.fetch.warn' => '1s',
  9. 'index.search.slowlog.threshold.fetch.info' => '800ms',
  10. 'index.search.slowlog.threshold.fetch.debug' => '500ms',
  11. 'index.search.slowlog.threshold.fetch.trace' => '200ms'
  12. }
  13. )
  14. puts response
  1. PUT /my-index-000001/_settings
  2. {
  3. "index.search.slowlog.threshold.query.warn": "10s",
  4. "index.search.slowlog.threshold.query.info": "5s",
  5. "index.search.slowlog.threshold.query.debug": "2s",
  6. "index.search.slowlog.threshold.query.trace": "500ms",
  7. "index.search.slowlog.threshold.fetch.warn": "1s",
  8. "index.search.slowlog.threshold.fetch.info": "800ms",
  9. "index.search.slowlog.threshold.fetch.debug": "500ms",
  10. "index.search.slowlog.threshold.fetch.trace": "200ms"
  11. }

By default thresholds are disabled (set to -1).

The logging is done on the shard level scope, meaning the execution of a search request within a specific shard. It does not encompass the whole search request, which can be broadcast to several shards in order to execute. Some of the benefits of shard level logging is the association of the actual execution on the specific machine, compared with request level.

The search slow log file is configured in the log4j2.properties file.

Identifying search slow log origin

It is often useful to identify what triggered a slow running query. To include information about the user that triggered a slow search, use the index.search.slowlog.include.user setting.

  1. resp = client.indices.put_settings(
  2. index="my-index-000001",
  3. settings={
  4. "index.search.slowlog.include.user": True
  5. },
  6. )
  7. print(resp)
  1. response = client.indices.put_settings(
  2. index: 'my-index-000001',
  3. body: {
  4. 'index.search.slowlog.include.user' => true
  5. }
  6. )
  7. puts response
  1. PUT /my-index-000001/_settings
  2. {
  3. "index.search.slowlog.include.user": true
  4. }

This will result in user information being included in the slow log.

  1. {
  2. "@timestamp": "2024-02-21T12:42:37.255Z",
  3. "log.level": "WARN",
  4. "auth.type": "REALM",
  5. "elasticsearch.slowlog.id": "tomcat-123",
  6. "elasticsearch.slowlog.message": "[index6][0]",
  7. "elasticsearch.slowlog.search_type": "QUERY_THEN_FETCH",
  8. "elasticsearch.slowlog.source": "{\"query\":{\"match_all\":{\"boost\":1.0}}}",
  9. "elasticsearch.slowlog.stats": "[]",
  10. "elasticsearch.slowlog.took": "747.3micros",
  11. "elasticsearch.slowlog.took_millis": 0,
  12. "elasticsearch.slowlog.total_hits": "1 hits",
  13. "elasticsearch.slowlog.total_shards": 1,
  14. "user.name": "elastic",
  15. "user.realm": "reserved",
  16. "ecs.version": "1.2.0",
  17. "service.name": "ES_ECS",
  18. "event.dataset": "elasticsearch.index_search_slowlog",
  19. "process.thread.name": "elasticsearch[runTask-0][search][T#5]",
  20. "log.logger": "index.search.slowlog.query",
  21. "elasticsearch.cluster.uuid": "Ui23kfF1SHKJwu_hI1iPPQ",
  22. "elasticsearch.node.id": "JK-jn-XpQ3OsDUsq5ZtfGg",
  23. "elasticsearch.node.name": "node-0",
  24. "elasticsearch.cluster.name": "distribution_run"
  25. }

If a call was initiated with an X-Opaque-ID header, then the ID is included in Search Slow logs in the elasticsearch.slowlog.id field. See X-Opaque-Id HTTP header for details and best practices.

Index Slow log

The indexing slow log, similar in functionality to the search slow log. The log file name ends with _index_indexing_slowlog.json. Log and the thresholds are configured in the same way as the search slowlog. Index slowlog sample:

  1. index.indexing.slowlog.threshold.index.warn: 10s
  2. index.indexing.slowlog.threshold.index.info: 5s
  3. index.indexing.slowlog.threshold.index.debug: 2s
  4. index.indexing.slowlog.threshold.index.trace: 500ms
  5. index.indexing.slowlog.source: 1000

All of the above settings are dynamic and can be set for each index using the update indices settings API. For example:

  1. resp = client.indices.put_settings(
  2. index="my-index-000001",
  3. settings={
  4. "index.indexing.slowlog.threshold.index.warn": "10s",
  5. "index.indexing.slowlog.threshold.index.info": "5s",
  6. "index.indexing.slowlog.threshold.index.debug": "2s",
  7. "index.indexing.slowlog.threshold.index.trace": "500ms",
  8. "index.indexing.slowlog.source": "1000"
  9. },
  10. )
  11. print(resp)
  1. response = client.indices.put_settings(
  2. index: 'my-index-000001',
  3. body: {
  4. 'index.indexing.slowlog.threshold.index.warn' => '10s',
  5. 'index.indexing.slowlog.threshold.index.info' => '5s',
  6. 'index.indexing.slowlog.threshold.index.debug' => '2s',
  7. 'index.indexing.slowlog.threshold.index.trace' => '500ms',
  8. 'index.indexing.slowlog.source' => '1000'
  9. }
  10. )
  11. puts response
  1. PUT /my-index-000001/_settings
  2. {
  3. "index.indexing.slowlog.threshold.index.warn": "10s",
  4. "index.indexing.slowlog.threshold.index.info": "5s",
  5. "index.indexing.slowlog.threshold.index.debug": "2s",
  6. "index.indexing.slowlog.threshold.index.trace": "500ms",
  7. "index.indexing.slowlog.source": "1000"
  8. }

To include information about the user that triggered a slow indexing event, use the index.indexing.slowlog.include.user setting.

  1. resp = client.indices.put_settings(
  2. index="my-index-000001",
  3. settings={
  4. "index.indexing.slowlog.include.user": True
  5. },
  6. )
  7. print(resp)
  1. response = client.indices.put_settings(
  2. index: 'my-index-000001',
  3. body: {
  4. 'index.indexing.slowlog.include.user' => true
  5. }
  6. )
  7. puts response
  1. PUT /my-index-000001/_settings
  2. {
  3. "index.indexing.slowlog.include.user": true
  4. }

By default Elasticsearch will log the first 1000 characters of the _source in the slowlog. You can change that with index.indexing.slowlog.source. Setting it to false or 0 will skip logging the source entirely, while setting it to true will log the entire source regardless of size. The original _source is reformatted by default to make sure that it fits on a single log line. If preserving the original document format is important, you can turn off reformatting by setting index.indexing.slowlog.reformat to false, which will cause the source to be logged “as is” and can potentially span multiple log lines.

The index slow log file is configured in the log4j2.properties file.

Slow log levels

You can mimic the search or indexing slow log level by setting appropriate threshold making “more verbose” loggers to be switched off. If for instance we want to simulate index.indexing.slowlog.level: INFO then all we need to do is to set index.indexing.slowlog.threshold.index.debug and index.indexing.slowlog.threshold.index.trace to -1.