aboutsummaryrefslogtreecommitdiff
path: root/spark-common/src/main/java/me/lucko/spark/common/sampler/window/ProtoTimeEncoder.java
blob: fb4a4fcd8c0619faac7563263c48b157b008da5d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/*
 * This file is part of spark.
 *
 *  Copyright (c) lucko (Luck) <luck@lucko.me>
 *  Copyright (c) contributors
 *
 *  This program is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

package me.lucko.spark.common.sampler.window;

import me.lucko.spark.common.sampler.async.jfr.Dictionary;
import me.lucko.spark.common.sampler.node.ThreadNode;

import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.LongAdder;
import java.util.function.LongToDoubleFunction;
import java.util.stream.IntStream;

/**
 * Encodes a map of int->double into a double array.
 */
public class ProtoTimeEncoder {

    /** A transformer function to transform the 'time' value from a long to a double */
    private final LongToDoubleFunction valueTransformer;

    /** A sorted array of all possible keys to encode */
    private final int[] keys;
    /** A map of key value -> index in the keys array */
    private final Map<Integer, Integer> keysToIndex;

    public ProtoTimeEncoder(LongToDoubleFunction valueTransformer, List<ThreadNode> sourceData) {
        this.valueTransformer = valueTransformer;

        // get an array of all keys that show up in the source data
        this.keys = sourceData.stream()
                .map(n -> n.getTimeWindows().stream().mapToInt(i -> i))
                .reduce(IntStream.empty(), IntStream::concat)
                .distinct()
                .sorted()
                .toArray();

        // construct a reverse index lookup
        this.keysToIndex = new HashMap<>(this.keys.length);
        for (int i = 0; i < this.keys.length; i++) {
            this.keysToIndex.put(this.keys[i], i);
        }
    }

    /**
     * Gets an array of the keys that could be encoded by this encoder.
     *
     * @return an array of keys
     */
    public int[] getKeys() {
        return this.keys;
    }

    /**
     * Encode a {@link Dictionary} (map) of times/durations into a double array.
     *
     * @param times a dictionary of times (unix-time millis -> duration in microseconds)
     * @return the times encoded as a double array
     */
    public double[] encode(Map<Integer, LongAdder> times) {
        // construct an array of values - length needs to exactly match the
        // number of keys, even if some values are zero.
        double[] array = new double[this.keys.length];

        times.forEach((key, value) -> {
            // get the index for the given key
            Integer idx = this.keysToIndex.get(key);
            if (idx == null) {
                throw new RuntimeException("No index for key " + key + " in " + this.keysToIndex.keySet());
            }

            // store in the array
            array[idx] = this.valueTransformer.applyAsDouble(value.longValue());
        });

        return array;
    }
}