mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-24 07:28:27 +00:00
Reorganize, add aggregation
This commit is contained in:
parent
f3bf4cf722
commit
3e1a562466
109
cmd/aggregate/main.go
Normal file
109
cmd/aggregate/main.go
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
_ "github.com/lib/pq"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var dbConn = getEnvDefault("UR_DB_URL", "postgres://user:password@localhost/ur?sslmode=disable")
|
||||||
|
|
||||||
|
func getEnvDefault(key, def string) string {
|
||||||
|
if val := os.Getenv(key); val != "" {
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
log.SetFlags(log.Ltime | log.Ldate)
|
||||||
|
log.SetOutput(os.Stdout)
|
||||||
|
|
||||||
|
db, err := sql.Open("postgres", dbConn)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalln("database:", err)
|
||||||
|
}
|
||||||
|
err = setupDB(db)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalln("database:", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
runAggregation(db)
|
||||||
|
// Sleep until one minute past next midnight
|
||||||
|
sleepUntilNext(24*time.Hour, 1*time.Minute)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func runAggregation(db *sql.DB) {
|
||||||
|
since := maxIndexedDay(db)
|
||||||
|
log.Println("Aggregating data since", since)
|
||||||
|
rows, err := aggregate(db, since)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalln("aggregate:", err)
|
||||||
|
}
|
||||||
|
log.Println("Inserted", rows, "rows")
|
||||||
|
}
|
||||||
|
|
||||||
|
func sleepUntilNext(intv, margin time.Duration) {
|
||||||
|
now := time.Now().UTC()
|
||||||
|
next := now.Truncate(intv).Add(intv).Add(margin)
|
||||||
|
log.Println("Sleeping until", next)
|
||||||
|
time.Sleep(next.Sub(now))
|
||||||
|
}
|
||||||
|
|
||||||
|
func setupDB(db *sql.DB) error {
|
||||||
|
_, err := db.Exec(`CREATE TABLE IF NOT EXISTS VersionSummary (
|
||||||
|
Day TIMESTAMP NOT NULL,
|
||||||
|
Version VARCHAR(8) NOT NULL,
|
||||||
|
Count INTEGER NOT NULL
|
||||||
|
)`)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
row := db.QueryRow(`SELECT 'UniqueDayVersionIndex'::regclass`)
|
||||||
|
if err := row.Scan(nil); err != nil {
|
||||||
|
_, err = db.Exec(`CREATE UNIQUE INDEX UniqueDayVersionIndex ON VersionSummary (Day, Version)`)
|
||||||
|
}
|
||||||
|
|
||||||
|
row = db.QueryRow(`SELECT 'DayIndex'::regclass`)
|
||||||
|
if err := row.Scan(nil); err != nil {
|
||||||
|
_, err = db.Exec(`CREATE INDEX DayIndex ON VerionSummary (Day)`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func maxIndexedDay(db *sql.DB) time.Time {
|
||||||
|
var t time.Time
|
||||||
|
row := db.QueryRow("SELECT MAX(Day) FROM VersionSummary")
|
||||||
|
err := row.Scan(&t)
|
||||||
|
if err != nil {
|
||||||
|
return time.Time{}
|
||||||
|
}
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
func aggregate(db *sql.DB, since time.Time) (int64, error) {
|
||||||
|
res, err := db.Exec(`INSERT INTO VersionSummary (
|
||||||
|
SELECT
|
||||||
|
DATE_TRUNC('day', Received) AS Day,
|
||||||
|
SUBSTRING(Version FROM '^v\d.\d+') AS Ver,
|
||||||
|
COUNT(*) AS Count
|
||||||
|
FROM Reports
|
||||||
|
WHERE
|
||||||
|
DATE_TRUNC('day', Received) > $1
|
||||||
|
AND DATE_TRUNC('day', Received) < DATE_TRUNC('day', NOW() - '1 day'::INTERVAL)
|
||||||
|
AND Version like 'v0.%'
|
||||||
|
GROUP BY Day, Ver
|
||||||
|
);
|
||||||
|
`, since)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.RowsAffected()
|
||||||
|
}
|
@ -13,6 +13,7 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"regexp"
|
"regexp"
|
||||||
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
@ -26,6 +27,7 @@ var (
|
|||||||
dbConn = getEnvDefault("UR_DB_URL", "postgres://user:password@localhost/ur?sslmode=disable")
|
dbConn = getEnvDefault("UR_DB_URL", "postgres://user:password@localhost/ur?sslmode=disable")
|
||||||
listenAddr = getEnvDefault("UR_LISTEN", "0.0.0.0:8443")
|
listenAddr = getEnvDefault("UR_LISTEN", "0.0.0.0:8443")
|
||||||
tpl *template.Template
|
tpl *template.Template
|
||||||
|
compilerRe = regexp.MustCompile(`\(([A-Za-z0-9()., -]+) [\w-]+ \w+\) ([\w@-]+)`)
|
||||||
)
|
)
|
||||||
|
|
||||||
var funcs = map[string]interface{}{
|
var funcs = map[string]interface{}{
|
||||||
@ -106,7 +108,7 @@ func setupDB(db *sql.DB) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func insertReport(db *sql.DB, r report) error {
|
func insertReport(db *sql.DB, r report) error {
|
||||||
_, err := db.Exec(`INSERT INTO Reports VALUES (now(), $1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14)`,
|
_, err := db.Exec(`INSERT INTO Reports VALUES (TIMEZONE('UTC', NOW()), $1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14)`,
|
||||||
r.UniqueID, r.Version, r.LongVersion, r.Platform, r.NumFolders,
|
r.UniqueID, r.Version, r.LongVersion, r.Platform, r.NumFolders,
|
||||||
r.NumDevices, r.TotFiles, r.FolderMaxFiles, r.TotMiB, r.FolderMaxMiB,
|
r.NumDevices, r.TotFiles, r.FolderMaxFiles, r.TotMiB, r.FolderMaxMiB,
|
||||||
r.MemoryUsageMiB, r.SHA256Perf, r.MemorySize, r.Date)
|
r.MemoryUsageMiB, r.SHA256Perf, r.MemorySize, r.Date)
|
||||||
@ -176,6 +178,7 @@ func main() {
|
|||||||
|
|
||||||
http.HandleFunc("/", withDB(db, rootHandler))
|
http.HandleFunc("/", withDB(db, rootHandler))
|
||||||
http.HandleFunc("/newdata", withDB(db, newDataHandler))
|
http.HandleFunc("/newdata", withDB(db, newDataHandler))
|
||||||
|
http.HandleFunc("/summary.json", withDB(db, summaryHandler))
|
||||||
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("static"))))
|
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("static"))))
|
||||||
|
|
||||||
err = srv.Serve(listener)
|
err = srv.Serve(listener)
|
||||||
@ -246,6 +249,25 @@ func newDataHandler(db *sql.DB, w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func summaryHandler(db *sql.DB, w http.ResponseWriter, r *http.Request) {
|
||||||
|
s, err := getSummary(db)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("summaryHandler:", err)
|
||||||
|
http.Error(w, "Database Error", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bs, err := s.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
log.Println("summaryHandler:", err)
|
||||||
|
http.Error(w, "JSON Encode Error", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.Write(bs)
|
||||||
|
}
|
||||||
|
|
||||||
type category struct {
|
type category struct {
|
||||||
Values [4]float64
|
Values [4]float64
|
||||||
Key string
|
Key string
|
||||||
@ -268,6 +290,8 @@ func getReport(db *sql.DB) map[string]interface{} {
|
|||||||
var memoryUsage []int
|
var memoryUsage []int
|
||||||
var sha256Perf []float64
|
var sha256Perf []float64
|
||||||
var memorySize []int
|
var memorySize []int
|
||||||
|
var compilers []string
|
||||||
|
var builders []string
|
||||||
|
|
||||||
rows, err := db.Query(`SELECT * FROM Reports WHERE Received > now() - '1 day'::INTERVAL`)
|
rows, err := db.Query(`SELECT * FROM Reports WHERE Received > now() - '1 day'::INTERVAL`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -294,6 +318,10 @@ func getReport(db *sql.DB) map[string]interface{} {
|
|||||||
platforms = append(platforms, rep.Platform)
|
platforms = append(platforms, rep.Platform)
|
||||||
ps := strings.Split(rep.Platform, "-")
|
ps := strings.Split(rep.Platform, "-")
|
||||||
oses = append(oses, ps[0])
|
oses = append(oses, ps[0])
|
||||||
|
if m := compilerRe.FindStringSubmatch(rep.LongVersion); len(m) == 3 {
|
||||||
|
compilers = append(compilers, m[1])
|
||||||
|
builders = append(builders, m[2])
|
||||||
|
}
|
||||||
if rep.NumFolders > 0 {
|
if rep.NumFolders > 0 {
|
||||||
numFolders = append(numFolders, rep.NumFolders)
|
numFolders = append(numFolders, rep.NumFolders)
|
||||||
}
|
}
|
||||||
@ -385,6 +413,8 @@ func getReport(db *sql.DB) map[string]interface{} {
|
|||||||
r["versions"] = analyticsFor(versions, 10)
|
r["versions"] = analyticsFor(versions, 10)
|
||||||
r["platforms"] = analyticsFor(platforms, 0)
|
r["platforms"] = analyticsFor(platforms, 0)
|
||||||
r["os"] = analyticsFor(oses, 0)
|
r["os"] = analyticsFor(oses, 0)
|
||||||
|
r["compilers"] = analyticsFor(compilers, 12)
|
||||||
|
r["builders"] = analyticsFor(builders, 12)
|
||||||
|
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
@ -426,3 +456,104 @@ func transformVersion(v string) string {
|
|||||||
|
|
||||||
return v
|
return v
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type summary struct {
|
||||||
|
versions map[string]int // version string to count index
|
||||||
|
rows map[string][]int // date to list of counts
|
||||||
|
}
|
||||||
|
|
||||||
|
func newSummary() summary {
|
||||||
|
return summary{
|
||||||
|
versions: make(map[string]int),
|
||||||
|
rows: make(map[string][]int),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *summary) setCount(date, version string, count int) {
|
||||||
|
idx, ok := s.versions[version]
|
||||||
|
if !ok {
|
||||||
|
idx = len(s.versions)
|
||||||
|
s.versions[version] = idx
|
||||||
|
}
|
||||||
|
|
||||||
|
row := s.rows[date]
|
||||||
|
if len(row) <= idx {
|
||||||
|
old := row
|
||||||
|
row = make([]int, idx+1)
|
||||||
|
copy(row, old)
|
||||||
|
s.rows[date] = row
|
||||||
|
}
|
||||||
|
|
||||||
|
row[idx] = count
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *summary) MarshalJSON() ([]byte, error) {
|
||||||
|
var versions []string
|
||||||
|
for v := range s.versions {
|
||||||
|
versions = append(versions, v)
|
||||||
|
}
|
||||||
|
sort.Strings(versions)
|
||||||
|
|
||||||
|
headerRow := []interface{}{"Day"}
|
||||||
|
for _, v := range versions {
|
||||||
|
headerRow = append(headerRow, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
var table [][]interface{}
|
||||||
|
table = append(table, headerRow)
|
||||||
|
|
||||||
|
var dates []string
|
||||||
|
for k := range s.rows {
|
||||||
|
dates = append(dates, k)
|
||||||
|
}
|
||||||
|
sort.Strings(dates)
|
||||||
|
|
||||||
|
for _, date := range dates {
|
||||||
|
row := []interface{}{date}
|
||||||
|
for _, ver := range versions {
|
||||||
|
idx := s.versions[ver]
|
||||||
|
if len(s.rows[date]) > idx {
|
||||||
|
row = append(row, s.rows[date][idx])
|
||||||
|
} else {
|
||||||
|
row = append(row, 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
table = append(table, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
return json.Marshal(table)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getSummary(db *sql.DB) (summary, error) {
|
||||||
|
s := newSummary()
|
||||||
|
|
||||||
|
rows, err := db.Query(`SELECT Day, Version, Count FROM VersionSummary;`)
|
||||||
|
if err != nil {
|
||||||
|
return summary{}, err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var day time.Time
|
||||||
|
var ver string
|
||||||
|
var num int
|
||||||
|
err := rows.Scan(&day, &ver, &num)
|
||||||
|
if err != nil {
|
||||||
|
return summary{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if ver == "v0.0" {
|
||||||
|
// ?
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// SUPER UGLY HACK to avoid having to do sorting properly
|
||||||
|
if len(ver) == 4 { // v0.x
|
||||||
|
ver = ver[:3] + "0" + ver[3:] // now v0.0x
|
||||||
|
}
|
||||||
|
|
||||||
|
s.setCount(day.Format("2006-01-02"), ver, num)
|
||||||
|
}
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user