Median absolute deviation aggregation

Median absolute deviation aggregation

This single-value aggregation approximates the median absolute deviation of its search results.

Median absolute deviation is a measure of variability. It is a robust statistic, meaning that it is useful for describing data that may have outliers, or may not be normally distributed. For such data it can be more descriptive than standard deviation.

It is calculated as the median of each data point’s deviation from the median of the entire sample. That is, for a random variable X, the median absolute deviation is median(|median(X) - Xi|).

Example

Assume our data represents product reviews on a one to five star scale. Such reviews are usually summarized as a mean, which is easily understandable but doesn’t describe the reviews’ variability. Estimating the median absolute deviation can provide insight into how much reviews vary from one another.

In this example we have a product which has an average rating of 3 stars. Let’s look at its ratings’ median absolute deviation to determine how much they vary

  1. resp = client.search(
  2. index="reviews",
  3. size=0,
  4. aggs={
  5. "review_average": {
  6. "avg": {
  7. "field": "rating"
  8. }
  9. },
  10. "review_variability": {
  11. "median_absolute_deviation": {
  12. "field": "rating"
  13. }
  14. }
  15. },
  16. )
  17. print(resp)
  1. response = client.search(
  2. index: 'reviews',
  3. body: {
  4. size: 0,
  5. aggregations: {
  6. review_average: {
  7. avg: {
  8. field: 'rating'
  9. }
  10. },
  11. review_variability: {
  12. median_absolute_deviation: {
  13. field: 'rating'
  14. }
  15. }
  16. }
  17. }
  18. )
  19. puts response
  1. const response = await client.search({
  2. index: "reviews",
  3. size: 0,
  4. aggs: {
  5. review_average: {
  6. avg: {
  7. field: "rating",
  8. },
  9. },
  10. review_variability: {
  11. median_absolute_deviation: {
  12. field: "rating",
  13. },
  14. },
  15. },
  16. });
  17. console.log(response);
  1. GET reviews/_search
  2. {
  3. "size": 0,
  4. "aggs": {
  5. "review_average": {
  6. "avg": {
  7. "field": "rating"
  8. }
  9. },
  10. "review_variability": {
  11. "median_absolute_deviation": {
  12. "field": "rating"
  13. }
  14. }
  15. }
  16. }

rating must be a numeric field

The resulting median absolute deviation of 2 tells us that there is a fair amount of variability in the ratings. Reviewers must have diverse opinions about this product.

  1. {
  2. ...
  3. "aggregations": {
  4. "review_average": {
  5. "value": 3.0
  6. },
  7. "review_variability": {
  8. "value": 2.0
  9. }
  10. }
  11. }

Approximation

The naive implementation of calculating median absolute deviation stores the entire sample in memory, so this aggregation instead calculates an approximation. It uses the TDigest data structure to approximate the sample median and the median of deviations from the sample median. For more about the approximation characteristics of TDigests, see Percentiles are (usually) approximate.

