String stats aggregation

String stats aggregation

A multi-value metrics aggregation that computes statistics over string values extracted from the aggregated documents. These values can be retrieved either from specific keyword fields.

The string stats aggregation returns the following results:

  • count - The number of non-empty fields counted.
  • min_length - The length of the shortest term.
  • max_length - The length of the longest term.
  • avg_length - The average length computed over all terms.
  • entropy - The Shannon Entropy) value computed over all terms collected by the aggregation. Shannon entropy quantifies the amount of information contained in the field. It is a very useful metric for measuring a wide range of properties of a data set, such as diversity, similarity, randomness etc.

For example:

  1. resp = client.search(
  2. index="my-index-000001",
  3. size="0",
  4. aggs={
  5. "message_stats": {
  6. "string_stats": {
  7. "field": "message.keyword"
  8. }
  9. }
  10. },
  11. )
  12. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. message_stats: {
  7. string_stats: {
  8. field: 'message.keyword'
  9. }
  10. }
  11. }
  12. }
  13. )
  14. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. size: 0,
  4. aggs: {
  5. message_stats: {
  6. string_stats: {
  7. field: "message.keyword",
  8. },
  9. },
  10. },
  11. });
  12. console.log(response);
  1. POST /my-index-000001/_search?size=0
  2. {
  3. "aggs": {
  4. "message_stats": { "string_stats": { "field": "message.keyword" } }
  5. }
  6. }

The above aggregation computes the string statistics for the message field in all documents. The aggregation type is string_stats and the field parameter defines the field of the documents the stats will be computed on. The above will return the following:

  1. {
  2. ...
  3. "aggregations": {
  4. "message_stats": {
  5. "count": 5,
  6. "min_length": 24,
  7. "max_length": 30,
  8. "avg_length": 28.8,
  9. "entropy": 3.94617750050791
  10. }
  11. }
  12. }

The name of the aggregation (message_stats above) also serves as the key by which the aggregation result can be retrieved from the returned response.

Character distribution

The computation of the Shannon Entropy value is based on the probability of each character appearing in all terms collected by the aggregation. To view the probability distribution for all characters, we can add the show_distribution (default: false) parameter.

  1. resp = client.search(
  2. index="my-index-000001",
  3. size="0",
  4. aggs={
  5. "message_stats": {
  6. "string_stats": {
  7. "field": "message.keyword",
  8. "show_distribution": True
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. message_stats: {
  7. string_stats: {
  8. field: 'message.keyword',
  9. show_distribution: true
  10. }
  11. }
  12. }
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. size: 0,
  4. aggs: {
  5. message_stats: {
  6. string_stats: {
  7. field: "message.keyword",
  8. show_distribution: true,
  9. },
  10. },
  11. },
  12. });
  13. console.log(response);
  1. POST /my-index-000001/_search?size=0
  2. {
  3. "aggs": {
  4. "message_stats": {
  5. "string_stats": {
  6. "field": "message.keyword",
  7. "show_distribution": true
  8. }
  9. }
  10. }
  11. }

Set the show_distribution parameter to true, so that probability distribution for all characters is returned in the results.

  1. {
  2. ...
  3. "aggregations": {
  4. "message_stats": {
  5. "count": 5,
  6. "min_length": 24,
  7. "max_length": 30,
  8. "avg_length": 28.8,
  9. "entropy": 3.94617750050791,
  10. "distribution": {
  11. " ": 0.1527777777777778,
  12. "e": 0.14583333333333334,
  13. "s": 0.09722222222222222,
  14. "m": 0.08333333333333333,
  15. "t": 0.0763888888888889,
  16. "h": 0.0625,
  17. "a": 0.041666666666666664,
  18. "i": 0.041666666666666664,
  19. "r": 0.041666666666666664,
  20. "g": 0.034722222222222224,
  21. "n": 0.034722222222222224,
  22. "o": 0.034722222222222224,
  23. "u": 0.034722222222222224,
  24. "b": 0.027777777777777776,
  25. "w": 0.027777777777777776,
  26. "c": 0.013888888888888888,
  27. "E": 0.006944444444444444,
  28. "l": 0.006944444444444444,
  29. "1": 0.006944444444444444,
  30. "2": 0.006944444444444444,
  31. "3": 0.006944444444444444,
  32. "4": 0.006944444444444444,
  33. "y": 0.006944444444444444
  34. }
  35. }
  36. }
  37. }

The distribution object shows the probability of each character appearing in all terms. The characters are sorted by descending probability.

Script

If you need to get the string_stats for something more complex than a single field, run the aggregation on a runtime field.

  1. resp = client.search(
  2. index="my-index-000001",
  3. size=0,
  4. runtime_mappings={
  5. "message_and_context": {
  6. "type": "keyword",
  7. "script": "\n emit(doc['message.keyword'].value + ' ' + doc['context.keyword'].value)\n "
  8. }
  9. },
  10. aggs={
  11. "message_stats": {
  12. "string_stats": {
  13. "field": "message_and_context"
  14. }
  15. }
  16. },
  17. )
  18. print(resp)
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. size: 0,
  4. runtime_mappings: {
  5. message_and_context: {
  6. type: "keyword",
  7. script:
  8. "\n emit(doc['message.keyword'].value + ' ' + doc['context.keyword'].value)\n ",
  9. },
  10. },
  11. aggs: {
  12. message_stats: {
  13. string_stats: {
  14. field: "message_and_context",
  15. },
  16. },
  17. },
  18. });
  19. console.log(response);
  1. POST /my-index-000001/_search
  2. {
  3. "size": 0,
  4. "runtime_mappings": {
  5. "message_and_context": {
  6. "type": "keyword",
  7. "script": """
  8. emit(doc['message.keyword'].value + ' ' + doc['context.keyword'].value)
  9. """
  10. }
  11. },
  12. "aggs": {
  13. "message_stats": {
  14. "string_stats": { "field": "message_and_context" }
  15. }
  16. }
  17. }

Missing value

The missing parameter defines how documents that are missing a value should be treated. By default they will be ignored but it is also possible to treat them as if they had a value.

  1. resp = client.search(
  2. index="my-index-000001",
  3. size="0",
  4. aggs={
  5. "message_stats": {
  6. "string_stats": {
  7. "field": "message.keyword",
  8. "missing": "[empty message]"
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. size: 0,
  4. body: {
  5. aggregations: {
  6. message_stats: {
  7. string_stats: {
  8. field: 'message.keyword',
  9. missing: '[empty message]'
  10. }
  11. }
  12. }
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. size: 0,
  4. aggs: {
  5. message_stats: {
  6. string_stats: {
  7. field: "message.keyword",
  8. missing: "[empty message]",
  9. },
  10. },
  11. },
  12. });
  13. console.log(response);
  1. POST /my-index-000001/_search?size=0
  2. {
  3. "aggs": {
  4. "message_stats": {
  5. "string_stats": {
  6. "field": "message.keyword",
  7. "missing": "[empty message]"
  8. }
  9. }
  10. }
  11. }

Documents without a value in the message field will be treated as documents that have the value [empty message].