Compare commits
9 Commits
Author | SHA1 | Date | |
---|---|---|---|
44ecd2d49c | |||
8958845f65 | |||
fe6ca819f4 | |||
101bdb92a3 | |||
55ea9b6edf | |||
96f278757b | |||
4daf0d4000 | |||
cc3fcba1d7 | |||
a32ae649f7 |
2
.gitignore
vendored
2
.gitignore
vendored
@ -1,4 +1,4 @@
|
|||||||
nodegopher
|
nodegopher
|
||||||
config.yaml*
|
config.yaml*
|
||||||
!config.yaml.sample
|
!config.sample.yml
|
||||||
main.go.*
|
main.go.*
|
||||||
|
12
README.md
12
README.md
@ -12,16 +12,20 @@ It gets time range values from Grafana, so it can relay them to prometheus if yo
|
|||||||
see config.yaml.sample
|
see config.yaml.sample
|
||||||
|
|
||||||
## Grafana datasource
|
## Grafana datasource
|
||||||
Successfully tested with yesoreyeram-infinity-datasource datasource
|
Successfully tested with [yesoreyeram-infinity-datasource](https://github.com/grafana/grafana-infinity-datasource) datasource
|
||||||
|
|
||||||
Create an infinity query, name it "Edges". Type "JSON", Parser "Backend", Source "URL", Format "Table", Method "GET"
|
Create an infinity query, name it "Edges". Type "JSON", Parser "Backend", Source "URL", Format "Table", Method "GET"
|
||||||
URL: "http://my-nodegopher-host:8080/graph1/edges?from=$__from&to=$__to&interval=$__interval"
|
URL: "http://my-nodegopher-host:8080/graph1/edges?from=$__from&to=$__to&interval=$__interval"
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
Create a second infinity query named "Nodes", same config,
|
Create a second infinity query named "Nodes", same config,
|
||||||
URL: "http://my-nodegopher-host:8080/graph1/nodes?from=$__from&to=$__to&interval=$__interval"
|
URL: "http://my-nodegopher-host:8080/graph1/nodes?from=$__from&to=$__to&interval=$__interval"
|
||||||
|
|
||||||
switch to visualization type "NodeGraph", and voila
|
switch to visualization type "NodeGraph", and voila
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## Instance management
|
## Instance management
|
||||||
### Reload configuration
|
### Reload configuration
|
||||||
You can reload configuration file when the API is running, with either sending a signal, or make a POST request.
|
You can reload configuration file when the API is running, with either sending a signal, or make a POST request.
|
||||||
@ -36,3 +40,9 @@ Sending POST on /reload :
|
|||||||
curl -XPOST http://my-nodegopher-host:8080/reload
|
curl -XPOST http://my-nodegopher-host:8080/reload
|
||||||
{"message":"configuration successfully reloaded"}
|
{"message":"configuration successfully reloaded"}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Reloading a badly formated configuration will produce an error and keep the old configuration running.
|
||||||
|
```
|
||||||
|
% curl -XPOST 127.1:8080/reload
|
||||||
|
{"error":"Unable to load new configuration, keeping old one. See logs."}
|
||||||
|
```
|
||||||
|
@ -1,43 +1,50 @@
|
|||||||
|
# Formatting metrics in main & secondarystat. Supported: "english", "french", "german", "ukrainian", "chinese", "arabic". Default is english.
|
||||||
|
language: 'english'
|
||||||
|
|
||||||
# datasource describe a way to get prometheus metrics.
|
# datasource describe a way to get prometheus metrics.
|
||||||
# Properties :
|
# Properties :
|
||||||
# - name: name of the query. To be used in edges or nodes mainstatquery or secondarystatquery. Result will be output in mainstat, or secondarystat.
|
# - name: name of the query. To be used in edges or nodes mainstatquery or secondarystatquery. Result will be output in mainstat, or secondarystat.
|
||||||
# - address: the address of prometheus.
|
# - address: the address of prometheus.
|
||||||
# - query: prometheus query. Same as typed in prometheus graph page.
|
# - query: prometheus query. Same as typed in prometheus graph page.
|
||||||
# - type: type of query. "query" will get instant value, "query_range" will get all samples for the grafana period. Result will be averaged.
|
# - type: type of query. "query" will get instant value, "query_range" will get all samples for the grafana period. Result will be averaged.
|
||||||
# - timeout: query timeout in seconds.
|
# - timeout: query timeout in seconds. default is 10.
|
||||||
datasources:
|
datasources:
|
||||||
- name: prom_samples_per_sec
|
- name: prom_samples_per_sec
|
||||||
type: query
|
type: query
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'rate(prometheus_tsdb_head_samples_appended_total{type="float"}[10m])'
|
query: 'rate(prometheus_tsdb_head_samples_appended_total{type="float"}[10m])'
|
||||||
timeout: 10
|
timeout: 15
|
||||||
- name: node_cpu_metric
|
- name: node_cpu_metric
|
||||||
# Simple query, return an instant metric
|
# Simple query, return an instant metric
|
||||||
type: query
|
type: query
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'sum(rate(node_cpu_seconds_total{instance="router01.local.lan:9100",job="node",mode!~"idle"}[30s]))*100'
|
query: 'sum(rate(node_cpu_seconds_total{instance="router01.local.lan:9100",job="node",mode!~"idle"}[30s]))*100'
|
||||||
timeout: 10
|
- name: node_cpu_metric_over_80
|
||||||
|
type: query
|
||||||
|
address: 'http://prometheus.local.lan:9090'
|
||||||
|
# Return 1 if cpu rate > 80%
|
||||||
|
query: '(sum(rate(node_cpu_seconds_total{instance="router01.local.lan:9100",job="node",mode!~"idle"}[30s]))*100) > bool 80'
|
||||||
- name: router01_net_down_rate
|
- name: router01_net_down_rate
|
||||||
# Range query. Return all metrics from a time range. Result will be averaged from these metrics. Time range will be provided by Grafana.
|
# Range query. Return all metrics from a time range. Result will be averaged from these metrics. Time range will be provided by Grafana.
|
||||||
type: query_range
|
type: query_range
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'rate(node_network_receive_bytes_total{device="igb0", instance="router01.local.lan:9100", job="node"}[30s])'
|
query: 'rate(node_network_receive_bytes_total{device="igb0", instance="router01.local.lan:9100", job="node"}[30s])'
|
||||||
timeout: 10
|
|
||||||
- name: router01_net_up_rate
|
- name: router01_net_up_rate
|
||||||
type: query_range
|
type: query_range
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'rate(node_network_transmit_bytes_total{device="igb0", instance="router01.local.lan:9100", job="node"}[30s])'
|
query: 'rate(node_network_transmit_bytes_total{device="igb0", instance="router01.local.lan:9100", job="node"}[30s])'
|
||||||
timeout: 10
|
|
||||||
- name: router01_lan_down_rate
|
- name: router01_lan_down_rate
|
||||||
type: query_range
|
type: query_range
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'rate(node_network_receive_bytes_total{device="ix3", instance="router01.local.lan:9100", job="node"}[30s])'
|
query: 'rate(node_network_receive_bytes_total{device="ix3", instance="router01.local.lan:9100", job="node"}[30s])'
|
||||||
timeout: 10
|
|
||||||
- name: router01_lan_up_rate
|
- name: router01_lan_up_rate
|
||||||
type: query_range
|
type: query_range
|
||||||
address: 'http://prometheus.local.lan:9090'
|
address: 'http://prometheus.local.lan:9090'
|
||||||
query: 'rate(node_network_transmit_bytes_total{device="ix3", instance="router01.local.lan:9100", job="node"}[30s])'
|
query: 'rate(node_network_transmit_bytes_total{device="ix3", instance="router01.local.lan:9100", job="node"}[30s])'
|
||||||
timeout: 10
|
- name: router01_net_down_rate_perten
|
||||||
|
type: query
|
||||||
|
address: 'http://prometheus.local.lan:9090'
|
||||||
|
query: 'rate(node_network_receive_bytes_total{device="igb0", instance="router01.local.lan:9100", job="node"}[30s])/62500000*10'
|
||||||
|
|
||||||
# graphs identifies context for a nodegraph. You can have many contexts, and your grafana query will mention this context name.
|
# graphs identifies context for a nodegraph. You can have many contexts, and your grafana query will mention this context name.
|
||||||
# For this example named "internet", grafana URL will be :
|
# For this example named "internet", grafana URL will be :
|
||||||
@ -47,16 +54,25 @@ datasources:
|
|||||||
# - name: name of the context
|
# - name: name of the context
|
||||||
# - nodes: list of nodegraph nodes
|
# - nodes: list of nodegraph nodes
|
||||||
# - edges: list of nodegraph edges
|
# - edges: list of nodegraph edges
|
||||||
|
#
|
||||||
|
# Dynamic fields :
|
||||||
|
# nodes:
|
||||||
|
# - mainstat : use mainstatquery and mainstatformat. mainstatquery should return a metric, mainstatformat is a printf format specifier.
|
||||||
|
# - secondarystat : use secondarystatquery and secondarystatformat. same as mainstat(query|format).
|
||||||
|
# - hightlighted : use highlightedquery. if result return > 0, item will be highligthed.
|
||||||
|
# edges:
|
||||||
|
# - same list as nodes.
|
||||||
|
# - thickness: use thicknessquery.
|
||||||
graphs:
|
graphs:
|
||||||
- name: internet
|
- name: internet
|
||||||
nodes:
|
nodes:
|
||||||
- name: internet
|
- name: host01
|
||||||
id: internet
|
id: host01
|
||||||
title: "internet"
|
title: host01
|
||||||
subtitle: "The internets"
|
subtitle: "A workstation"
|
||||||
color: "grey"
|
color: "green"
|
||||||
# icons come from https://developers.grafana.com/ui/latest/index.html?path=/story/docs-overview-icon--icons-overview
|
# icon come from https://developers.grafana.com/ui/latest/index.html?path=/story/docs-overview-icon--icons-overview
|
||||||
icon: "globe"
|
icon: "user"
|
||||||
- name: router01
|
- name: router01
|
||||||
id: router01
|
id: router01
|
||||||
title: router01
|
title: router01
|
||||||
@ -65,39 +81,43 @@ graphs:
|
|||||||
mainstatquery: '{{ node_cpu_metric }}'
|
mainstatquery: '{{ node_cpu_metric }}'
|
||||||
# Use %% if you want to display '%' in you metric label
|
# Use %% if you want to display '%' in you metric label
|
||||||
mainstatformat: '%0.2f%% cpu'
|
mainstatformat: '%0.2f%% cpu'
|
||||||
- name: host01
|
# highlight router if cpu > 80%
|
||||||
id: host01
|
highlightedquery: '{{ node_cpu_metric_over_80 }}'
|
||||||
title: host01
|
- name: internet
|
||||||
subtitle: "A workstation"
|
id: internet
|
||||||
color: "green"
|
title: "internet"
|
||||||
icon: "user"
|
subtitle: "The internets"
|
||||||
|
color: "grey"
|
||||||
|
icon: "globe"
|
||||||
edges:
|
edges:
|
||||||
- id: edge0
|
- id: edge0
|
||||||
source: internet
|
|
||||||
target: router01
|
|
||||||
mainstatquery: '{{ router01_net_up_rate }}'
|
|
||||||
mainstatformat: 'up %0.0f bps'
|
|
||||||
secondarystatquery: '{{ router01_net_down_rate }}'
|
|
||||||
secondarystatformat: 'down %0.0f bps'
|
|
||||||
- id: edge1
|
|
||||||
source: router01
|
|
||||||
target: internet
|
|
||||||
mainstatquery: '{{ router01_net_up_rate }}'
|
|
||||||
mainstatformat: 'up %0.0f bps'
|
|
||||||
secondarystatquery: '{{ router01_net_down_rate }}'
|
|
||||||
secondarystatformat: 'down %0.0f bps'
|
|
||||||
- id: edge2
|
|
||||||
source: router01
|
|
||||||
target: host01
|
|
||||||
mainstatquery: '{{ router01_lan_down_rate }}'
|
|
||||||
mainstatformat: 'up %0.0f bps'
|
|
||||||
secondarystatquery: '{{ router01_lan_up_rate }}'
|
|
||||||
secondarystatformat: 'down %0.0f bps'
|
|
||||||
- id: edge3
|
|
||||||
source: host01
|
source: host01
|
||||||
target: router01
|
target: router01
|
||||||
mainstatquery: '{{ router01_lan_down_rate }}'
|
mainstatquery: '{{ router01_lan_down_rate }}'
|
||||||
mainstatformat: 'up %0.0f bps'
|
mainstatformat: 'up %0.0f Bps'
|
||||||
secondarystatquery: '{{ router01_lan_up_rate }}'
|
secondarystatquery: '{{ router01_lan_up_rate }}'
|
||||||
secondarystatformat: 'down %0.0f bps'
|
secondarystatformat: 'down %0.0f Bps'
|
||||||
|
- id: edge1
|
||||||
|
source: router01
|
||||||
|
target: host01
|
||||||
|
mainstatquery: '{{ router01_lan_down_rate }}'
|
||||||
|
mainstatformat: 'up %0.0f Bps'
|
||||||
|
secondarystatquery: '{{ router01_lan_up_rate }}'
|
||||||
|
secondarystatformat: 'down %0.0f Bps'
|
||||||
|
thicknessquery: '{{ router01_net_down_rate_perten }}'
|
||||||
|
- id: edge2
|
||||||
|
source: router01
|
||||||
|
target: internet
|
||||||
|
mainstatquery: '{{ router01_net_up_rate }}'
|
||||||
|
mainstatformat: 'up %0.0f Bps'
|
||||||
|
secondarystatquery: '{{ router01_net_down_rate }}'
|
||||||
|
secondarystatformat: 'down %0.0f Bps'
|
||||||
|
- id: edge3
|
||||||
|
source: internet
|
||||||
|
target: router01
|
||||||
|
mainstatquery: '{{ router01_net_up_rate }}'
|
||||||
|
mainstatformat: 'up %0.0f Bps'
|
||||||
|
secondarystatquery: '{{ router01_net_down_rate }}'
|
||||||
|
secondarystatformat: 'down %0.0f Bps'
|
||||||
|
thicknessquery: '{{ router01_net_down_rate_perten }}'
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 30 KiB After Width: | Height: | Size: 34 KiB |
22
edges.go
22
edges.go
@ -20,7 +20,9 @@ type Edge struct {
|
|||||||
SecondaryStatFormat string `yaml:"secondarystatformat,omitempty" json:"secondarystatformat,omitempty"`
|
SecondaryStatFormat string `yaml:"secondarystatformat,omitempty" json:"secondarystatformat,omitempty"`
|
||||||
Color string `yaml:"color,omitempty" json:"color,omitempty"`
|
Color string `yaml:"color,omitempty" json:"color,omitempty"`
|
||||||
Thickness int `yaml:"thickness,omitempty" json:"thickness,omitempty"`
|
Thickness int `yaml:"thickness,omitempty" json:"thickness,omitempty"`
|
||||||
HighLighted bool `yaml:"highlighted,omitempty" json:"hightlighted,omitempty"`
|
ThicknessQuery string `yaml:"thicknessquery,omitempty" json:"thicknessquery,omitempty"`
|
||||||
|
Highlighted bool `yaml:"highlighted,omitempty" json:"hightlighted,omitempty"`
|
||||||
|
HighlightedQuery string `yaml:"highlightedquery,omitempty" json:"highlightedquery,omitempty"`
|
||||||
StrokeDashArray float32 `yaml:"strokeDasharray,omitempty" json:"strokeDasharray,omitempty"`
|
StrokeDashArray float32 `yaml:"strokeDasharray,omitempty" json:"strokeDasharray,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -56,10 +58,26 @@ func (e *Edge) SetMainStat(stat string) {
|
|||||||
e.MainStat = fmt.Sprintf("%s", stat)
|
e.MainStat = fmt.Sprintf("%s", stat)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *Edge) GetThicknessQuery() string {
|
||||||
|
return e.ThicknessQuery
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *Edge) SetThickness(thickness float64) {
|
||||||
|
e.Thickness = int(thickness)
|
||||||
|
}
|
||||||
|
|
||||||
func (e *Edge) SetSecondaryStat(stat string) {
|
func (e *Edge) SetSecondaryStat(stat string) {
|
||||||
e.SecondaryStat = fmt.Sprintf("%s", stat)
|
e.SecondaryStat = fmt.Sprintf("%s", stat)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *Edge) GetHighlightedQuery() string {
|
||||||
|
return e.HighlightedQuery
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *Edge) SetHighlighted(highlighted bool) {
|
||||||
|
e.Highlighted = highlighted
|
||||||
|
}
|
||||||
|
|
||||||
// Custom marshaler to not send (main|secondary)statquery
|
// Custom marshaler to not send (main|secondary)statquery
|
||||||
func (e Edge) MarshalJSON() ([]byte, error) {
|
func (e Edge) MarshalJSON() ([]byte, error) {
|
||||||
jsonRes := `{"id":"` + e.Id + `","source":"` + e.Source + `","target":"` + e.Target + `"`
|
jsonRes := `{"id":"` + e.Id + `","source":"` + e.Source + `","target":"` + e.Target + `"`
|
||||||
@ -75,7 +93,7 @@ func (e Edge) MarshalJSON() ([]byte, error) {
|
|||||||
if e.Thickness > 0 {
|
if e.Thickness > 0 {
|
||||||
jsonRes += `,"thickness":` + strconv.Itoa(e.Thickness)
|
jsonRes += `,"thickness":` + strconv.Itoa(e.Thickness)
|
||||||
}
|
}
|
||||||
if e.HighLighted {
|
if e.Highlighted {
|
||||||
jsonRes += `,"highlighted":true`
|
jsonRes += `,"highlighted":true`
|
||||||
}
|
}
|
||||||
// TODO : e.StrokeDashArray
|
// TODO : e.StrokeDashArray
|
||||||
|
223
main.go
223
main.go
@ -25,13 +25,18 @@ import (
|
|||||||
"github.com/fsnotify/fsnotify"
|
"github.com/fsnotify/fsnotify"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
|
|
||||||
|
"golang.org/x/text/language"
|
||||||
|
"golang.org/x/text/message"
|
||||||
|
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
"github.com/prometheus/client_golang/api"
|
"github.com/prometheus/client_golang/api"
|
||||||
v1 "github.com/prometheus/client_golang/api/prometheus/v1"
|
v1 "github.com/prometheus/client_golang/api/prometheus/v1"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
gVersion = "0.2.1"
|
gVersion = "0.2.5"
|
||||||
|
// Default datasource timeout is 10 seconds
|
||||||
|
gDefaultDSTimeout = 10
|
||||||
)
|
)
|
||||||
|
|
||||||
type PromDataSourceConfig struct {
|
type PromDataSourceConfig struct {
|
||||||
@ -58,6 +63,10 @@ type Item interface {
|
|||||||
GetSecondaryStatQuery() string
|
GetSecondaryStatQuery() string
|
||||||
GetSecondaryStatFormat() string
|
GetSecondaryStatFormat() string
|
||||||
SetSecondaryStat(string)
|
SetSecondaryStat(string)
|
||||||
|
GetThicknessQuery() string
|
||||||
|
SetThickness(float64)
|
||||||
|
GetHighlightedQuery() string
|
||||||
|
SetHighlighted(bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Query arguments. Based on grafana internal variables.
|
// Query arguments. Based on grafana internal variables.
|
||||||
@ -75,6 +84,7 @@ type MyRange struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
gPrinter *message.Printer
|
||||||
gDebug bool
|
gDebug bool
|
||||||
gGraphs []Graph
|
gGraphs []Graph
|
||||||
gDataSources []PromDataSourceConfig
|
gDataSources []PromDataSourceConfig
|
||||||
@ -112,13 +122,13 @@ func (d *PromDataSourceConfig) GetData(timeRange *MyRange) (float64, error) {
|
|||||||
result, warnings, err = v1api.QueryRange(ctx, d.Query, rng, v1.WithTimeout(time.Duration(d.Timeout)*time.Second))
|
result, warnings, err = v1api.QueryRange(ctx, d.Query, rng, v1.WithTimeout(time.Duration(d.Timeout)*time.Second))
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Errorf("DataSourceConfig.GetData: Error querying Prometheus: %v\n", err)
|
log.Errorf("DataSourceConfig.GetData: Error querying Prometheus: %v. Query is: %s\n", err, d.Query)
|
||||||
return 0.0, err
|
return 0.0, err
|
||||||
}
|
}
|
||||||
if len(warnings) > 0 {
|
if len(warnings) > 0 {
|
||||||
log.Warningf("DataSourceConfig.GetData: Warnings: %v\n", warnings)
|
log.Warningf("DataSourceConfig.GetData: Warnings: %v\n", warnings)
|
||||||
}
|
}
|
||||||
log.Debugf("DataSourceConfig.GetData: Result: %v\n", result)
|
//log.Debugf("DataSourceConfig.GetData: Result: %v\n", result)
|
||||||
|
|
||||||
switch {
|
switch {
|
||||||
case result.Type() == model.ValScalar:
|
case result.Type() == model.ValScalar:
|
||||||
@ -165,6 +175,7 @@ func (g Graph) BuildMetrics(items *[]Item, timeRange *MyRange) (*[]Item, error)
|
|||||||
defer gCfgMutex.RUnlock()
|
defer gCfgMutex.RUnlock()
|
||||||
|
|
||||||
for _, item := range *items {
|
for _, item := range *items {
|
||||||
|
// Handle mainStat
|
||||||
if len(item.GetMainStatQuery()) > 0 {
|
if len(item.GetMainStatQuery()) > 0 {
|
||||||
log.Debugf("Item %s have mainstatquery: %s\n", item.GetId(), item.GetMainStatQuery())
|
log.Debugf("Item %s have mainstatquery: %s\n", item.GetId(), item.GetMainStatQuery())
|
||||||
r := gDSVarCompRegex.FindStringSubmatch(item.GetMainStatQuery())
|
r := gDSVarCompRegex.FindStringSubmatch(item.GetMainStatQuery())
|
||||||
@ -185,12 +196,13 @@ func (g Graph) BuildMetrics(items *[]Item, timeRange *MyRange) (*[]Item, error)
|
|||||||
if len(format) == 0 {
|
if len(format) == 0 {
|
||||||
format = "%f"
|
format = "%f"
|
||||||
}
|
}
|
||||||
log.Debugf("buildMetrics: Replace %s mainstat with %s\n", item.GetId(), fmt.Sprintf(format, value))
|
log.Debugf("buildMetrics: Replace %s mainstat with %s\n", item.GetId(), gPrinter.Sprintf(format, value))
|
||||||
item.SetMainStat(fmt.Sprintf(format, value))
|
item.SetMainStat(gPrinter.Sprintf(format, value))
|
||||||
} else {
|
} else {
|
||||||
log.Errorf("buildMetrics: Item %s mainstatquery unparseable: %s\n", item.GetId(), item.GetMainStatQuery())
|
log.Errorf("buildMetrics: Item %s mainstatquery unparseable: %s\n", item.GetId(), item.GetMainStatQuery())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Handle secondaryStat
|
||||||
if len(item.GetSecondaryStatQuery()) > 0 {
|
if len(item.GetSecondaryStatQuery()) > 0 {
|
||||||
log.Debugf("Item %s have mainstatquery: %s\n", item.GetId(), item.GetSecondaryStatQuery())
|
log.Debugf("Item %s have mainstatquery: %s\n", item.GetId(), item.GetSecondaryStatQuery())
|
||||||
r := gDSVarCompRegex.FindStringSubmatch(item.GetSecondaryStatQuery())
|
r := gDSVarCompRegex.FindStringSubmatch(item.GetSecondaryStatQuery())
|
||||||
@ -211,12 +223,66 @@ func (g Graph) BuildMetrics(items *[]Item, timeRange *MyRange) (*[]Item, error)
|
|||||||
if len(format) == 0 {
|
if len(format) == 0 {
|
||||||
format = "%f"
|
format = "%f"
|
||||||
}
|
}
|
||||||
log.Debugf("buildMetrics: Replace %s secondarystat with %s\n", item.GetId(), fmt.Sprintf(format, value))
|
log.Debugf("buildMetrics: Replace %s secondarystat with %s\n", item.GetId(), gPrinter.Sprintf(format, value))
|
||||||
item.SetSecondaryStat(fmt.Sprintf(format, value))
|
item.SetSecondaryStat(gPrinter.Sprintf(format, value))
|
||||||
} else {
|
} else {
|
||||||
log.Errorf("buildMetrics: Item %s secondarystatquery unparseable: %s\n", item.GetId(), item.GetSecondaryStatQuery())
|
log.Errorf("buildMetrics: Item %s secondarystatquery unparseable: %s\n", item.GetId(), item.GetSecondaryStatQuery())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Handle highlighted
|
||||||
|
if len(item.GetHighlightedQuery()) > 0 {
|
||||||
|
log.Debugf("buildMetrics: Item %s have highlightedquery: %s\n", item.GetId(), item.GetHighlightedQuery())
|
||||||
|
r := gDSVarCompRegex.FindStringSubmatch(item.GetHighlightedQuery())
|
||||||
|
if len(r) > 1 {
|
||||||
|
var value float64
|
||||||
|
dsname := strings.TrimSpace(r[1])
|
||||||
|
log.Debugf("buildMetrics: datasource from highlightedquery : %s\n", dsname)
|
||||||
|
for _, d := range gDataSources {
|
||||||
|
if strings.EqualFold(d.Name, dsname) {
|
||||||
|
value, err = d.GetData(timeRange)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
highlight := false
|
||||||
|
if value == 0 {
|
||||||
|
highlight = false
|
||||||
|
} else {
|
||||||
|
highlight = true
|
||||||
|
}
|
||||||
|
log.Debugf("buildMetrics: Replace %s highlighted with %t\n", item.GetId(), highlight)
|
||||||
|
item.SetHighlighted(highlight)
|
||||||
|
} else {
|
||||||
|
log.Errorf("buildMetrics: Item %s highlightedquery unparseable: %s\n", item.GetId(), item.GetHighlightedQuery())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch item.(type) {
|
||||||
|
case *Edge:
|
||||||
|
if len(item.GetThicknessQuery()) > 0 {
|
||||||
|
log.Debugf("buildMetrics: Item %s have thicknessquery: %s\n", item.GetId(), item.GetThicknessQuery())
|
||||||
|
r := gDSVarCompRegex.FindStringSubmatch(item.GetThicknessQuery())
|
||||||
|
if len(r) > 1 {
|
||||||
|
var value float64
|
||||||
|
dsname := strings.TrimSpace(r[1])
|
||||||
|
log.Debugf("buildMetrics: datasource from thicknessquery : %s\n", dsname)
|
||||||
|
for _, d := range gDataSources {
|
||||||
|
if strings.EqualFold(d.Name, dsname) {
|
||||||
|
value, err = d.GetData(timeRange)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Debugf("buildMetrics: Replace %s thickness with %s\n", item.GetId(), fmt.Sprintf("%0.0f", value))
|
||||||
|
item.SetThickness(value)
|
||||||
|
} else {
|
||||||
|
log.Errorf("buildMetrics: Item %s thicknessquery unparseable: %s\n", item.GetId(), item.GetThicknessQuery())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return items, nil
|
return items, nil
|
||||||
}
|
}
|
||||||
@ -263,7 +329,7 @@ func getGraph(name string) (Graph, error) {
|
|||||||
return Graph{}, fmt.Errorf("Graph not found: %s", name)
|
return Graph{}, fmt.Errorf("Graph not found: %s", name)
|
||||||
}
|
}
|
||||||
|
|
||||||
func initRoutes(r *gin.Engine) {
|
func initRoutes(r *gin.Engine, confFile string) {
|
||||||
r.GET("/ping", func(c *gin.Context) {
|
r.GET("/ping", func(c *gin.Context) {
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"message": "pong",
|
"message": "pong",
|
||||||
@ -272,7 +338,10 @@ func initRoutes(r *gin.Engine) {
|
|||||||
|
|
||||||
// An endpoint to force read of configuration file
|
// An endpoint to force read of configuration file
|
||||||
r.POST("/reload", func(c *gin.Context) {
|
r.POST("/reload", func(c *gin.Context) {
|
||||||
reloadConfigFile()
|
if err := reloadConfigFile(confFile); err != nil {
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"message": "configuration successfully reloaded",
|
"message": "configuration successfully reloaded",
|
||||||
})
|
})
|
||||||
@ -361,21 +430,99 @@ func toggleDebug() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func reloadConfigFile() {
|
// Deep copy src Node into a new memory space
|
||||||
// First reread config file
|
func newNodeClone(src *Node) *Node {
|
||||||
if err := viper.ReadInConfig(); err != nil {
|
return &Node{
|
||||||
if _, ok := err.(viper.ConfigFileNotFoundError); ok {
|
Name: src.Name,
|
||||||
log.Fatalf("config file not found")
|
Id: src.Id,
|
||||||
os.Exit(1)
|
Title: src.Title,
|
||||||
} else {
|
Subtitle: src.Subtitle,
|
||||||
log.Fatalf("unknown error looking for config file: %v", err)
|
MainStat: src.MainStat,
|
||||||
os.Exit(1)
|
MainStatQuery: src.MainStatQuery,
|
||||||
}
|
MainStatFormat: src.MainStatFormat,
|
||||||
|
SecondaryStat: src.SecondaryStat,
|
||||||
|
SecondaryStatQuery: src.SecondaryStatQuery,
|
||||||
|
SecondaryStatFormat: src.SecondaryStatFormat,
|
||||||
|
Color: src.Color,
|
||||||
|
Icon: src.Icon,
|
||||||
|
NodeRadius: src.NodeRadius,
|
||||||
|
Highlighted: src.Highlighted,
|
||||||
|
HighlightedQuery: src.HighlightedQuery,
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deep copy src Edge into a new memory space
|
||||||
|
func newEdgeClone(src *Edge) *Edge {
|
||||||
|
return &Edge{
|
||||||
|
Id: src.Id,
|
||||||
|
Source: src.Source,
|
||||||
|
Target: src.Target,
|
||||||
|
MainStat: src.MainStat,
|
||||||
|
MainStatQuery: src.MainStatQuery,
|
||||||
|
MainStatFormat: src.MainStatFormat,
|
||||||
|
SecondaryStat: src.SecondaryStat,
|
||||||
|
SecondaryStatQuery: src.SecondaryStatQuery,
|
||||||
|
SecondaryStatFormat: src.SecondaryStatFormat,
|
||||||
|
Color: src.Color,
|
||||||
|
Thickness: src.Thickness,
|
||||||
|
ThicknessQuery: src.ThicknessQuery,
|
||||||
|
Highlighted: src.Highlighted,
|
||||||
|
HighlightedQuery: src.HighlightedQuery,
|
||||||
|
StrokeDashArray: src.StrokeDashArray,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This function assume we already have a running configuration.
|
||||||
|
func reloadConfigFile(confFile string) error {
|
||||||
|
oldConfigRestored := false
|
||||||
|
// We need to keep this config, incase the new one is b0rken
|
||||||
|
fname := fmt.Sprintf("/tmp/nodegopher.%d.yaml", os.Getpid())
|
||||||
|
if err := viper.WriteConfigAs(fname); err != nil {
|
||||||
|
log.Errorf("Unable to save current running config to %s, wont reload configuration.\n", fname)
|
||||||
|
return fmt.Errorf("Unable to save current configuration, configuration not reloaded. See logs.")
|
||||||
|
}
|
||||||
|
defer os.Remove(fname)
|
||||||
|
|
||||||
|
// Reread config file
|
||||||
|
if oldErr := viper.ReadInConfig(); oldErr != nil {
|
||||||
|
if _, ok := oldErr.(viper.ConfigFileNotFoundError); ok {
|
||||||
|
log.Errorf("config file not found")
|
||||||
|
} else {
|
||||||
|
log.Errorf("unknown error looking for config file: %v", oldErr)
|
||||||
|
}
|
||||||
|
// Restore old configuration and notify.
|
||||||
|
log.Debugf("Fallback on previous configuration.\n")
|
||||||
|
viper.SetConfigFile(fname)
|
||||||
|
if err := viper.ReadInConfig(); err != nil {
|
||||||
|
log.Fatalf("Unable to restore configuration, and new is invalid. fix it now.\n")
|
||||||
|
}
|
||||||
|
viper.SetConfigFile(confFile)
|
||||||
|
oldConfigRestored = true
|
||||||
|
}
|
||||||
|
|
||||||
|
switch viper.Get("language").(string) {
|
||||||
|
case "english":
|
||||||
|
gPrinter = message.NewPrinter(language.English)
|
||||||
|
case "french":
|
||||||
|
gPrinter = message.NewPrinter(language.French)
|
||||||
|
case "german":
|
||||||
|
gPrinter = message.NewPrinter(language.German)
|
||||||
|
case "ukrainian":
|
||||||
|
gPrinter = message.NewPrinter(language.Ukrainian)
|
||||||
|
case "arabic":
|
||||||
|
gPrinter = message.NewPrinter(language.Arabic)
|
||||||
|
case "chinese":
|
||||||
|
gPrinter = message.NewPrinter(language.Chinese)
|
||||||
|
default:
|
||||||
|
log.Errorf("Language not implemented: %s. Fallback to english\n", viper.Get("language").(string))
|
||||||
|
gPrinter = message.NewPrinter(language.English)
|
||||||
|
}
|
||||||
|
|
||||||
// then clear current config, after acquiring WriteLock
|
// then clear current config, after acquiring WriteLock
|
||||||
gCfgMutex.Lock()
|
gCfgMutex.Lock()
|
||||||
defer gCfgMutex.Unlock()
|
defer gCfgMutex.Unlock()
|
||||||
|
|
||||||
|
// We need to keep this config, incase the new one is b0rken
|
||||||
for _, g := range gGraphs {
|
for _, g := range gGraphs {
|
||||||
g.Nodes = nil
|
g.Nodes = nil
|
||||||
g.Edges = nil
|
g.Edges = nil
|
||||||
@ -395,34 +542,40 @@ func reloadConfigFile() {
|
|||||||
Edges []Edge `yaml:"edges"`
|
Edges []Edge `yaml:"edges"`
|
||||||
}{}
|
}{}
|
||||||
yaml.Unmarshal(yd, &tmp)
|
yaml.Unmarshal(yd, &tmp)
|
||||||
var graphNodes []Item
|
|
||||||
var graphEdges []Item
|
|
||||||
for _, n := range tmp.Nodes {
|
|
||||||
graphNodes = append(graphNodes, &n)
|
|
||||||
}
|
|
||||||
for _, e := range tmp.Edges {
|
|
||||||
graphEdges = append(graphEdges, &e)
|
|
||||||
}
|
|
||||||
|
|
||||||
graph := Graph{
|
graph := Graph{
|
||||||
Name: tmp.Name,
|
Name: tmp.Name,
|
||||||
Nodes: graphNodes,
|
}
|
||||||
Edges: graphEdges,
|
for _, n := range tmp.Nodes {
|
||||||
|
// Deep copy Node so garbage collecting tmp won't pull the carpet under our feet
|
||||||
|
graph.Nodes = append(graph.Nodes, newNodeClone(&n))
|
||||||
|
}
|
||||||
|
for _, e := range tmp.Edges {
|
||||||
|
// Deep copy Edge
|
||||||
|
graph.Edges = append(graph.Edges, newEdgeClone(&e))
|
||||||
}
|
}
|
||||||
gGraphs = append(gGraphs, graph)
|
gGraphs = append(gGraphs, graph)
|
||||||
}
|
}
|
||||||
|
|
||||||
if viper.Get("datasources") == nil {
|
if viper.Get("datasources") == nil {
|
||||||
log.Printf("no datasources found, data will be static")
|
log.Warningf("no datasources found, data will be static")
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
dss := viper.Get("datasources").([]interface{})
|
dss := viper.Get("datasources").([]interface{})
|
||||||
for _, d := range dss {
|
for _, d := range dss {
|
||||||
yd, _ := yaml.Marshal(d)
|
yd, _ := yaml.Marshal(d)
|
||||||
var ds PromDataSourceConfig
|
var ds PromDataSourceConfig
|
||||||
yaml.Unmarshal(yd, &ds)
|
yaml.Unmarshal(yd, &ds)
|
||||||
|
// Set default Values
|
||||||
|
if ds.Timeout == 0 {
|
||||||
|
ds.Timeout = gDefaultDSTimeout
|
||||||
|
}
|
||||||
gDataSources = append(gDataSources, ds)
|
gDataSources = append(gDataSources, ds)
|
||||||
}
|
}
|
||||||
|
if oldConfigRestored {
|
||||||
|
return fmt.Errorf("Unable to load new configuration, keeping old one. See logs.")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
@ -475,7 +628,7 @@ func main() {
|
|||||||
// FIXME: Watch config changes. Does not work on FreeBSD. TODO: Test with linux
|
// FIXME: Watch config changes. Does not work on FreeBSD. TODO: Test with linux
|
||||||
viper.OnConfigChange(func(e fsnotify.Event) {
|
viper.OnConfigChange(func(e fsnotify.Event) {
|
||||||
log.Printf("Config file changed, reloading data\n")
|
log.Printf("Config file changed, reloading data\n")
|
||||||
reloadConfigFile()
|
reloadConfigFile(confFile)
|
||||||
})
|
})
|
||||||
|
|
||||||
// Lets reload config on SIGHUP
|
// Lets reload config on SIGHUP
|
||||||
@ -485,11 +638,11 @@ func main() {
|
|||||||
for {
|
for {
|
||||||
_ = <- sigs
|
_ = <- sigs
|
||||||
log.Infof("SIGHUP received, reloading configuration\n")
|
log.Infof("SIGHUP received, reloading configuration\n")
|
||||||
reloadConfigFile()
|
reloadConfigFile(confFile)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
reloadConfigFile()
|
reloadConfigFile(confFile)
|
||||||
|
|
||||||
// Capture variable name. There should be only one variable. Space is tolerated before and after name.
|
// Capture variable name. There should be only one variable. Space is tolerated before and after name.
|
||||||
gDSVarCompRegex = regexp.MustCompile(`^\{\{(?:\ )?([a-zA-Z0-9\-_]+)(?:\ )?\}\}$`)
|
gDSVarCompRegex = regexp.MustCompile(`^\{\{(?:\ )?([a-zA-Z0-9\-_]+)(?:\ )?\}\}$`)
|
||||||
@ -500,6 +653,6 @@ func main() {
|
|||||||
log.Printf("Starting NodeGopher v.%s\n", gVersion)
|
log.Printf("Starting NodeGopher v.%s\n", gVersion)
|
||||||
|
|
||||||
r := gin.Default()
|
r := gin.Default()
|
||||||
initRoutes(r)
|
initRoutes(r, confFile)
|
||||||
r.Run(listen)
|
r.Run(listen)
|
||||||
}
|
}
|
||||||
|
24
nodes.go
24
nodes.go
@ -22,7 +22,8 @@ type Node struct {
|
|||||||
Color string `yaml:"color,omitempty" json:"color,omitempty"`
|
Color string `yaml:"color,omitempty" json:"color,omitempty"`
|
||||||
Icon string `yaml:"icon,omitempty" json:"icon,omitempty"`
|
Icon string `yaml:"icon,omitempty" json:"icon,omitempty"`
|
||||||
NodeRadius int `yaml:"noderadius,omitempty" json:"noderadius,omitempty"`
|
NodeRadius int `yaml:"noderadius,omitempty" json:"noderadius,omitempty"`
|
||||||
HighLighted bool `yaml:"highlighted,omitempty" json:"hightlighted,omitempty"`
|
Highlighted bool `yaml:"highlighted,omitempty" json:"hightlighted,omitempty"`
|
||||||
|
HighlightedQuery string `yaml:"highlightedquery,omitempty" json:"highlightedquery,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (n Node) GetId() string {
|
func (n Node) GetId() string {
|
||||||
@ -61,6 +62,25 @@ func (n *Node) SetSecondaryStat(stat string) {
|
|||||||
n.SecondaryStat = fmt.Sprintf("%s", stat)
|
n.SecondaryStat = fmt.Sprintf("%s", stat)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Have to be implemented to satisfy Item interface
|
||||||
|
func (n *Node) GetThicknessQuery() string {
|
||||||
|
panic("Not an Edge")
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Have to be implemented to satisfy Item interface
|
||||||
|
func (n *Node) SetThickness(thickness float64) {
|
||||||
|
panic("Not an Edge")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Node) GetHighlightedQuery() string {
|
||||||
|
return n.HighlightedQuery
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Node) SetHighlighted(highlighted bool) {
|
||||||
|
n.Highlighted = highlighted
|
||||||
|
}
|
||||||
|
|
||||||
// Custom marshaler to not send (main|secondary)statquery
|
// Custom marshaler to not send (main|secondary)statquery
|
||||||
func (n Node) MarshalJSON() ([]byte, error) {
|
func (n Node) MarshalJSON() ([]byte, error) {
|
||||||
jsonRes := `{"name":"` + n.Name + `","id":"` + n.Id + `"`
|
jsonRes := `{"name":"` + n.Name + `","id":"` + n.Id + `"`
|
||||||
@ -85,7 +105,7 @@ func (n Node) MarshalJSON() ([]byte, error) {
|
|||||||
if n.NodeRadius > 0 {
|
if n.NodeRadius > 0 {
|
||||||
jsonRes += `,"noderadius":` + strconv.Itoa(n.NodeRadius)
|
jsonRes += `,"noderadius":` + strconv.Itoa(n.NodeRadius)
|
||||||
}
|
}
|
||||||
if n.HighLighted {
|
if n.Highlighted {
|
||||||
jsonRes += `,"highlighted":true`
|
jsonRes += `,"highlighted":true`
|
||||||
}
|
}
|
||||||
// TODO : n.StrokeDashArray
|
// TODO : n.StrokeDashArray
|
||||||
|
Loading…
x
Reference in New Issue
Block a user