The tradeoff between resource usage and accuracy of a TDigest’s quantile approximation, and therefore the accuracy of this aggregation’s approximation of median absolute deviation, is controlled by the compression parameter. A higher compression setting provides a more accurate approximation at the cost of higher memory usage. For more about the characteristics of the TDigest compression parameter see Compression.

  1. resp = client.search(
  2. index="reviews",
  3. size=0,
  4. aggs={
  5. "review_variability": {
  6. "median_absolute_deviation": {
  7. "field": "rating",
  8. "compression": 100
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'reviews',
  3. body: {
  4. size: 0,
  5. aggregations: {
  6. review_variability: {
  7. median_absolute_deviation: {
  8. field: 'rating',
  9. compression: 100
  10. }
  11. }
  12. }
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "reviews",
  3. size: 0,
  4. aggs: {
  5. review_variability: {
  6. median_absolute_deviation: {
  7. field: "rating",
  8. compression: 100,
  9. },
  10. },
  11. },
  12. });
  13. console.log(response);
  1. GET reviews/_search
  2. {
  3. "size": 0,
  4. "aggs": {
  5. "review_variability": {
  6. "median_absolute_deviation": {
  7. "field": "rating",
  8. "compression": 100
  9. }
  10. }
  11. }
  12. }

The default compression value for this aggregation is 1000. At this compression level this aggregation is usually within 5% of the exact result, but observed performance will depend on the sample data.

Script

In the example above, product reviews are on a scale of one to five. If you want to modify them to a scale of one to ten, use a runtime field.

  1. resp = client.search(
  2. index="reviews",
  3. filter_path="aggregations",
  4. size=0,
  5. runtime_mappings={
  6. "rating.out_of_ten": {
  7. "type": "long",
  8. "script": {
  9. "source": "emit(doc['rating'].value * params.scaleFactor)",
  10. "params": {
  11. "scaleFactor": 2
  12. }
  13. }
  14. }
  15. },
  16. aggs={
  17. "review_average": {
  18. "avg": {
  19. "field": "rating.out_of_ten"
  20. }
  21. },
  22. "review_variability": {
  23. "median_absolute_deviation": {
  24. "field": "rating.out_of_ten"
  25. }
  26. }
  27. },
  28. )
  29. print(resp)
  1. response = client.search(
  2. index: 'reviews',
  3. filter_path: 'aggregations',
  4. body: {
  5. size: 0,
  6. runtime_mappings: {
  7. 'rating.out_of_ten' => {
  8. type: 'long',
  9. script: {
  10. source: "emit(doc['rating'].value * params.scaleFactor)",
  11. params: {
  12. "scaleFactor": 2
  13. }
  14. }
  15. }
  16. },
  17. aggregations: {
  18. review_average: {
  19. avg: {
  20. field: 'rating.out_of_ten'
  21. }
  22. },
  23. review_variability: {
  24. median_absolute_deviation: {
  25. field: 'rating.out_of_ten'
  26. }
  27. }
  28. }
  29. }
  30. )
  31. puts response
  1. const response = await client.search({
  2. index: "reviews",
  3. filter_path: "aggregations",
  4. size: 0,
  5. runtime_mappings: {
  6. "rating.out_of_ten": {
  7. type: "long",
  8. script: {
  9. source: "emit(doc['rating'].value * params.scaleFactor)",
  10. params: {
  11. scaleFactor: 2,
  12. },
  13. },
  14. },
  15. },
  16. aggs: {
  17. review_average: {
  18. avg: {
  19. field: "rating.out_of_ten",
  20. },
  21. },
  22. review_variability: {
  23. median_absolute_deviation: {
  24. field: "rating.out_of_ten",
  25. },
  26. },
  27. },
  28. });
  29. console.log(response);
  1. GET reviews/_search?filter_path=aggregations
  2. {
  3. "size": 0,
  4. "runtime_mappings": {
  5. "rating.out_of_ten": {
  6. "type": "long",
  7. "script": {
  8. "source": "emit(doc['rating'].value * params.scaleFactor)",
  9. "params": {
  10. "scaleFactor": 2
  11. }
  12. }
  13. }
  14. },
  15. "aggs": {
  16. "review_average": {
  17. "avg": {
  18. "field": "rating.out_of_ten"
  19. }
  20. },
  21. "review_variability": {
  22. "median_absolute_deviation": {
  23. "field": "rating.out_of_ten"
  24. }
  25. }
  26. }
  27. }

Which will result in:

  1. {
  2. "aggregations": {
  3. "review_average": {
  4. "value": 6.0
  5. },
  6. "review_variability": {
  7. "value": 4.0
  8. }
  9. }
  10. }

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.

Let’s be optimistic and assume some reviewers loved the product so much that they forgot to give it a rating. We’ll assign them five stars

  1. resp = client.search(
  2. index="reviews",
  3. size=0,
  4. aggs={
  5. "review_variability": {
  6. "median_absolute_deviation": {
  7. "field": "rating",
  8. "missing": 5
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'reviews',
  3. body: {
  4. size: 0,
  5. aggregations: {
  6. review_variability: {
  7. median_absolute_deviation: {
  8. field: 'rating',
  9. missing: 5
  10. }
  11. }
  12. }
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "reviews",
  3. size: 0,
  4. aggs: {
  5. review_variability: {
  6. median_absolute_deviation: {
  7. field: "rating",
  8. missing: 5,
  9. },
  10. },
  11. },
  12. });
  13. console.log(response);
  1. GET reviews/_search
  2. {
  3. "size": 0,
  4. "aggs": {
  5. "review_variability": {
  6. "median_absolute_deviation": {
  7. "field": "rating",
  8. "missing": 5
  9. }
  10. }
  11. }
  12. }