mirror of
https://github.com/ClusterCockpit/cc-metric-store.git
synced 2025-01-14 16:29:05 +01:00
New selector type for better selection of sockets/cpus
This commit is contained in:
parent
a269ab423b
commit
b55a67f869
4
api.go
4
api.go
@ -19,7 +19,7 @@ import (
|
|||||||
// }
|
// }
|
||||||
type ApiRequestBody struct {
|
type ApiRequestBody struct {
|
||||||
Metrics []string `json:"metrics"`
|
Metrics []string `json:"metrics"`
|
||||||
Selectors [][]string `json:"selectors"`
|
Selectors []Selector `json:"selectors"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ApiMetricData struct {
|
type ApiMetricData struct {
|
||||||
@ -165,7 +165,7 @@ func handleFree(rw http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bodyDec := json.NewDecoder(r.Body)
|
bodyDec := json.NewDecoder(r.Body)
|
||||||
var selectors [][]string
|
var selectors []Selector
|
||||||
err = bodyDec.Decode(&selectors)
|
err = bodyDec.Decode(&selectors)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(rw, err.Error(), http.StatusBadRequest)
|
http.Error(rw, err.Error(), http.StatusBadRequest)
|
||||||
|
127
memstore.go
127
memstore.go
@ -206,89 +206,6 @@ func (l *level) findLevelOrCreate(selector []string, nMetrics int) *level {
|
|||||||
return child.findLevelOrCreate(selector[1:], nMetrics)
|
return child.findLevelOrCreate(selector[1:], nMetrics)
|
||||||
}
|
}
|
||||||
|
|
||||||
// This function assmumes that `l.lock` is LOCKED!
|
|
||||||
// Read `buffer.read` for context.
|
|
||||||
// If this level does not have data for the requested metric, the data
|
|
||||||
// is aggregated timestep-wise from all the children (recursively).
|
|
||||||
func (l *level) read(offset int, from, to int64, data []Float) ([]Float, int, int64, int64, error) {
|
|
||||||
if b := l.metrics[offset]; b != nil {
|
|
||||||
// Whoo, this is the "native" level of this metric:
|
|
||||||
data, from, to, err := b.read(from, to, data)
|
|
||||||
return data, 1, from, to, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(l.children) == 0 {
|
|
||||||
return nil, 1, 0, 0, ErrNoData
|
|
||||||
}
|
|
||||||
|
|
||||||
n := 0
|
|
||||||
for _, child := range l.children {
|
|
||||||
child.lock.RLock()
|
|
||||||
cdata, cn, cfrom, cto, err := child.read(offset, from, to, data)
|
|
||||||
child.lock.RUnlock()
|
|
||||||
|
|
||||||
if err == ErrNoData {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, 0, 0, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if n == 0 {
|
|
||||||
data = cdata
|
|
||||||
from = cfrom
|
|
||||||
to = cto
|
|
||||||
n += cn
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if cfrom != from || cto != to {
|
|
||||||
return nil, 0, 0, 0, ErrDataDoesNotAlign
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(data) != len(cdata) {
|
|
||||||
panic("WTF? Different freq. at different levels?")
|
|
||||||
}
|
|
||||||
|
|
||||||
n += cn
|
|
||||||
}
|
|
||||||
|
|
||||||
if n == 0 {
|
|
||||||
return nil, 0, 0, 0, ErrNoData
|
|
||||||
}
|
|
||||||
|
|
||||||
return data, n, from, to, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *level) free(t int64) (int, error) {
|
|
||||||
l.lock.Lock()
|
|
||||||
defer l.lock.Unlock()
|
|
||||||
|
|
||||||
n := 0
|
|
||||||
for _, b := range l.metrics {
|
|
||||||
if b == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
m, err := b.free(t)
|
|
||||||
n += m
|
|
||||||
if err != nil {
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, l := range l.children {
|
|
||||||
m, err := l.free(t)
|
|
||||||
n += m
|
|
||||||
if err != nil {
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return n, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type AggregationStrategy int
|
type AggregationStrategy int
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -382,11 +299,7 @@ func (m *MemoryStore) Write(selector []string, ts int64, metrics []Metric) error
|
|||||||
// Returns all values for metric `metric` from `from` to `to` for the selected level.
|
// Returns all values for metric `metric` from `from` to `to` for the selected level.
|
||||||
// If the level does not hold the metric itself, the data will be aggregated recursively from the children.
|
// If the level does not hold the metric itself, the data will be aggregated recursively from the children.
|
||||||
// See `level.read` for more information.
|
// See `level.read` for more information.
|
||||||
func (m *MemoryStore) Read(selector []string, metric string, from, to int64) ([]Float, int64, int64, error) {
|
func (m *MemoryStore) Read(selector Selector, metric string, from, to int64) ([]Float, int64, int64, error) {
|
||||||
l := m.root.findLevelOrCreate(selector, len(m.metrics))
|
|
||||||
l.lock.RLock()
|
|
||||||
defer l.lock.RUnlock()
|
|
||||||
|
|
||||||
if from > to {
|
if from > to {
|
||||||
return nil, 0, 0, errors.New("invalid time range")
|
return nil, 0, 0, errors.New("invalid time range")
|
||||||
}
|
}
|
||||||
@ -396,13 +309,29 @@ func (m *MemoryStore) Read(selector []string, metric string, from, to int64) ([]
|
|||||||
return nil, 0, 0, errors.New("unkown metric: " + metric)
|
return nil, 0, 0, errors.New("unkown metric: " + metric)
|
||||||
}
|
}
|
||||||
|
|
||||||
data := make([]Float, (to-from)/minfo.frequency+1)
|
n, data := 0, make([]Float, (to-from)/minfo.frequency+1)
|
||||||
data, n, from, to, err := l.read(minfo.offset, from, to, data)
|
err := m.root.findBuffers(selector, minfo.offset, func(b *buffer) error {
|
||||||
|
cdata, cfrom, cto, err := b.read(from, to, data)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if n == 0 {
|
||||||
|
from, to = cfrom, cto
|
||||||
|
} else if from != cfrom || to != cto || len(data) != len(cdata) {
|
||||||
|
return ErrDataDoesNotAlign
|
||||||
|
}
|
||||||
|
|
||||||
|
data = cdata
|
||||||
|
n += 1
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, 0, 0, err
|
return nil, 0, 0, err
|
||||||
}
|
} else if n == 0 {
|
||||||
|
return nil, 0, 0, errors.New("metric not found")
|
||||||
if n > 1 {
|
} else if n > 1 {
|
||||||
if minfo.aggregation == AvgAggregation {
|
if minfo.aggregation == AvgAggregation {
|
||||||
normalize := 1. / Float(n)
|
normalize := 1. / Float(n)
|
||||||
for i := 0; i < len(data); i++ {
|
for i := 0; i < len(data); i++ {
|
||||||
@ -413,11 +342,17 @@ func (m *MemoryStore) Read(selector []string, metric string, from, to int64) ([]
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return data, from, to, err
|
return data, from, to, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Release all buffers for the selected level and all its children that contain only
|
// Release all buffers for the selected level and all its children that contain only
|
||||||
// values older than `t`.
|
// values older than `t`.
|
||||||
func (m *MemoryStore) Free(selector []string, t int64) (int, error) {
|
func (m *MemoryStore) Free(selector Selector, t int64) (int, error) {
|
||||||
return m.root.findLevelOrCreate(selector, len(m.metrics)).free(t)
|
n := 0
|
||||||
|
err := m.root.findBuffers(selector, -1, func(b *buffer) error {
|
||||||
|
m, err := b.free(t)
|
||||||
|
n += m
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
return n, err
|
||||||
}
|
}
|
||||||
|
@ -27,12 +27,13 @@ func TestMemoryStoreBasics(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
adata, from, to, err := store.Read([]string{"testhost"}, "a", 0, count*frequency)
|
sel := Selector{{String: "testhost"}}
|
||||||
|
adata, from, to, err := store.Read(sel, "a", 0, count*frequency)
|
||||||
if err != nil || from != 0 || to != count*frequency {
|
if err != nil || from != 0 || to != count*frequency {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
bdata, _, _, err := store.Read([]string{"testhost"}, "b", 0, count*frequency)
|
bdata, _, _, err := store.Read(sel, "b", 0, count*frequency)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
@ -80,7 +81,8 @@ func TestMemoryStoreMissingDatapoints(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
adata, _, _, err := store.Read([]string{"testhost"}, "a", 0, int64(count))
|
sel := Selector{{String: "testhost"}}
|
||||||
|
adata, _, _, err := store.Read(sel, "a", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
@ -133,7 +135,7 @@ func TestMemoryStoreAggregation(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
adata, from, to, err := store.Read([]string{"host0"}, "a", int64(0), int64(count))
|
adata, from, to, err := store.Read(Selector{{String: "host0"}}, "a", int64(0), int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
@ -152,7 +154,7 @@ func TestMemoryStoreAggregation(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bdata, from, to, err := store.Read([]string{"host0"}, "b", int64(0), int64(count))
|
bdata, from, to, err := store.Read(Selector{{String: "host0"}}, "b", int64(0), int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
@ -215,7 +217,7 @@ func TestMemoryStoreStats(t *testing.T) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
stats, from, to, err := store.Stats(sel1, "a", 0, int64(count))
|
stats, from, to, err := store.Stats(Selector{{String: "cluster"}, {String: "host1"}}, "a", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -228,7 +230,7 @@ func TestMemoryStoreStats(t *testing.T) {
|
|||||||
t.Fatalf("wrong stats: %#v\n", stats)
|
t.Fatalf("wrong stats: %#v\n", stats)
|
||||||
}
|
}
|
||||||
|
|
||||||
stats, from, to, err = store.Stats([]string{"cluster", "host2"}, "b", 0, int64(count))
|
stats, from, to, err = store.Stats(Selector{{String: "cluster"}, {String: "host2"}}, "b", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -283,7 +285,8 @@ func TestMemoryStoreArchive(t *testing.T) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
adata, from, to, err := store2.Read([]string{"cluster", "host", "cpu0"}, "a", 100, int64(100+count))
|
sel := Selector{{String: "cluster"}, {String: "host"}, {String: "cpu0"}}
|
||||||
|
adata, from, to, err := store2.Read(sel, "a", 100, int64(100+count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
@ -320,7 +323,7 @@ func TestMemoryStoreFree(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
n, err := store.Free([]string{"cluster", "host"}, int64(BUFFER_CAP*2)+100)
|
n, err := store.Free(Selector{{String: "cluster"}, {String: "host"}}, int64(BUFFER_CAP*2)+100)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -329,7 +332,7 @@ func TestMemoryStoreFree(t *testing.T) {
|
|||||||
t.Fatal("two buffers expected to be released")
|
t.Fatal("two buffers expected to be released")
|
||||||
}
|
}
|
||||||
|
|
||||||
adata, from, to, err := store.Read([]string{"cluster", "host", "1"}, "a", 0, int64(count))
|
adata, from, to, err := store.Read(Selector{{String: "cluster"}, {String: "host"}, {String: "1"}}, "a", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -338,7 +341,7 @@ func TestMemoryStoreFree(t *testing.T) {
|
|||||||
t.Fatalf("unexpected values from call to `Read`: from=%d, to=%d, len=%d", from, to, len(adata))
|
t.Fatalf("unexpected values from call to `Read`: from=%d, to=%d, len=%d", from, to, len(adata))
|
||||||
}
|
}
|
||||||
|
|
||||||
bdata, from, to, err := store.Read([]string{"cluster", "host", "1"}, "b", 0, int64(count))
|
bdata, from, to, err := store.Read(Selector{{String: "cluster"}, {String: "host"}, {String: "1"}}, "b", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -380,7 +383,8 @@ func BenchmarkMemoryStoreConcurrentWrites(b *testing.B) {
|
|||||||
|
|
||||||
for g := 0; g < goroutines; g++ {
|
for g := 0; g < goroutines; g++ {
|
||||||
host := fmt.Sprintf("host%d", g)
|
host := fmt.Sprintf("host%d", g)
|
||||||
adata, _, _, err := store.Read([]string{"cluster", host, "cpu0"}, "a", 0, int64(count)*frequency)
|
sel := Selector{{String: "cluster"}, {String: host}, {String: "cpu0"}}
|
||||||
|
adata, _, _, err := store.Read(sel, "a", 0, int64(count)*frequency)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
b.Error(err)
|
b.Error(err)
|
||||||
return
|
return
|
||||||
@ -429,7 +433,7 @@ func BenchmarkMemoryStoreAggregation(b *testing.B) {
|
|||||||
|
|
||||||
b.StartTimer()
|
b.StartTimer()
|
||||||
for n := 0; n < b.N; n++ {
|
for n := 0; n < b.N; n++ {
|
||||||
data, from, to, err := store.Read(sel[0:2], "flops_any", 0, int64(count))
|
data, from, to, err := store.Read(Selector{{String: "testcluster"}, {String: "host123"}}, "flops_any", 0, int64(count))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
b.Fatal(err)
|
b.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -124,7 +124,7 @@ func main() {
|
|||||||
if conf.RetentionHours > 0 {
|
if conf.RetentionHours > 0 {
|
||||||
log.Println("Freeing up memory...")
|
log.Println("Freeing up memory...")
|
||||||
t := now.Add(-time.Duration(conf.RetentionHours) * time.Hour)
|
t := now.Add(-time.Duration(conf.RetentionHours) * time.Hour)
|
||||||
freed, err := memoryStore.Free([]string{}, t.Unix())
|
freed, err := memoryStore.Free(Selector{}, t.Unix())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Freeing up memory failed: %s\n", err.Error())
|
log.Printf("Freeing up memory failed: %s\n", err.Error())
|
||||||
}
|
}
|
||||||
|
95
selector.go
Normal file
95
selector.go
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
type SelectorElement struct {
|
||||||
|
String string
|
||||||
|
Group []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (se *SelectorElement) UnmarshalJSON(input []byte) error {
|
||||||
|
if input[0] == '"' {
|
||||||
|
return json.Unmarshal(input, &se.String)
|
||||||
|
}
|
||||||
|
|
||||||
|
if input[0] == '[' {
|
||||||
|
return json.Unmarshal(input, &se.Group)
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors.New("the Go SelectorElement type can only be a string or an array of strings")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (se *SelectorElement) MarshalJSON() ([]byte, error) {
|
||||||
|
if se.String != "" {
|
||||||
|
return json.Marshal(se.String)
|
||||||
|
}
|
||||||
|
|
||||||
|
if se.Group != nil {
|
||||||
|
return json.Marshal(se.Group)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, errors.New("a Go Selector must be a non-empty string or a non-empty slice of strings")
|
||||||
|
}
|
||||||
|
|
||||||
|
type Selector []SelectorElement
|
||||||
|
|
||||||
|
func (l *level) findBuffers(selector Selector, offset int, f func(b *buffer) error) error {
|
||||||
|
l.lock.RLock()
|
||||||
|
defer l.lock.RUnlock()
|
||||||
|
|
||||||
|
if len(selector) == 0 {
|
||||||
|
if offset == -1 {
|
||||||
|
for _, b := range l.metrics {
|
||||||
|
if b != nil {
|
||||||
|
err := f(b)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
b := l.metrics[offset]
|
||||||
|
if b != nil {
|
||||||
|
return f(b)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, lvl := range l.children {
|
||||||
|
err := lvl.findBuffers(nil, offset, f)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
sel := selector[0]
|
||||||
|
if len(sel.String) != 0 {
|
||||||
|
lvl, ok := l.children[sel.String]
|
||||||
|
if ok {
|
||||||
|
err := lvl.findBuffers(selector[1:], offset, f)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if sel.Group != nil {
|
||||||
|
for _, key := range sel.Group {
|
||||||
|
lvl, ok := l.children[key]
|
||||||
|
if ok {
|
||||||
|
err := lvl.findBuffers(selector[1:], offset, f)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
panic("impossible")
|
||||||
|
}
|
50
stats.go
50
stats.go
@ -118,11 +118,7 @@ func (l *level) stats(offset int, from, to int64, aggreg AggregationStrategy) (S
|
|||||||
}, from, to, nil
|
}, from, to, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *MemoryStore) Stats(selector []string, metric string, from, to int64) (*Stats, int64, int64, error) {
|
func (m *MemoryStore) Stats(selector Selector, metric string, from, to int64) (*Stats, int64, int64, error) {
|
||||||
l := m.root.findLevelOrCreate(selector, len(m.metrics))
|
|
||||||
l.lock.RLock()
|
|
||||||
defer l.lock.RUnlock()
|
|
||||||
|
|
||||||
if from > to {
|
if from > to {
|
||||||
return nil, 0, 0, errors.New("invalid time range")
|
return nil, 0, 0, errors.New("invalid time range")
|
||||||
}
|
}
|
||||||
@ -132,6 +128,46 @@ func (m *MemoryStore) Stats(selector []string, metric string, from, to int64) (*
|
|||||||
return nil, 0, 0, errors.New("unkown metric: " + metric)
|
return nil, 0, 0, errors.New("unkown metric: " + metric)
|
||||||
}
|
}
|
||||||
|
|
||||||
stats, from, to, err := l.stats(minfo.offset, from, to, minfo.aggregation)
|
n, samples := 0, 0
|
||||||
return &stats, from, to, err
|
avg, min, max := Float(0), math.MaxFloat32, -math.MaxFloat32
|
||||||
|
err := m.root.findBuffers(selector, minfo.offset, func(b *buffer) error {
|
||||||
|
stats, cfrom, cto, err := b.stats(from, to)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if n == 0 {
|
||||||
|
from, to = cfrom, cto
|
||||||
|
} else if from != cfrom || to != cto {
|
||||||
|
return ErrDataDoesNotAlign
|
||||||
|
}
|
||||||
|
|
||||||
|
samples += stats.Samples
|
||||||
|
avg += stats.Avg
|
||||||
|
min = math.Min(min, float64(stats.Min))
|
||||||
|
max = math.Max(max, float64(stats.Max))
|
||||||
|
n += 1
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, 0, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if n == 0 {
|
||||||
|
return nil, 0, 0, ErrNoData
|
||||||
|
}
|
||||||
|
|
||||||
|
if minfo.aggregation == AvgAggregation {
|
||||||
|
avg /= Float(n)
|
||||||
|
} else if n > 1 && minfo.aggregation != SumAggregation {
|
||||||
|
return nil, 0, 0, errors.New("invalid aggregation")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &Stats{
|
||||||
|
Samples: samples,
|
||||||
|
Avg: avg,
|
||||||
|
Min: Float(min),
|
||||||
|
Max: Float(max),
|
||||||
|
}, from, to, nil
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user