2018-09-09 15:52:59 +02:00
|
|
|
// Copyright (C) 2018 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
|
2014-06-28 11:24:25 +02:00
|
|
|
package main
|
|
|
|
|
2016-05-30 09:52:38 +02:00
|
|
|
import (
|
2018-02-20 08:03:34 +01:00
|
|
|
"regexp"
|
2016-05-30 09:52:38 +02:00
|
|
|
"sort"
|
2018-02-25 17:55:08 +01:00
|
|
|
"strconv"
|
2016-05-30 09:52:38 +02:00
|
|
|
"strings"
|
|
|
|
)
|
2014-06-28 11:24:25 +02:00
|
|
|
|
|
|
|
type analytic struct {
|
|
|
|
Key string
|
|
|
|
Count int
|
|
|
|
Percentage float64
|
2016-05-30 09:52:38 +02:00
|
|
|
Items []analytic `json:",omitempty"`
|
2014-06-28 11:24:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type analyticList []analytic
|
|
|
|
|
|
|
|
func (l analyticList) Less(a, b int) bool {
|
2016-05-30 09:52:38 +02:00
|
|
|
if l[a].Key == "Others" {
|
2019-05-15 13:42:55 +02:00
|
|
|
return false
|
2016-05-30 09:52:38 +02:00
|
|
|
}
|
|
|
|
if l[b].Key == "Others" {
|
2019-05-15 13:42:55 +02:00
|
|
|
return true
|
2016-05-30 09:52:38 +02:00
|
|
|
}
|
2014-06-28 11:24:25 +02:00
|
|
|
return l[b].Count < l[a].Count // inverse
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l analyticList) Swap(a, b int) {
|
|
|
|
l[a], l[b] = l[b], l[a]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l analyticList) Len() int {
|
|
|
|
return len(l)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a list of frequency analytics for a given list of strings.
|
2014-12-09 16:52:02 +01:00
|
|
|
func analyticsFor(ss []string, cutoff int) []analytic {
|
2014-06-28 11:24:25 +02:00
|
|
|
m := make(map[string]int)
|
|
|
|
t := 0
|
|
|
|
for _, s := range ss {
|
|
|
|
m[s]++
|
|
|
|
t++
|
|
|
|
}
|
|
|
|
|
|
|
|
l := make([]analytic, 0, len(m))
|
|
|
|
for k, c := range m {
|
2016-05-30 09:52:38 +02:00
|
|
|
l = append(l, analytic{
|
|
|
|
Key: k,
|
|
|
|
Count: c,
|
|
|
|
Percentage: 100 * float64(c) / float64(t),
|
|
|
|
})
|
2014-06-28 11:24:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sort.Sort(analyticList(l))
|
2014-12-09 16:52:02 +01:00
|
|
|
|
|
|
|
if cutoff > 0 && len(l) > cutoff {
|
|
|
|
c := 0
|
|
|
|
for _, i := range l[cutoff:] {
|
|
|
|
c += i.Count
|
|
|
|
}
|
2016-05-30 09:52:38 +02:00
|
|
|
l = append(l[:cutoff], analytic{
|
|
|
|
Key: "Others",
|
|
|
|
Count: c,
|
|
|
|
Percentage: 100 * float64(c) / float64(t),
|
|
|
|
})
|
2014-12-09 16:52:02 +01:00
|
|
|
}
|
|
|
|
|
2014-06-28 11:24:25 +02:00
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
2018-02-25 17:55:08 +01:00
|
|
|
// Find the points at which certain penetration levels are met
|
|
|
|
func penetrationLevels(as []analytic, points []float64) []analytic {
|
|
|
|
sort.Slice(as, func(a, b int) bool {
|
|
|
|
return versionLess(as[b].Key, as[a].Key)
|
|
|
|
})
|
|
|
|
|
|
|
|
var res []analytic
|
|
|
|
|
|
|
|
idx := 0
|
|
|
|
sum := 0.0
|
|
|
|
for _, a := range as {
|
|
|
|
sum += a.Percentage
|
|
|
|
if sum >= points[idx] {
|
|
|
|
a.Count = int(points[idx])
|
|
|
|
a.Percentage = sum
|
|
|
|
res = append(res, a)
|
|
|
|
idx++
|
|
|
|
if idx == len(points) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2014-06-28 11:24:25 +02:00
|
|
|
func statsForInts(data []int) [4]float64 {
|
|
|
|
var res [4]float64
|
|
|
|
if len(data) == 0 {
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Ints(data)
|
|
|
|
res[0] = float64(data[int(float64(len(data))*0.05)])
|
|
|
|
res[1] = float64(data[len(data)/2])
|
|
|
|
res[2] = float64(data[int(float64(len(data))*0.95)])
|
|
|
|
res[3] = float64(data[len(data)-1])
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2020-06-23 09:47:15 +01:00
|
|
|
func statsForInt64s(data []int64) [4]float64 {
|
|
|
|
var res [4]float64
|
|
|
|
if len(data) == 0 {
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(data, func(a, b int) bool {
|
|
|
|
return data[a] < data[b]
|
|
|
|
})
|
|
|
|
|
|
|
|
res[0] = float64(data[int(float64(len(data))*0.05)])
|
|
|
|
res[1] = float64(data[len(data)/2])
|
|
|
|
res[2] = float64(data[int(float64(len(data))*0.95)])
|
|
|
|
res[3] = float64(data[len(data)-1])
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2014-06-28 11:24:25 +02:00
|
|
|
func statsForFloats(data []float64) [4]float64 {
|
|
|
|
var res [4]float64
|
|
|
|
if len(data) == 0 {
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Float64s(data)
|
|
|
|
res[0] = data[int(float64(len(data))*0.05)]
|
|
|
|
res[1] = data[len(data)/2]
|
|
|
|
res[2] = data[int(float64(len(data))*0.95)]
|
|
|
|
res[3] = data[len(data)-1]
|
|
|
|
return res
|
|
|
|
}
|
2016-05-30 09:52:38 +02:00
|
|
|
|
|
|
|
func group(by func(string) string, as []analytic, perGroup int) []analytic {
|
|
|
|
var res []analytic
|
|
|
|
|
|
|
|
next:
|
|
|
|
for _, a := range as {
|
|
|
|
group := by(a.Key)
|
|
|
|
for i := range res {
|
|
|
|
if res[i].Key == group {
|
|
|
|
res[i].Count += a.Count
|
|
|
|
res[i].Percentage += a.Percentage
|
|
|
|
if len(res[i].Items) < perGroup {
|
|
|
|
res[i].Items = append(res[i].Items, a)
|
|
|
|
}
|
|
|
|
continue next
|
|
|
|
}
|
|
|
|
}
|
|
|
|
res = append(res, analytic{
|
|
|
|
Key: group,
|
|
|
|
Count: a.Count,
|
|
|
|
Percentage: a.Percentage,
|
|
|
|
Items: []analytic{a},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Sort(analyticList(res))
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func byVersion(s string) string {
|
|
|
|
parts := strings.Split(s, ".")
|
|
|
|
if len(parts) >= 2 {
|
|
|
|
return strings.Join(parts[:2], ".")
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func byPlatform(s string) string {
|
|
|
|
parts := strings.Split(s, "-")
|
|
|
|
if len(parts) >= 2 {
|
|
|
|
return parts[0]
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
2016-06-07 08:12:32 +02:00
|
|
|
|
2018-02-20 08:03:34 +01:00
|
|
|
var numericGoVersion = regexp.MustCompile(`^go[0-9]\.[0-9]+`)
|
|
|
|
|
2016-06-07 08:12:32 +02:00
|
|
|
func byCompiler(s string) string {
|
2018-02-20 08:03:34 +01:00
|
|
|
if m := numericGoVersion.FindString(s); m != "" {
|
|
|
|
return m
|
2016-06-07 08:12:32 +02:00
|
|
|
}
|
|
|
|
return "Other"
|
|
|
|
}
|
2018-02-25 17:55:08 +01:00
|
|
|
|
|
|
|
func versionLess(a, b string) bool {
|
|
|
|
arel, apre := versionParts(a)
|
|
|
|
brel, bpre := versionParts(b)
|
|
|
|
|
|
|
|
minlen := len(arel)
|
|
|
|
if l := len(brel); l < minlen {
|
|
|
|
minlen = l
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < minlen; i++ {
|
|
|
|
if arel[i] != brel[i] {
|
|
|
|
return arel[i] < brel[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-24 09:08:58 +01:00
|
|
|
// Longer version is newer, when the preceding parts are equal
|
2018-02-25 17:55:08 +01:00
|
|
|
if len(arel) != len(brel) {
|
|
|
|
return len(arel) < len(brel)
|
|
|
|
}
|
|
|
|
|
2018-03-24 09:08:58 +01:00
|
|
|
if apre != bpre {
|
|
|
|
// "(+dev)" versions are ahead
|
|
|
|
if apre == plusStr {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if bpre == plusStr {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return apre < bpre
|
|
|
|
}
|
|
|
|
|
2018-02-25 17:55:08 +01:00
|
|
|
// don't actually care how the prerelease stuff compares for our purposes
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2018-03-24 09:08:58 +01:00
|
|
|
// Split a version as returned from transformVersion into parts.
|
2018-02-25 17:55:08 +01:00
|
|
|
// "1.2.3-beta.2" -> []int{1, 2, 3}, "beta.2"}
|
|
|
|
func versionParts(v string) ([]int, string) {
|
2018-03-24 09:08:58 +01:00
|
|
|
parts := strings.SplitN(v[1:], " ", 2) // " (+dev)" versions
|
|
|
|
if len(parts) == 1 {
|
|
|
|
parts = strings.SplitN(parts[0], "-", 2) // "-rc.1" type versions
|
2018-02-25 17:55:08 +01:00
|
|
|
}
|
|
|
|
fields := strings.Split(parts[0], ".")
|
|
|
|
|
|
|
|
release := make([]int, len(fields))
|
|
|
|
for i, s := range fields {
|
|
|
|
v, _ := strconv.Atoi(s)
|
|
|
|
release[i] = v
|
|
|
|
}
|
|
|
|
|
|
|
|
var prerelease string
|
|
|
|
if len(parts) > 1 {
|
|
|
|
prerelease = parts[1]
|
|
|
|
}
|
|
|
|
|
|
|
|
return release, prerelease
|
|
|
|
}
|