All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.elasticsearch.xpack.core.ml.utils.ExponentialAverageCalculationContext Maven / Gradle / Ivy

/*
 * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 * or more contributor license agreements. Licensed under the Elastic License
 * 2.0; you may not use this file except in compliance with the Elastic License
 * 2.0.
 */
package org.elasticsearch.xpack.core.ml.utils;

import org.elasticsearch.common.Nullable;
import org.elasticsearch.common.ParseField;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.io.stream.StreamInput;
import org.elasticsearch.common.io.stream.StreamOutput;
import org.elasticsearch.common.io.stream.Writeable;
import org.elasticsearch.common.xcontent.ConstructingObjectParser;
import org.elasticsearch.common.xcontent.ObjectParser;
import org.elasticsearch.common.xcontent.ToXContentObject;
import org.elasticsearch.common.xcontent.XContentBuilder;
import org.elasticsearch.xpack.core.common.time.TimeUtils;

import java.io.IOException;
import java.time.Duration;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.time.temporal.TemporalUnit;
import java.util.Objects;

import static org.elasticsearch.common.xcontent.ConstructingObjectParser.optionalConstructorArg;

/**
 * Utility for calculating current value of exponentially-weighted moving average per fixed-sized time window.
 *
 * The formula for the current value of the exponentially-weighted moving average is:
 *
 *   currentExponentialAverageMs = alpha * previousExponentialAverageMs + (1 - alpha) * incrementalMetricValueMs
 *
 * where alpha depends on what fraction of the current time window we've already seen:
 *
 *   alpha = e^(-time_elapsed_since_window_start/window_size)
 *   time_elapsed_since_window_start = latestTimestamp - window_start
 *
 * The class holds 3 values based on which it performs the calculation:
 *  - incrementalMetricValueMs - accumulated value of the metric in the current time window
 *  - latestTimestamp - timestamp updated as the time passes through the current time window
 *  - previousExponentialAverageMs - exponential average for previous time windows
 *
 * incrementalMetricValueMs should be updated using {@link #increment}.
 * latestTimestamp should be updated using {@link #setLatestTimestamp}.
 * Because it can happen that the timestamp is not available while incrementing the metric value, it is the responsibility of the user
 * of this class to always call {@link #setLatestTimestamp} *after* all the relevant (i.e. referring to the points in time before the
 * latest timestamp mentioned) {@link #increment} calls are made.
 */
public class ExponentialAverageCalculationContext implements Writeable, ToXContentObject {

    public static final ParseField INCREMENTAL_METRIC_VALUE_MS = new ParseField("incremental_metric_value_ms");
    public static final ParseField LATEST_TIMESTAMP = new ParseField("latest_timestamp");
    public static final ParseField PREVIOUS_EXPONENTIAL_AVERAGE_MS = new ParseField("previous_exponential_average_ms");

    public static final ConstructingObjectParser PARSER =
        new ConstructingObjectParser<>(
            "exponential_average_calculation_context",
            true,
            args -> {
                Double incrementalMetricValueMs = (Double) args[0];
                Instant latestTimestamp = (Instant) args[1];
                Double previousExponentialAverageMs = (Double) args[2];
                return new ExponentialAverageCalculationContext(
                    getOrDefault(incrementalMetricValueMs, 0.0),
                    latestTimestamp,
                    previousExponentialAverageMs);
            });

    static {
        PARSER.declareDouble(optionalConstructorArg(), INCREMENTAL_METRIC_VALUE_MS);
        PARSER.declareField(
            optionalConstructorArg(),
            p -> TimeUtils.parseTimeFieldToInstant(p, LATEST_TIMESTAMP.getPreferredName()),
            LATEST_TIMESTAMP,
            ObjectParser.ValueType.VALUE);
        PARSER.declareDouble(optionalConstructorArg(), PREVIOUS_EXPONENTIAL_AVERAGE_MS);
    }

    private static final TemporalUnit WINDOW_UNIT = ChronoUnit.HOURS;
    private static final Duration WINDOW_SIZE = WINDOW_UNIT.getDuration();

    private double incrementalMetricValueMs;
    private Instant latestTimestamp;
    private Double previousExponentialAverageMs;

    public ExponentialAverageCalculationContext() {
        this(0.0, null, null);
    }

    public ExponentialAverageCalculationContext(
            double incrementalMetricValueMs,
            @Nullable Instant latestTimestamp,
            @Nullable Double previousExponentialAverageMs) {
        this.incrementalMetricValueMs = incrementalMetricValueMs;
        this.latestTimestamp = latestTimestamp != null ? Instant.ofEpochMilli(latestTimestamp.toEpochMilli()) : null;
        this.previousExponentialAverageMs = previousExponentialAverageMs;
    }

