mirror of
https://github.com/ClusterCockpit/cc-metric-collector.git
synced 2025-01-24 21:09:06 +01:00
Merge pull request #1 from ClusterCockpit/dev-jan
Go Modules and further changes
This commit is contained in:
commit
63cdc060c7
307
clusterdaemon.go
307
clusterdaemon.go
@ -1,199 +1,196 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"os/exec"
|
||||
// "context"
|
||||
"strings"
|
||||
"time"
|
||||
"sort"
|
||||
"path/filepath"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"os"
|
||||
"os/exec"
|
||||
|
||||
// "context"
|
||||
"encoding/json"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
|
||||
type GlobalConfig struct {
|
||||
Sink struct {
|
||||
User string `json:"user"`
|
||||
Password string `json:"password"`
|
||||
} `json:"sink"`
|
||||
Host string `json:"host"`
|
||||
Port string `json:"port"`
|
||||
Report struct {
|
||||
Levels string `json:"levels"`
|
||||
Interval int `json:"interval"`
|
||||
} `json:"report"`
|
||||
Schedule struct {
|
||||
Core struct {
|
||||
Frequency int `json:"frequency"`
|
||||
Duration int `json:"duration"`
|
||||
} `json:"core"`
|
||||
Node struct {
|
||||
Frequency int `json:"frequency"`
|
||||
Duration int `json:"duration"`
|
||||
} `json:"node"`
|
||||
} `json:"schedule"`
|
||||
Metrics string `json:"metrics"`
|
||||
CollectorPath string `json:"collector_path"`
|
||||
Sink struct {
|
||||
User string `json:"user"`
|
||||
Password string `json:"password"`
|
||||
} `json:"sink"`
|
||||
Host string `json:"host"`
|
||||
Port string `json:"port"`
|
||||
Report struct {
|
||||
Levels string `json:"levels"`
|
||||
Interval int `json:"interval"`
|
||||
} `json:"report"`
|
||||
Schedule struct {
|
||||
Core struct {
|
||||
Frequency int `json:"frequency"`
|
||||
Duration int `json:"duration"`
|
||||
} `json:"core"`
|
||||
Node struct {
|
||||
Frequency int `json:"frequency"`
|
||||
Duration int `json:"duration"`
|
||||
} `json:"node"`
|
||||
} `json:"schedule"`
|
||||
Metrics []string `json:"metrics"`
|
||||
CollectorPath string `json:"collector_path"`
|
||||
}
|
||||
|
||||
type CollectorConfig struct {
|
||||
Command string `json:"command"`
|
||||
Args string `json:"arguments"`
|
||||
Command string `json:"command"`
|
||||
Args string `json:"arguments"`
|
||||
}
|
||||
|
||||
func LoadGlobalConfiguration(file string, config* GlobalConfig) error {
|
||||
configFile, err := os.Open(file)
|
||||
defer configFile.Close()
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
}
|
||||
jsonParser := json.NewDecoder(configFile)
|
||||
jsonParser.Decode(config)
|
||||
return err
|
||||
func LoadGlobalConfiguration(file string, config *GlobalConfig) error {
|
||||
configFile, err := os.Open(file)
|
||||
defer configFile.Close()
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
}
|
||||
jsonParser := json.NewDecoder(configFile)
|
||||
jsonParser.Decode(config)
|
||||
return err
|
||||
}
|
||||
|
||||
func LoadCollectorConfiguration(file string, config* CollectorConfig) error {
|
||||
configFile, err := os.Open(file)
|
||||
defer configFile.Close()
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
}
|
||||
jsonParser := json.NewDecoder(configFile)
|
||||
jsonParser.Decode(config)
|
||||
return err
|
||||
func LoadCollectorConfiguration(file string, config *CollectorConfig) error {
|
||||
configFile, err := os.Open(file)
|
||||
defer configFile.Close()
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
}
|
||||
jsonParser := json.NewDecoder(configFile)
|
||||
jsonParser.Decode(config)
|
||||
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)
|
||||
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
|
||||
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)
|
||||
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
|
||||
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)
|
||||
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()
|
||||
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):
|
||||
//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()
|
||||
stdout, err := command.Output()
|
||||
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
return ""
|
||||
}
|
||||
return (string(stdout))
|
||||
if err != nil {
|
||||
fmt.Println(err.Error())
|
||||
return ""
|
||||
}
|
||||
return (string(stdout))
|
||||
}
|
||||
|
||||
func GetSingleCollector(folders *[]string) filepath.WalkFunc {
|
||||
return func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
if (info.IsDir()) {
|
||||
configfile := filepath.Join(path, "config.json")
|
||||
if _, err := os.Stat(configfile); err == nil {
|
||||
*folders = append(*folders, path)
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
return func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
if info.IsDir() {
|
||||
configfile := filepath.Join(path, "config.json")
|
||||
if _, err := os.Stat(configfile); err == nil {
|
||||
*folders = append(*folders, path)
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func GetCollectorFolders(root string, folders *[]string) error {
|
||||
|
||||
err := filepath.Walk(root, GetSingleCollector(folders))
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return err
|
||||
|
||||
err := filepath.Walk(root, GetSingleCollector(folders))
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
func main() {
|
||||
// fmt.Println("Hello")
|
||||
// cmd_opts := []string{"la","le","lu"}
|
||||
// cmd := "echo"
|
||||
// s := run_cmd(cmd, cmd_opts)
|
||||
// fmt.Println(s)
|
||||
// tags := map[string]string {
|
||||
// "host" : "broadep2",
|
||||
// }
|
||||
// fields := map[string]interface{} {
|
||||
// "value" : float64(1.0),
|
||||
// }
|
||||
// fmt.Println(CreatePoint("flops_any", tags, fields, time.Now().UnixNano()))
|
||||
var config GlobalConfig
|
||||
LoadGlobalConfiguration("config.json", &config)
|
||||
// fmt.Println(config)
|
||||
var folders []string
|
||||
GetCollectorFolders(config.CollectorPath, &folders)
|
||||
// fmt.Println(folders)
|
||||
for _, path := range folders {
|
||||
var col_config CollectorConfig
|
||||
configfile := filepath.Join(path, "config.json")
|
||||
LoadCollectorConfiguration(configfile, &col_config)
|
||||
cmd := filepath.Join(path, col_config.Command)
|
||||
stdout := run_cmd(cmd, col_config.Args)
|
||||
metrics := strings.Split(stdout, "\n")
|
||||
for _, m := range metrics {
|
||||
if len(m) > 0 {
|
||||
t := strings.Fields(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)
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// fmt.Println("Hello")
|
||||
// cmd_opts := []string{"la","le","lu"}
|
||||
// cmd := "echo"
|
||||
// s := run_cmd(cmd, cmd_opts)
|
||||
// fmt.Println(s)
|
||||
// tags := map[string]string {
|
||||
// "host" : "broadep2",
|
||||
// }
|
||||
// fields := map[string]interface{} {
|
||||
// "value" : float64(1.0),
|
||||
// }
|
||||
// fmt.Println(CreatePoint("flops_any", tags, fields, time.Now().UnixNano()))
|
||||
var config GlobalConfig
|
||||
LoadGlobalConfiguration("config.json", &config)
|
||||
|
||||
var folders []string
|
||||
GetCollectorFolders(config.CollectorPath, &folders)
|
||||
|
||||
for _, path := range folders {
|
||||
var col_config CollectorConfig
|
||||
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 _, m := range metrics {
|
||||
if len(m) > 0 {
|
||||
t := strings.Fields(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)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
64
collectors/likwid.go
Normal file
64
collectors/likwid.go
Normal file
@ -0,0 +1,64 @@
|
||||
package collectors
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
protocol "github.com/influxdata/line-protocol"
|
||||
)
|
||||
|
||||
type LikwidCollector struct {
|
||||
name string
|
||||
tags []*protocol.Tag
|
||||
fields []*protocol.Field
|
||||
t time.Time
|
||||
encoder *protocol.Encoder
|
||||
}
|
||||
|
||||
func (c *LikwidCollector) Name() string {
|
||||
return c.name
|
||||
}
|
||||
func (c *LikwidCollector) TagList() []*protocol.Tag {
|
||||
return c.tags
|
||||
}
|
||||
|
||||
func (c *LikwidCollector) FieldList() []*protocol.Field {
|
||||
return c.fields
|
||||
}
|
||||
|
||||
func (c *LikwidCollector) Time() time.Time {
|
||||
return c.t
|
||||
}
|
||||
|
||||
func (c *LikwidCollector) New() {
|
||||
buf := &bytes.Buffer{}
|
||||
c.encoder = protocol.NewEncoder(buf)
|
||||
c.encoder.SetMaxLineBytes(1024)
|
||||
}
|
||||
|
||||
func (c *LikwidCollector) Start(
|
||||
level string,
|
||||
frequency time.Duration,
|
||||
duration int) {
|
||||
ticker := time.NewTicker(frequency * time.Second)
|
||||
done := make(chan bool)
|
||||
|
||||
go func() {
|
||||
for {
|
||||
select {
|
||||
case <-done:
|
||||
return
|
||||
case t := <-ticker.C:
|
||||
fmt.Println("Tick at", t)
|
||||
|
||||
c.encoder.Encode(c)
|
||||
}
|
||||
}
|
||||
}()
|
||||
|
||||
time.Sleep(1600 * time.Second)
|
||||
ticker.Stop()
|
||||
done <- true
|
||||
fmt.Println("Ticker stopped")
|
||||
}
|
5
go.mod
Normal file
5
go.mod
Normal file
@ -0,0 +1,5 @@
|
||||
module github.com/ClusterCockpit/cc-metric-collector
|
||||
|
||||
go 1.16
|
||||
|
||||
require github.com/influxdata/line-protocol v0.0.0-20210311194329-9aa0e372d097
|
Loading…
Reference in New Issue
Block a user