mirror of
https://github.com/ClusterCockpit/cc-metric-collector.git
synced 2024-11-10 04:27:25 +01:00
Run collectors and sink periodically
This commit is contained in:
parent
627d268d0b
commit
d49fda0820
314
clusterdaemon.go
314
clusterdaemon.go
@ -4,13 +4,15 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
//"bytes"
|
||||||
// "context"
|
// "context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sort"
|
//"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
"errors"
|
||||||
|
protocol "github.com/influxdata/line-protocol"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GlobalConfig struct {
|
type GlobalConfig struct {
|
||||||
@ -41,106 +43,58 @@ type GlobalConfig struct {
|
|||||||
type CollectorConfig struct {
|
type CollectorConfig struct {
|
||||||
Command string `json:"command"`
|
Command string `json:"command"`
|
||||||
Args string `json:"arguments"`
|
Args string `json:"arguments"`
|
||||||
|
Provides []string `json:"provides"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type InternalCollectorConfig struct {
|
||||||
|
Config CollectorConfig
|
||||||
|
Location string
|
||||||
|
LastRun time.Time
|
||||||
|
encoder *protocol.Encoder
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Load global configuration from JSON file
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
func LoadGlobalConfiguration(file string, config *GlobalConfig) error {
|
func LoadGlobalConfiguration(file string, config *GlobalConfig) error {
|
||||||
configFile, err := os.Open(file)
|
configFile, err := os.Open(file)
|
||||||
defer configFile.Close()
|
defer configFile.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err.Error())
|
return err
|
||||||
}
|
}
|
||||||
jsonParser := json.NewDecoder(configFile)
|
jsonParser := json.NewDecoder(configFile)
|
||||||
jsonParser.Decode(config)
|
jsonParser.Decode(config)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Load collector configuration from JSON file
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
func LoadCollectorConfiguration(file string, config *CollectorConfig) error {
|
func LoadCollectorConfiguration(file string, config *CollectorConfig) error {
|
||||||
configFile, err := os.Open(file)
|
configFile, err := os.Open(file)
|
||||||
defer configFile.Close()
|
defer configFile.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err.Error())
|
return err
|
||||||
}
|
}
|
||||||
jsonParser := json.NewDecoder(configFile)
|
jsonParser := json.NewDecoder(configFile)
|
||||||
jsonParser.Decode(config)
|
jsonParser.Decode(config)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func SortStringStringMap(input map[string]string) []string {
|
|
||||||
keys := make([]string, 0, len(input))
|
|
||||||
output := make([]string, len(input))
|
|
||||||
for k := range input {
|
|
||||||
keys = append(keys, k)
|
|
||||||
}
|
|
||||||
sort.Strings(keys)
|
|
||||||
|
|
||||||
for i, k := range keys {
|
|
||||||
var s strings.Builder
|
|
||||||
fmt.Fprintf(&s, "%s=%s", k, string(input[k]))
|
|
||||||
output[i] = s.String()
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
func SortStringInterfaceMap(input map[string]interface{}) []string {
|
|
||||||
keys := make([]string, 0, len(input))
|
|
||||||
output := make([]string, len(input))
|
|
||||||
for k := range input {
|
|
||||||
keys = append(keys, k)
|
|
||||||
}
|
|
||||||
sort.Strings(keys)
|
|
||||||
|
|
||||||
for i, k := range keys {
|
|
||||||
var s strings.Builder
|
|
||||||
fmt.Fprintf(&s, "%s=%v", k, input[k])
|
|
||||||
output[i] = s.String()
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
func CreatePoint(metricname string, tags map[string]string, fields map[string]interface{}, timestamp int64) string {
|
|
||||||
var s strings.Builder
|
|
||||||
taglist := SortStringStringMap(tags)
|
|
||||||
fieldlist := SortStringInterfaceMap(fields)
|
|
||||||
|
|
||||||
if len(taglist) > 0 {
|
|
||||||
fmt.Fprintf(&s, "%s,%s %s %d", metricname, strings.Join(taglist, ","), strings.Join(fieldlist, ","), timestamp)
|
|
||||||
} else {
|
|
||||||
fmt.Fprintf(&s, "%s %s %d", metricname, strings.Join(fieldlist, ","), timestamp)
|
|
||||||
}
|
|
||||||
return s.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
func run_cmd(cmd string, cmd_opts string) string {
|
|
||||||
//ctx, cancel := context.WithTimeout(context.Background(), 100*time.Second)
|
|
||||||
//defer cancel()
|
|
||||||
//command := exec.CommandContext(ctx, cmd, strings.Join(cmd_opts, " "))
|
|
||||||
command := exec.Command(cmd, cmd_opts)
|
|
||||||
command.Wait()
|
|
||||||
//select {
|
|
||||||
// case <-time.After(101 * time.Second):
|
|
||||||
// fmt.Println("overslept")
|
|
||||||
// case <-ctx.Done():
|
|
||||||
// fmt.Println(ctx.Err()) // prints "context deadline exceeded"
|
|
||||||
//}
|
|
||||||
|
|
||||||
stdout, err := command.Output()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println(err.Error())
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return (string(stdout))
|
|
||||||
}
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Load collector configurations
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
func GetSingleCollector(folders *[]string) filepath.WalkFunc {
|
func GetSingleCollector(folders *[]string) filepath.WalkFunc {
|
||||||
return func(path string, info os.FileInfo, err error) error {
|
return func(path string, info os.FileInfo, err error) error {
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
if info.IsDir() {
|
if info.IsDir() {
|
||||||
configfile := filepath.Join(path, "config.json")
|
configfile := filepath.Join(path, "config.json")
|
||||||
if _, err := os.Stat(configfile); err == nil {
|
if _, err := os.Stat(configfile); err == nil {
|
||||||
*folders = append(*folders, path)
|
// TODO: Validate config?
|
||||||
|
p, err := filepath.Abs(path)
|
||||||
|
if err == nil {
|
||||||
|
*folders = append(*folders, p)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@ -148,14 +102,176 @@ func GetSingleCollector(folders *[]string) filepath.WalkFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func GetCollectorFolders(root string, folders *[]string) error {
|
func GetCollectorFolders(root string, folders *[]string) error {
|
||||||
|
|
||||||
err := filepath.Walk(root, GetSingleCollector(folders))
|
err := filepath.Walk(root, GetSingleCollector(folders))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
err = errors.New("Cannot get collectors")
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Setup all collectors
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
func SetupCollectors(config GlobalConfig) ([]InternalCollectorConfig, error) {
|
||||||
|
var folders []string
|
||||||
|
var outconfig []InternalCollectorConfig
|
||||||
|
//encoder := protocol.NewEncoder(buf)
|
||||||
|
//encoder.SetMaxLineBytes(1024)
|
||||||
|
GetCollectorFolders(config.CollectorPath, &folders)
|
||||||
|
for _, path := range folders {
|
||||||
|
var col_config InternalCollectorConfig
|
||||||
|
LoadCollectorConfiguration(filepath.Join(path, "config.json"), &col_config.Config)
|
||||||
|
col_config.LastRun = time.Now()
|
||||||
|
col_config.Location = path
|
||||||
|
//buf := &bytes.Buffer{}
|
||||||
|
//col_config.Encoder := protocol.NewEncoder(buf)
|
||||||
|
//col_config.Encoder.SetMaxLineBytes(1024)
|
||||||
|
outconfig = append(outconfig, col_config)
|
||||||
|
}
|
||||||
|
return outconfig, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Run collector
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
func RunCollector(config InternalCollectorConfig) ([]string, error) {
|
||||||
|
var results []string
|
||||||
|
var err error
|
||||||
|
cmd := config.Config.Command
|
||||||
|
|
||||||
|
if _, err = os.Stat(cmd); err != nil {
|
||||||
|
//fmt.Println(err.Error())
|
||||||
|
if ! strings.HasPrefix(cmd, "/") {
|
||||||
|
cmd = filepath.Join(config.Location, config.Config.Command)
|
||||||
|
if _, err = os.Stat(cmd); err != nil {
|
||||||
|
//fmt.Println(err.Error())
|
||||||
|
cmd, err = exec.LookPath(config.Config.Command)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err.Error())
|
||||||
|
return results, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Add timeout
|
||||||
|
|
||||||
|
command := exec.Command(cmd, config.Config.Args)
|
||||||
|
command.Dir = config.Location
|
||||||
|
command.Wait()
|
||||||
|
stdout, err := command.Output()
|
||||||
|
if err != nil {
|
||||||
|
//log.error(err.Error())
|
||||||
|
fmt.Println(err.Error())
|
||||||
|
return results, err
|
||||||
|
}
|
||||||
|
|
||||||
|
lines := strings.Split(string(stdout), "\n")
|
||||||
|
|
||||||
|
for _, l := range lines {
|
||||||
|
if strings.HasPrefix(l, "#") { continue }
|
||||||
|
results = append(results, l)
|
||||||
|
}
|
||||||
|
return results, err
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Setup sink
|
||||||
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
func SetupSink(config GlobalConfig) chan string {
|
||||||
|
|
||||||
|
c := make(chan string, 300)
|
||||||
|
|
||||||
|
// TODO: Setup something for sending? Establish HTTP connection?
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
func RunSink(config GlobalConfig, queue* chan string) (*time.Ticker, chan bool) {
|
||||||
|
var interval time.Duration
|
||||||
|
|
||||||
|
interval = time.Duration(config.Report.Interval) * time.Second
|
||||||
|
ticker := time.NewTicker(interval)
|
||||||
|
done := make(chan bool)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <- done:
|
||||||
|
return
|
||||||
|
case t := <-ticker.C:
|
||||||
|
fmt.Println("SinkTick at", t)
|
||||||
|
empty := false
|
||||||
|
var batch []string
|
||||||
|
for empty == false {
|
||||||
|
select {
|
||||||
|
case metric := <- *queue:
|
||||||
|
fmt.Println(metric)
|
||||||
|
batch = append(batch, metric)
|
||||||
|
default:
|
||||||
|
// No metric available, wait for the next iteration
|
||||||
|
empty = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, m := range batch {
|
||||||
|
fmt.Println(m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
return ticker, done
|
||||||
|
}
|
||||||
|
|
||||||
|
func CloseSink(config GlobalConfig, queue *chan string, ticker *time.Ticker, done chan bool) {
|
||||||
|
ticker.Stop()
|
||||||
|
done <- true
|
||||||
|
close(*queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func MainLoop(config GlobalConfig, sink *chan string) (*time.Ticker, chan bool) {
|
||||||
|
var intConfig []InternalCollectorConfig
|
||||||
|
intConfig, err := SetupCollectors(config)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
var interval time.Duration
|
||||||
|
|
||||||
|
interval = time.Duration(config.Schedule.Node.Frequency) * time.Second
|
||||||
|
|
||||||
|
ticker := time.NewTicker(time.Second)
|
||||||
|
done := make(chan bool)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
return
|
||||||
|
case t := <-ticker.C:
|
||||||
|
fmt.Println("CollectorTick at", t)
|
||||||
|
unix := time.Now()
|
||||||
|
for i, _ := range intConfig {
|
||||||
|
if time.Duration(unix.Sub(intConfig[i].LastRun)) > interval {
|
||||||
|
res, err := RunCollector(intConfig[i])
|
||||||
|
if err != nil {
|
||||||
|
//log.error("Collector failed: ", err.Error())
|
||||||
|
} else {
|
||||||
|
//TODO: parse and skip in case of error, encode to []string
|
||||||
|
for _, r := range res {
|
||||||
|
if len(r) > 0 {
|
||||||
|
*sink <- r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
intConfig[i].LastRun = time.Now()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
return ticker, done
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
// fmt.Println("Hello")
|
// fmt.Println("Hello")
|
||||||
// cmd_opts := []string{"la","le","lu"}
|
// cmd_opts := []string{"la","le","lu"}
|
||||||
@ -172,25 +288,33 @@ func main() {
|
|||||||
var config GlobalConfig
|
var config GlobalConfig
|
||||||
LoadGlobalConfiguration("config.json", &config)
|
LoadGlobalConfiguration("config.json", &config)
|
||||||
|
|
||||||
var folders []string
|
queue := SetupSink(config)
|
||||||
GetCollectorFolders(config.CollectorPath, &folders)
|
sinkTicker, sinkDone := RunSink(config, &queue)
|
||||||
|
collectTicker, collectDone := MainLoop(config, &queue)
|
||||||
|
time.Sleep(1600 * time.Second)
|
||||||
|
collectTicker.Stop()
|
||||||
|
collectDone <- true
|
||||||
|
CloseSink(config, &queue, sinkTicker, sinkDone)
|
||||||
|
|
||||||
for _, path := range folders {
|
// var folders []string
|
||||||
var col_config CollectorConfig
|
// GetCollectorFolders(config.CollectorPath, &folders)
|
||||||
LoadCollectorConfiguration(filepath.Join(path, "config.json"), &col_config)
|
|
||||||
stdout := run_cmd(filepath.Join(path, col_config.Command), col_config.Args)
|
|
||||||
|
|
||||||
metrics := strings.Split(stdout, "\n")
|
// for _, path := range folders {
|
||||||
for _, m := range metrics {
|
// var col_config CollectorConfig
|
||||||
if len(m) > 0 {
|
// LoadCollectorConfiguration(filepath.Join(path, "config.json"), &col_config)
|
||||||
t := strings.Fields(m)
|
// stdout := run_cmd(filepath.Join(path, col_config.Command), col_config.Args)
|
||||||
if len(t) == 2 {
|
|
||||||
var s strings.Builder
|
// metrics := strings.Split(stdout, "\n")
|
||||||
fmt.Fprintf(&s, "%s %d", m, time.Now().UnixNano())
|
// for _, m := range metrics {
|
||||||
m = s.String()
|
// if len(m) > 0 {
|
||||||
}
|
// t := strings.Fields(m)
|
||||||
fmt.Println("SEND", m)
|
// if len(t) == 2 {
|
||||||
}
|
// var s strings.Builder
|
||||||
}
|
// fmt.Fprintf(&s, "%s %d", m, time.Now().UnixNano())
|
||||||
}
|
// m = s.String()
|
||||||
|
// }
|
||||||
|
// fmt.Println("SEND", m)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user