    public ExponentialAverageCalculationContext(ExponentialAverageCalculationContext lhs) {
        this(lhs.incrementalMetricValueMs, lhs.latestTimestamp, lhs.previousExponentialAverageMs);
    }

    public ExponentialAverageCalculationContext(StreamInput in) throws IOException {
        this.incrementalMetricValueMs = in.readDouble();
        this.latestTimestamp = in.readOptionalInstant();
        this.previousExponentialAverageMs = in.readOptionalDouble();
    }

    // Visible for testing
    public double getIncrementalMetricValueMs() {
        return incrementalMetricValueMs;
    }

    // Visible for testing
    public Instant getLatestTimestamp() {
        return latestTimestamp;
    }

    // Visible for testing
    public Double getPreviousExponentialAverageMs() {
        return previousExponentialAverageMs;
    }

    public Double getCurrentExponentialAverageMs() {
        if (previousExponentialAverageMs == null || latestTimestamp == null) return incrementalMetricValueMs;
        Instant currentWindowStartTimestamp = latestTimestamp.truncatedTo(WINDOW_UNIT);
        double alpha = Math.exp(
            - (double) Duration.between(currentWindowStartTimestamp, latestTimestamp).toMillis() / WINDOW_SIZE.toMillis());
        return alpha * previousExponentialAverageMs + (1 - alpha) * incrementalMetricValueMs;
    }

    /**
     * Increments the current accumulated metric value by the given delta.
     */
    public void increment(double metricValueDeltaMs) {
        incrementalMetricValueMs += metricValueDeltaMs;
    }

    /**
     * Sets the latest timestamp that serves as an indication of the current point in time.
     * Before calling this method make sure all the associated calls to {@link #increment} were already made.
     */
    public void setLatestTimestamp(Instant newLatestTimestamp) {
        Objects.requireNonNull(newLatestTimestamp);
        if (this.latestTimestamp != null) {
            Instant nextWindowStartTimestamp = this.latestTimestamp.truncatedTo(WINDOW_UNIT).plus(WINDOW_SIZE);
            if (newLatestTimestamp.compareTo(nextWindowStartTimestamp) >= 0) {
                // When we cross the boundary between windows, we update the exponential average with metric values accumulated so far in
                // incrementalMetricValueMs variable.
                this.previousExponentialAverageMs = getCurrentExponentialAverageMs();
                this.incrementalMetricValueMs = 0.0;
            }
        } else {
            // This is the first time {@link #setLatestRecordTimestamp} is called on this object.
        }
        if (this.latestTimestamp == null || newLatestTimestamp.isAfter(this.latestTimestamp)) {
            this.latestTimestamp = newLatestTimestamp;
        }
    }

    @Override
    public void writeTo(StreamOutput out) throws IOException {
        out.writeDouble(incrementalMetricValueMs);
        out.writeOptionalInstant(latestTimestamp);
        out.writeOptionalDouble(previousExponentialAverageMs);
    }

    @Override
    public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
        builder.startObject();
        builder.field(INCREMENTAL_METRIC_VALUE_MS.getPreferredName(), incrementalMetricValueMs);
        if (latestTimestamp != null) {
            builder.timeField(
                LATEST_TIMESTAMP.getPreferredName(),
                LATEST_TIMESTAMP.getPreferredName() + "_string",
                latestTimestamp.toEpochMilli());
        }
        if (previousExponentialAverageMs != null) {
            builder.field(PREVIOUS_EXPONENTIAL_AVERAGE_MS.getPreferredName(), previousExponentialAverageMs);
        }
        builder.endObject();
        return builder;
    }

    @Override
    public boolean equals(Object o) {
        if (o == this) return true;
        if (o == null || getClass() != o.getClass()) return false;
        ExponentialAverageCalculationContext that = (ExponentialAverageCalculationContext) o;
        return this.incrementalMetricValueMs == that.incrementalMetricValueMs
            && Objects.equals(this.latestTimestamp, that.latestTimestamp)
            && Objects.equals(this.previousExponentialAverageMs, that.previousExponentialAverageMs);
    }

    @Override
    public int hashCode() {
        return Objects.hash(incrementalMetricValueMs, latestTimestamp, previousExponentialAverageMs);
    }

    @Override
    public String toString() {
        return Strings.toString(this);
    }

    @SuppressWarnings("unchecked")
    private static  T getOrDefault(@Nullable T value, T defaultValue) {
        return value != null ? value : defaultValue;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy