mirror of
https://github.com/ClusterCockpit/cc-metric-store.git
synced 2025-01-15 00:39:05 +01:00
Merge pull request #7 from ClusterCockpit/new-line-protocol-format
New line protocol format
This commit is contained in:
commit
26528151b1
2
.gitignore
vendored
2
.gitignore
vendored
@ -16,4 +16,4 @@
|
|||||||
# vendor/
|
# vendor/
|
||||||
|
|
||||||
# Project specific ignores
|
# Project specific ignores
|
||||||
/archive
|
/var
|
||||||
|
@ -84,6 +84,8 @@ Example selectors:
|
|||||||
|
|
||||||
### Config file
|
### Config file
|
||||||
|
|
||||||
|
All durations are specified in seconds.
|
||||||
|
|
||||||
- `metrics`: Map of metric-name to objects with the following properties
|
- `metrics`: Map of metric-name to objects with the following properties
|
||||||
- `frequency`: Timestep/Interval/Resolution of this metric (In seconds)
|
- `frequency`: Timestep/Interval/Resolution of this metric (In seconds)
|
||||||
- `aggregation`: Can be `"sum"`, `"avg"` or `null`
|
- `aggregation`: Can be `"sum"`, `"avg"` or `null`
|
||||||
@ -92,9 +94,7 @@ Example selectors:
|
|||||||
- `"avg"` means that values from the child levels are averaged for the parent level
|
- `"avg"` means that values from the child levels are averaged for the parent level
|
||||||
- `scope`: Unused at the moment, should be something like `"node"`, `"socket"` or `"cpu"`
|
- `scope`: Unused at the moment, should be something like `"node"`, `"socket"` or `"cpu"`
|
||||||
- `nats`: Url of NATS.io server (The `updates` channel will be subscribed for metrics)
|
- `nats`: Url of NATS.io server (The `updates` channel will be subscribed for metrics)
|
||||||
- `archive-root`: Directory to be used as archive
|
- `jwt-public-key`: Base64 encoded string, use this to verify requests to the HTTP API
|
||||||
- `restore-last-hours`: After restart, load data from the past *X* hours back to memory
|
|
||||||
- `checkpoint-interval-hours`: Every *X* hours, write currently held data to disk
|
|
||||||
|
|
||||||
### Test the complete setup (excluding ClusterCockpit itself)
|
### Test the complete setup (excluding ClusterCockpit itself)
|
||||||
|
|
||||||
|
19
api.go
19
api.go
@ -1,6 +1,7 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"context"
|
"context"
|
||||||
"crypto/ed25519"
|
"crypto/ed25519"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
@ -15,6 +16,7 @@ import (
|
|||||||
|
|
||||||
"github.com/golang-jwt/jwt/v4"
|
"github.com/golang-jwt/jwt/v4"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/influxdata/line-protocol/v2/lineprotocol"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Example:
|
// Example:
|
||||||
@ -208,6 +210,22 @@ func handlePeek(rw http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func handleWrite(rw http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Method != http.MethodPost {
|
||||||
|
http.Error(rw, "Method Not Allowed", http.StatusMethodNotAllowed)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
reader := bufio.NewReader(r.Body)
|
||||||
|
dec := lineprotocol.NewDecoder(reader)
|
||||||
|
// Unlike the name suggests, handleLine can handle multiple lines
|
||||||
|
if err := handleLine(dec); err != nil {
|
||||||
|
http.Error(rw, err.Error(), http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rw.WriteHeader(http.StatusOK)
|
||||||
|
}
|
||||||
|
|
||||||
func authentication(next http.Handler, publicKey ed25519.PublicKey) http.Handler {
|
func authentication(next http.Handler, publicKey ed25519.PublicKey) http.Handler {
|
||||||
return http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
||||||
authheader := r.Header.Get("Authorization")
|
authheader := r.Header.Get("Authorization")
|
||||||
@ -244,6 +262,7 @@ func StartApiServer(address string, ctx context.Context) error {
|
|||||||
r.HandleFunc("/api/{from:[0-9]+}/{to:[0-9]+}/stats", handleStats)
|
r.HandleFunc("/api/{from:[0-9]+}/{to:[0-9]+}/stats", handleStats)
|
||||||
r.HandleFunc("/api/{to:[0-9]+}/free", handleFree)
|
r.HandleFunc("/api/{to:[0-9]+}/free", handleFree)
|
||||||
r.HandleFunc("/api/{cluster}/peek", handlePeek)
|
r.HandleFunc("/api/{cluster}/peek", handlePeek)
|
||||||
|
r.HandleFunc("/api/write", handleWrite)
|
||||||
|
|
||||||
server := &http.Server{
|
server := &http.Server{
|
||||||
Handler: r,
|
Handler: r,
|
||||||
|
@ -192,6 +192,10 @@ func (l *level) loadFile(cf *CheckpointFile, m *MemoryStore) error {
|
|||||||
func (l *level) fromCheckpoint(dir string, from int64, m *MemoryStore) (int, error) {
|
func (l *level) fromCheckpoint(dir string, from int64, m *MemoryStore) (int, error) {
|
||||||
direntries, err := os.ReadDir(dir)
|
direntries, err := os.ReadDir(dir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
3
go.mod
3
go.mod
@ -3,9 +3,10 @@ module github.com/ClusterCockpit/cc-metric-store
|
|||||||
go 1.16
|
go 1.16
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/golang-jwt/jwt/v4 v4.0.0 // indirect
|
github.com/golang-jwt/jwt/v4 v4.0.0
|
||||||
github.com/golang/protobuf v1.5.2 // indirect
|
github.com/golang/protobuf v1.5.2 // indirect
|
||||||
github.com/gorilla/mux v1.8.0
|
github.com/gorilla/mux v1.8.0
|
||||||
|
github.com/influxdata/line-protocol/v2 v2.2.0
|
||||||
github.com/nats-io/nats-server/v2 v2.2.6 // indirect
|
github.com/nats-io/nats-server/v2 v2.2.6 // indirect
|
||||||
github.com/nats-io/nats.go v1.11.0
|
github.com/nats-io/nats.go v1.11.0
|
||||||
)
|
)
|
||||||
|
19
go.sum
19
go.sum
@ -1,3 +1,7 @@
|
|||||||
|
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
||||||
|
github.com/frankban/quicktest v1.11.0/go.mod h1:K+q6oSqb0W0Ininfk863uOk1lMy69l/P6txr3mVT54s=
|
||||||
|
github.com/frankban/quicktest v1.11.2/go.mod h1:K+q6oSqb0W0Ininfk863uOk1lMy69l/P6txr3mVT54s=
|
||||||
|
github.com/frankban/quicktest v1.13.0/go.mod h1:qLE0fzW0VuyUAJgPU19zByoIr0HtCHN/r/VLSOOIySU=
|
||||||
github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY=
|
github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY=
|
||||||
github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I=
|
github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I=
|
||||||
github.com/golang-jwt/jwt/v4 v4.0.0 h1:RAqyYixv1p7uEnocuy8P1nru5wprCh/MH2BIlW5z5/o=
|
github.com/golang-jwt/jwt/v4 v4.0.0 h1:RAqyYixv1p7uEnocuy8P1nru5wprCh/MH2BIlW5z5/o=
|
||||||
@ -14,11 +18,22 @@ github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiu
|
|||||||
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||||
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
|
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI=
|
github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI=
|
||||||
github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So=
|
github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So=
|
||||||
|
github.com/influxdata/line-protocol-corpus v0.0.0-20210519164801-ca6fa5da0184/go.mod h1:03nmhxzZ7Xk2pdG+lmMd7mHDfeVOYFyhOgwO61qWU98=
|
||||||
|
github.com/influxdata/line-protocol-corpus v0.0.0-20210922080147-aa28ccfb8937/go.mod h1:BKR9c0uHSmRgM/se9JhFHtTT7JTO67X23MtKMHtZcpo=
|
||||||
|
github.com/influxdata/line-protocol/v2 v2.0.0-20210312151457-c52fdecb625a/go.mod h1:6+9Xt5Sq1rWx+glMgxhcg2c0DUaehK+5TDcPZ76GypY=
|
||||||
|
github.com/influxdata/line-protocol/v2 v2.1.0/go.mod h1:QKw43hdUBg3GTk2iC3iyCxksNj7PX9aUSeYOYE/ceHY=
|
||||||
|
github.com/influxdata/line-protocol/v2 v2.2.0 h1:UPmAqE15Hw5zu9E10SYhoXVLWnEJkWnuCbaCiRsA3c0=
|
||||||
|
github.com/influxdata/line-protocol/v2 v2.2.0/go.mod h1:DmB3Cnh+3oxmG6LOBIxce4oaL4CPj3OmMPgvauXh+tM=
|
||||||
github.com/klauspost/compress v1.11.12 h1:famVnQVu7QwryBN4jNseQdUKES71ZAOnB6UQQJPZvqk=
|
github.com/klauspost/compress v1.11.12 h1:famVnQVu7QwryBN4jNseQdUKES71ZAOnB6UQQJPZvqk=
|
||||||
github.com/klauspost/compress v1.11.12/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs=
|
github.com/klauspost/compress v1.11.12/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs=
|
||||||
|
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
|
||||||
|
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
|
||||||
|
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
|
||||||
|
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
|
||||||
github.com/minio/highwayhash v1.0.1 h1:dZ6IIu8Z14VlC0VpfKofAhCy74wu/Qb5gcn52yWoz/0=
|
github.com/minio/highwayhash v1.0.1 h1:dZ6IIu8Z14VlC0VpfKofAhCy74wu/Qb5gcn52yWoz/0=
|
||||||
github.com/minio/highwayhash v1.0.1/go.mod h1:BQskDq+xkJ12lmlUUi7U0M5Swg3EWR+dLTk+kldvVxY=
|
github.com/minio/highwayhash v1.0.1/go.mod h1:BQskDq+xkJ12lmlUUi7U0M5Swg3EWR+dLTk+kldvVxY=
|
||||||
github.com/nats-io/jwt v1.2.2 h1:w3GMTO969dFg+UOKTmmyuu7IGdusK+7Ytlt//OYH/uU=
|
github.com/nats-io/jwt v1.2.2 h1:w3GMTO969dFg+UOKTmmyuu7IGdusK+7Ytlt//OYH/uU=
|
||||||
@ -34,6 +49,7 @@ github.com/nats-io/nkeys v0.3.0 h1:cgM5tL53EvYRU+2YLXIK0G2mJtK12Ft9oeooSZMA2G8=
|
|||||||
github.com/nats-io/nkeys v0.3.0/go.mod h1:gvUNGjVcM2IPr5rCsRsC6Wb3Hr2CQAm08dsxtV6A5y4=
|
github.com/nats-io/nkeys v0.3.0/go.mod h1:gvUNGjVcM2IPr5rCsRsC6Wb3Hr2CQAm08dsxtV6A5y4=
|
||||||
github.com/nats-io/nuid v1.0.1 h1:5iA8DT8V7q8WK2EScv2padNa/rTESc1KdnPw4TC2paw=
|
github.com/nats-io/nuid v1.0.1 h1:5iA8DT8V7q8WK2EScv2padNa/rTESc1KdnPw4TC2paw=
|
||||||
github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c=
|
github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c=
|
||||||
|
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno=
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
||||||
golang.org/x/crypto v0.0.0-20210314154223-e6e6c4f2bb5b h1:wSOdpTq0/eI46Ez/LkDwIsAKA71YP2SRKBODiRWM0as=
|
golang.org/x/crypto v0.0.0-20210314154223-e6e6c4f2bb5b h1:wSOdpTq0/eI46Ez/LkDwIsAKA71YP2SRKBODiRWM0as=
|
||||||
@ -61,3 +77,6 @@ google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2
|
|||||||
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
||||||
google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk=
|
google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk=
|
||||||
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
|
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
186
lineprotocol.go
186
lineprotocol.go
@ -1,18 +1,13 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
|
||||||
"io"
|
|
||||||
"log"
|
"log"
|
||||||
"math"
|
"math"
|
||||||
"net"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
|
||||||
|
|
||||||
|
"github.com/influxdata/line-protocol/v2/lineprotocol"
|
||||||
"github.com/nats-io/nats.go"
|
"github.com/nats-io/nats.go"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -56,185 +51,30 @@ type Metric struct {
|
|||||||
Value Float
|
Value Float
|
||||||
}
|
}
|
||||||
|
|
||||||
// measurement: node or cpu
|
|
||||||
// tags: host, cluster, cpu (cpu only if measurement is cpu)
|
|
||||||
// fields: metrics...
|
|
||||||
// t: timestamp (accuracy: seconds)
|
|
||||||
type Line struct {
|
|
||||||
Measurement string
|
|
||||||
Tags map[string]string
|
|
||||||
Fields []Metric
|
|
||||||
Ts time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse a single line as string.
|
|
||||||
//
|
|
||||||
// There is performance to be gained by implementing a parser
|
|
||||||
// that directly reads from a bufio.Scanner.
|
|
||||||
func ParseLine(rawline string) (*Line, error) {
|
|
||||||
line := &Line{}
|
|
||||||
parts := strings.Fields(rawline)
|
|
||||||
if len(parts) != 3 {
|
|
||||||
return nil, errors.New("line format error")
|
|
||||||
}
|
|
||||||
|
|
||||||
tagsAndMeasurement := strings.Split(parts[0], ",")
|
|
||||||
line.Measurement = tagsAndMeasurement[0]
|
|
||||||
line.Tags = map[string]string{}
|
|
||||||
for i := 1; i < len(tagsAndMeasurement); i++ {
|
|
||||||
pair := strings.Split(tagsAndMeasurement[i], "=")
|
|
||||||
if len(pair) != 2 {
|
|
||||||
return nil, errors.New("line format error")
|
|
||||||
}
|
|
||||||
line.Tags[pair[0]] = pair[1]
|
|
||||||
}
|
|
||||||
|
|
||||||
rawfields := strings.Split(parts[1], ",")
|
|
||||||
line.Fields = []Metric{}
|
|
||||||
for i := 0; i < len(rawfields); i++ {
|
|
||||||
pair := strings.Split(rawfields[i], "=")
|
|
||||||
if len(pair) != 2 {
|
|
||||||
return nil, errors.New("line format error")
|
|
||||||
}
|
|
||||||
field, err := strconv.ParseFloat(pair[1], 64)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
line.Fields = append(line.Fields, Metric{
|
|
||||||
Name: pair[0],
|
|
||||||
Value: Float(field),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
unixTimestamp, err := strconv.ParseInt(parts[2], 10, 64)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
line.Ts = time.Unix(unixTimestamp, 0)
|
|
||||||
return line, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseLines(raw string) ([]*Line, error) {
|
|
||||||
lines := make([]*Line, 0, 1)
|
|
||||||
for _, line := range strings.Split(raw, "\n") {
|
|
||||||
if len(line) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
line, err := ParseLine(line)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
lines = append(lines, line)
|
|
||||||
}
|
|
||||||
return lines, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Listen for connections sending metric data in the line protocol format.
|
|
||||||
//
|
|
||||||
// This is a blocking function, send `true` through the channel argument to shut down the server.
|
|
||||||
// `handleLine` will be called from different go routines for different connections.
|
|
||||||
//
|
|
||||||
func ReceiveTCP(address string, handleLine func(line *Line), done chan bool) error {
|
|
||||||
ln, err := net.Listen("tcp", address)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
handleConnection := func(conn net.Conn, handleLine func(line *Line)) {
|
|
||||||
reader := bufio.NewReader(conn)
|
|
||||||
for {
|
|
||||||
rawline, err := reader.ReadString('\n')
|
|
||||||
if err == io.EOF {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("reading from connection failed: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
line, err := ParseLine(rawline)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("parsing line failed: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
handleLine(line)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
stop := <-done
|
|
||||||
if stop {
|
|
||||||
err := ln.Close()
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("closing listener failed: %s\n", err.Error())
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
for {
|
|
||||||
conn, err := ln.Accept()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
go handleConnection(conn, handleLine)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Connect to a nats server and subscribe to "updates". This is a blocking
|
// Connect to a nats server and subscribe to "updates". This is a blocking
|
||||||
// function. handleLine will be called for each line recieved via nats.
|
// function. handleLine will be called for each line recieved via nats.
|
||||||
// Send `true` through the done channel for gracefull termination.
|
// Send `true` through the done channel for gracefull termination.
|
||||||
func ReceiveNats(address string, handleLine func(line *Line), workers int, ctx context.Context) error {
|
func ReceiveNats(address string, handleLine func(dec *lineprotocol.Decoder) error, workers int, ctx context.Context) error {
|
||||||
nc, err := nats.Connect(nats.DefaultURL)
|
nc, err := nats.Connect(nats.DefaultURL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer nc.Close()
|
defer nc.Close()
|
||||||
|
|
||||||
|
var wg sync.WaitGroup
|
||||||
var sub *nats.Subscription
|
var sub *nats.Subscription
|
||||||
|
|
||||||
if workers < 2 {
|
msgs := make(chan *nats.Msg, workers*2)
|
||||||
sub, err = nc.Subscribe("updates", func(m *nats.Msg) {
|
|
||||||
lines, err := ParseLines(string(m.Data))
|
|
||||||
if err != nil {
|
|
||||||
log.Println(err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, line := range lines {
|
if workers > 1 {
|
||||||
handleLine(line)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Printf("NATS subscription to 'updates' on '%s' established\n", address)
|
|
||||||
|
|
||||||
<-ctx.Done()
|
|
||||||
err = sub.Unsubscribe()
|
|
||||||
} else {
|
|
||||||
msgs := make(chan *nats.Msg, 16)
|
|
||||||
var wg sync.WaitGroup
|
|
||||||
wg.Add(workers)
|
wg.Add(workers)
|
||||||
|
|
||||||
for i := 0; i < workers; i++ {
|
for i := 0; i < workers; i++ {
|
||||||
go func() {
|
go func() {
|
||||||
for m := range msgs {
|
for m := range msgs {
|
||||||
lines, err := ParseLines(string(m.Data))
|
dec := lineprotocol.NewDecoderWithBytes(m.Data)
|
||||||
if err != nil {
|
if err := handleLine(dec); err != nil {
|
||||||
log.Println(err.Error())
|
log.Printf("error: %s\n", err.Error())
|
||||||
}
|
|
||||||
|
|
||||||
for _, line := range lines {
|
|
||||||
handleLine(line)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,6 +85,15 @@ func ReceiveNats(address string, handleLine func(line *Line), workers int, ctx c
|
|||||||
sub, err = nc.Subscribe("updates", func(m *nats.Msg) {
|
sub, err = nc.Subscribe("updates", func(m *nats.Msg) {
|
||||||
msgs <- m
|
msgs <- m
|
||||||
})
|
})
|
||||||
|
} else {
|
||||||
|
sub, err = nc.Subscribe("updates", func(m *nats.Msg) {
|
||||||
|
dec := lineprotocol.NewDecoderWithBytes(m.Data)
|
||||||
|
if err := handleLine(dec); err != nil {
|
||||||
|
log.Printf("error: %s\n", err.Error())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -255,7 +104,6 @@ func ReceiveNats(address string, handleLine func(line *Line), workers int, ctx c
|
|||||||
err = sub.Unsubscribe()
|
err = sub.Unsubscribe()
|
||||||
close(msgs)
|
close(msgs)
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -1,89 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
var raw = "node,host=lousxps,cluster=test mem_used=4692.252,proc_total=1083,load_five=0.91,cpu_user=1.424336e+06,cpu_guest_nice=0,cpu_guest=0,mem_available=9829.848,mem_slab=514.796,mem_free=4537.956,proc_run=2,cpu_idle=2.1589764e+07,swap_total=0,mem_cached=6368.5,swap_free=0,load_fifteen=0.93,cpu_nice=196,cpu_softirq=41456,mem_buffers=489.992,mem_total=16088.7,load_one=0.84,cpu_system=517223,cpu_iowait=8994,cpu_steal=0,cpu_irq=113265,mem_sreclaimable=362.452 1629356936\n"
|
|
||||||
var expectedMeasurement = `node`
|
|
||||||
var expectedTags = map[string]string{
|
|
||||||
"host": "lousxps",
|
|
||||||
"cluster": "test",
|
|
||||||
}
|
|
||||||
var expectedFields = []Metric{
|
|
||||||
{"mem_used", 4692.252},
|
|
||||||
{"proc_total", 1083},
|
|
||||||
{"load_five", 0.91},
|
|
||||||
{"cpu_user", 1.424336e+06},
|
|
||||||
{"cpu_guest_nice", 0},
|
|
||||||
{"cpu_guest", 0},
|
|
||||||
{"mem_available", 9829.848},
|
|
||||||
{"mem_slab", 514.796},
|
|
||||||
{"mem_free", 4537.956},
|
|
||||||
{"proc_run", 2},
|
|
||||||
{"cpu_idle", 2.1589764e+07},
|
|
||||||
{"swap_total", 0},
|
|
||||||
{"mem_cached", 6368.5},
|
|
||||||
{"swap_free", 0},
|
|
||||||
{"load_fifteen", 0.93},
|
|
||||||
{"cpu_nice", 196},
|
|
||||||
{"cpu_softirq", 41456},
|
|
||||||
{"mem_buffers", 489.992},
|
|
||||||
{"mem_total", 16088.7},
|
|
||||||
{"load_one", 0.84},
|
|
||||||
{"cpu_system", 517223},
|
|
||||||
{"cpu_iowait", 8994},
|
|
||||||
{"cpu_steal", 0},
|
|
||||||
{"cpu_irq", 113265},
|
|
||||||
{"mem_sreclaimable", 362.452},
|
|
||||||
}
|
|
||||||
var expectedTimestamp int64 = 1629356936
|
|
||||||
|
|
||||||
func TestParseLine(t *testing.T) {
|
|
||||||
line, err := ParseLine(raw)
|
|
||||||
if err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if line.Measurement != expectedMeasurement {
|
|
||||||
t.Error("measurement not as expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
if line.Ts.Unix() != int64(expectedTimestamp) {
|
|
||||||
t.Error("timestamp not as expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
if !reflect.DeepEqual(line.Tags, expectedTags) {
|
|
||||||
t.Error("tags not as expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
if !reflect.DeepEqual(line.Fields, expectedFields) {
|
|
||||||
t.Error("fields not as expected")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func BenchmarkParseLine(b *testing.B) {
|
|
||||||
b.StopTimer()
|
|
||||||
lines := strings.Repeat(raw, b.N)
|
|
||||||
scanner := bufio.NewScanner(strings.NewReader(lines))
|
|
||||||
scanner.Split(bufio.ScanLines)
|
|
||||||
|
|
||||||
b.StartTimer()
|
|
||||||
for i := 0; i < b.N; i++ {
|
|
||||||
ok := scanner.Scan()
|
|
||||||
if !ok {
|
|
||||||
b.Error("woops")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
line := scanner.Text()
|
|
||||||
_, err := ParseLine(line)
|
|
||||||
if err != nil {
|
|
||||||
b.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
25
memstore.go
25
memstore.go
@ -16,7 +16,9 @@ const (
|
|||||||
// So that we can reuse allocations
|
// So that we can reuse allocations
|
||||||
var bufferPool sync.Pool = sync.Pool{
|
var bufferPool sync.Pool = sync.Pool{
|
||||||
New: func() interface{} {
|
New: func() interface{} {
|
||||||
return make([]Float, 0, BUFFER_CAP)
|
return &buffer{
|
||||||
|
data: make([]Float, 0, BUFFER_CAP),
|
||||||
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -37,13 +39,12 @@ type buffer struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newBuffer(ts, freq int64) *buffer {
|
func newBuffer(ts, freq int64) *buffer {
|
||||||
return &buffer{
|
b := bufferPool.Get().(*buffer)
|
||||||
frequency: freq,
|
b.frequency = freq
|
||||||
start: ts,
|
b.start = ts
|
||||||
data: bufferPool.Get().([]Float)[:0],
|
b.prev = nil
|
||||||
prev: nil,
|
b.next = nil
|
||||||
next: nil,
|
return b
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If a new buffer was created, the new head is returnd.
|
// If a new buffer was created, the new head is returnd.
|
||||||
@ -137,10 +138,11 @@ func (b *buffer) free(t int64) (int, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
n += 1
|
n += 1
|
||||||
bufferPool.Put(b.data)
|
b.frequency = 0
|
||||||
b.data = nil
|
b.start = 0
|
||||||
b.next = nil
|
b.next = nil
|
||||||
b.prev = nil
|
b.prev = nil
|
||||||
|
bufferPool.Put(b)
|
||||||
b = prev
|
b = prev
|
||||||
}
|
}
|
||||||
return n, nil
|
return n, nil
|
||||||
@ -283,7 +285,8 @@ func (m *MemoryStore) Write(selector []string, ts int64, metrics []Metric) error
|
|||||||
for _, metric := range metrics {
|
for _, metric := range metrics {
|
||||||
minfo, ok := m.metrics[metric.Name]
|
minfo, ok := m.metrics[metric.Name]
|
||||||
if !ok {
|
if !ok {
|
||||||
return errors.New("Unknown metric: " + metric.Name)
|
// return errors.New("Unknown metric: " + metric.Name)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
b := l.metrics[minfo.offset]
|
b := l.metrics[minfo.offset]
|
||||||
|
@ -10,6 +10,8 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/influxdata/line-protocol/v2/lineprotocol"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MetricConfig struct {
|
type MetricConfig struct {
|
||||||
@ -50,30 +52,80 @@ func loadConfiguration(file string) Config {
|
|||||||
return config
|
return config
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleLine(line *Line) {
|
func handleLine(dec *lineprotocol.Decoder) error {
|
||||||
cluster, ok := line.Tags["cluster"]
|
for dec.Next() {
|
||||||
if !ok {
|
measurement, err := dec.Measurement()
|
||||||
log.Println("'cluster' tag missing")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
host, ok := line.Tags["host"]
|
|
||||||
if !ok {
|
|
||||||
log.Println("'host' tag missing")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
selector := []string{cluster, host}
|
|
||||||
if id, ok := line.Tags[line.Measurement]; ok {
|
|
||||||
selector = append(selector, line.Measurement+id)
|
|
||||||
}
|
|
||||||
|
|
||||||
ts := line.Ts.Unix()
|
|
||||||
// log.Printf("ts=%d, tags=%v\n", ts, selector)
|
|
||||||
err := memoryStore.Write(selector, ts, line.Fields)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("error: %s\n", err.Error())
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var cluster, host, typeName, typeId string
|
||||||
|
for {
|
||||||
|
key, val, err := dec.NextTag()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if key == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
switch string(key) {
|
||||||
|
case "cluster":
|
||||||
|
cluster = string(val)
|
||||||
|
case "hostname":
|
||||||
|
host = string(val)
|
||||||
|
case "type":
|
||||||
|
typeName = string(val)
|
||||||
|
case "type-id":
|
||||||
|
typeId = string(val)
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("unkown tag: '%s' (value: '%s')", string(key), string(val))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
selector := make([]string, 0, 3)
|
||||||
|
selector = append(selector, cluster)
|
||||||
|
selector = append(selector, host)
|
||||||
|
if len(typeId) > 0 {
|
||||||
|
selector = append(selector, typeName+typeId)
|
||||||
|
}
|
||||||
|
|
||||||
|
var value Float
|
||||||
|
for {
|
||||||
|
key, val, err := dec.NextField()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if key == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if string(key) != "value" {
|
||||||
|
return fmt.Errorf("unkown field: '%s' (value: %#v)", string(key), val)
|
||||||
|
}
|
||||||
|
|
||||||
|
if val.Kind() == lineprotocol.Float {
|
||||||
|
value = Float(val.FloatV())
|
||||||
|
} else if val.Kind() == lineprotocol.Int {
|
||||||
|
value = Float(val.IntV())
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("unsupported value type in message: %s", val.Kind().String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
t, err := dec.Time(lineprotocol.Second, time.Now())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := memoryStore.Write(selector, t.Unix(), []Metric{
|
||||||
|
{Name: string(measurement), Value: value},
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func intervals(wg *sync.WaitGroup, ctx context.Context) {
|
func intervals(wg *sync.WaitGroup, ctx context.Context) {
|
||||||
@ -159,7 +211,7 @@ func main() {
|
|||||||
conf = loadConfiguration("config.json")
|
conf = loadConfiguration("config.json")
|
||||||
memoryStore = NewMemoryStore(conf.Metrics)
|
memoryStore = NewMemoryStore(conf.Metrics)
|
||||||
|
|
||||||
restoreFrom := startupTime.Add(-time.Duration(conf.Checkpoints.Restore))
|
restoreFrom := startupTime.Add(-time.Duration(conf.Checkpoints.Restore) * time.Second)
|
||||||
files, err := memoryStore.FromCheckpoint(conf.Checkpoints.RootDir, restoreFrom.Unix())
|
files, err := memoryStore.FromCheckpoint(conf.Checkpoints.RootDir, restoreFrom.Unix())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Loading checkpoints failed: %s\n", err.Error())
|
log.Fatalf("Loading checkpoints failed: %s\n", err.Error())
|
||||||
|
Loading…
Reference in New Issue
Block a user