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

com.vmware.xenon.common.ServiceStats Maven / Gradle / Ivy

There is a newer version: 1.6.18
Show newest version
/*
 * Copyright (c) 2014-2015 VMware, Inc. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
 * use this file except in compliance with the License.  You may obtain a copy of
 * the License at http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed
 * under the License is distributed on an "AS IS" BASIS, without warranties or
 * conditions of any kind, EITHER EXPRESS OR IMPLIED.  See the License for the
 * specific language governing permissions and limitations under the License.
 */

package com.vmware.xenon.common;

import java.net.URI;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.Map;
import java.util.SortedMap;
import java.util.concurrent.ConcurrentSkipListMap;
import java.util.concurrent.TimeUnit;

import com.esotericsoftware.kryo.serializers.VersionFieldSerializer.Since;

import com.vmware.xenon.common.serialization.ReleaseConstants;

/**
 * Document describing the /stats REST API
 */
public class ServiceStats extends ServiceDocument {
    public static final String KIND = Utils.buildKind(ServiceStats.class);
    public static final String STAT_NAME_SUFFIX_PER_DAY = "PerDay";
    public static final String STAT_NAME_SUFFIX_PER_HOUR = "PerHour";

    public static class ServiceStatLogHistogram {
        /**
         * Each bin tracks a power of 10. Bin[0] tracks all values between 0 and 9, Bin[1] tracks
         * values between 10 and 99, Bin[2] tracks values between 100 and 999, and so forth
         */
        public long[] bins = new long[15];
    }

    /**
     * Data structure representing time series data
     * Users can specify the number of bins to be stored and the granularity of the bins
     * Any data to be stored will be normalized to an UTC boundary based on the the data granularity
     * If a bin already exists for that timestamp, the existing value is updated based on the
     * specified AggregationType
     * If the number of bins equals the configured number of bins, the oldest bin will be dropped
     * on any further insertion
     */
    public static class TimeSeriesStats {

        public static class TimeBin {
            public Double avg;
            @Since(ReleaseConstants.RELEASE_VERSION_1_6_0)
            public Double var;
            public Double min;
            public Double max;
            public Double sum;
            public Double latest;
            public double count;
        }

        public enum AggregationType {
            AVG, MIN, MAX, SUM, LATEST
        }

        public SortedMap bins;
        public int numBins;
        public long binDurationMillis;
        public EnumSet aggregationType;

        public TimeSeriesStats() {
            // no-op
        }

        public TimeSeriesStats(int numBins, long binDurationMillis,
                EnumSet aggregationType) {
            this.numBins = numBins;
            this.binDurationMillis = binDurationMillis;
            this.bins = new ConcurrentSkipListMap<>();
            this.aggregationType = aggregationType;
        }

        public void add(long timestampMicros, double value, double delta) {
            synchronized (this) {
                long binId = normalizeTimestamp(timestampMicros, this.binDurationMillis);
                TimeBin dataBin = null;
                if (this.bins.containsKey(binId)) {
                    dataBin = this.bins.get(binId);
                } else {
                    if (this.bins.size() == this.numBins) {
                        if (this.bins.firstKey() > timestampMicros) {
                            // incoming data is too old; ignore
                            return;
                        }
                        // remove the oldest entry
                        this.bins.remove(this.bins.firstKey());
                    }
                    dataBin = new TimeBin();
                    this.bins.put(binId, dataBin);
                }
                if (this.aggregationType.contains(AggregationType.AVG)) {
                    if (dataBin.avg == null) {
                        dataBin.avg = value;
                        dataBin.var = 0.0;
                        dataBin.count = 1;
                    } else {
                        // Use Welford's algorithm for online computation of average and variance
                        // see https://en.wikipedia.org/wiki/Algorithms_for_calculating_variance
                        dataBin.count++;
                        double diff = value - dataBin.avg;
                        dataBin.avg += diff / dataBin.count;
                        double diffAfter = value - dataBin.avg;
                        dataBin.var += diff * diffAfter;
                    }
                }
                if (this.aggregationType.contains(AggregationType.SUM)) {
                    if (dataBin.sum == null) {
                        dataBin.sum = delta;
                    } else {
                        dataBin.sum += delta;
                    }
                }
                if (this.aggregationType.contains(AggregationType.MAX)) {
                    if (dataBin.max == null) {
                        dataBin.max = value;
                    } else if (dataBin.max < value) {
                        dataBin.max = value;
                    }
                }
                if (this.aggregationType.contains(AggregationType.MIN)) {
                    if (dataBin.min == null) {
                        dataBin.min = value;
                    } else if (dataBin.min > value) {
                        dataBin.min = value;
                    }
                }
                if (this.aggregationType.contains(AggregationType.LATEST)) {
                    dataBin.latest = value;
                }
            }
        }

        /**
         * This method normalizes the input timestamp at UTC time boundaries
         *  based on the bin size, effectively creating time series that are comparable to each other
         */
        private long normalizeTimestamp(long timestampMicros, long binDurationMillis) {
            long timeMillis = TimeUnit.MICROSECONDS.toMillis(timestampMicros);
            timeMillis -= (timeMillis % binDurationMillis);
            return timeMillis;
        }

    }

    public static class ServiceStat {
        public static final String KIND = Utils.buildKind(ServiceStat.class);

        public static final String FIELD_NAME_NAME = "name";

        public static final String FIELD_NAME_VERSION = "version";

        public static final String FIELD_NAME_LAST_UPDATE_TIME_MICROS_UTC = "lastUpdateMicrosUtc";

        public static final String FIELD_NAME_LATEST_VALUE = "latestValue";

        public static final String FIELD_NAME_UNIT = "unit";

        /**
         * Name of the stat.
         */
        public String name;

        /**
         * Latest value of the stat.
         */
        public double latestValue;

        /**
         * The value accumulated over time for the stat.
         */
        public double accumulatedValue;

        /**
         * The stat document version.
         */
        public long version;

        /**
         * Time, in microseconds since UNIX epoch, the stat was received at the service.
         */
        public long lastUpdateMicrosUtc;

        /**
         * The kind of document, in this case the ServiceStat kind.
         */
        public String kind = KIND;

        /**
         * The unit of measurement associated with the stat.
         */
        public String unit;

        /**
         * Time, in microseconds since UNIX epoch, the data value was acquired at the source.
         */
        public Long sourceTimeMicrosUtc;

        /**
         * Source (provider) for this stat
         */
        public URI serviceReference;

        /**
         *  histogram of logarithmic time scale
         */
        public ServiceStatLogHistogram logHistogram;

        /**
         * time series data. If set,  the stat value is
         * maintained over time. Users can choose the number
         * of samples to maintain and the time window into
         * which each datapoint falls. If more than one entry
         * exists, aggregates (average, minimum, maximum; based
         * on user choice) are maintained
         */
        public TimeSeriesStats timeSeriesStats;
    }

    public String kind = KIND;

    public Map entries = new HashMap<>();

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy