Browse Source

first test for lua calls

go-modules
sberk42 4 years ago
parent
commit
81979f3749
  1. 318
      fritzbox_lua/lua_client.go
  2. 298
      main.go

318
fritzbox_lua/lua_client.go

@ -0,0 +1,318 @@
// client for fritzbox lua API
package lua_client
// Copyright 2020 Andreas Krebs
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import (
"bytes"
"crypto/md5"
"encoding/json"
"encoding/xml"
"errors"
"fmt"
"io/ioutil"
"net/http"
"regexp"
"strconv"
"strings"
"golang.org/x/text/encoding/unicode"
"golang.org/x/text/transform"
)
// session XML from login_sid.lua
type SessionInfo struct {
SID string `xml:"SID"`
Challenge string `xml:"Challenge"`
BlockTime int `xml:"BlockTime"`
Rights string `xml:"Rights"`
}
type LuaSession struct {
BaseUrl string
Username string
Password string
SID string
SessionInfo SessionInfo
}
type LuaPage struct {
Path string
Params string
}
type LuaMetricValueDefinition struct {
Path string
Key string
Labels []string
}
type LuaMetricValue struct {
Name string
Value string
Labels map[string]string
}
type LabelRename struct {
Pattern regexp.Regexp
Name string
}
func (lua *LuaSession) do_Login(response string) error {
url_params := ""
if response != "" {
url_params = fmt.Sprintf("?response=%s&user=%s", response, lua.Username)
}
resp, err := http.Get(fmt.Sprintf("%s/login_sid.lua%s", lua.BaseUrl, url_params))
if err != nil {
return errors.New(fmt.Sprintf("Error calling login_sid.lua: %s", err.Error()))
}
defer resp.Body.Close()
dec := xml.NewDecoder(resp.Body)
err = dec.Decode(&lua.SessionInfo)
if err != nil {
return errors.New(fmt.Sprintf("Error decoding SessionInfo: %s", err.Error()))
}
return nil
}
func (lua *LuaSession) Login() error {
err := lua.do_Login("")
if err != nil {
return err
}
challenge := lua.SessionInfo.Challenge
if lua.SessionInfo.SID == "0000000000000000" && challenge != "" {
// no SID, but challenge so calc response
hash := utf16leMd5(fmt.Sprintf("%s-%s", challenge, lua.Password))
response := fmt.Sprintf("%s-%x", challenge, hash)
err := lua.do_Login(response)
if err != nil {
return err
}
}
sid := lua.SessionInfo.SID
if sid == "0000000000000000" || sid == "" {
return errors.New("LUA login failed - no SID received - check username and password")
}
lua.SID = sid
return nil
}
func (lua *LuaSession) LoadData(page LuaPage) ([]byte, error) {
data_url := fmt.Sprintf("%s/%s", lua.BaseUrl, page.Path)
callDone := false
var resp *http.Response
var err error
for !callDone {
// perform login if no SID or previous call failed with (403)
if lua.SID == "" || resp != nil {
err = lua.Login()
callDone = true // consider call done, since we tried login
if err != nil {
return nil, err
}
}
// send by UI for data.lua: xhr=1&sid=xxxxxxx&lang=de&page=energy&xhrId=all&no_sidrenew=
// but SID and page seem to be enough
params := "sid=" + lua.SID
if page.Params != "" {
params += "&" + page.Params
}
resp, err = http.Post(data_url, "application/x-www-form-urlencoded", bytes.NewBuffer([]byte(params)))
if err != nil {
return nil, err
}
defer resp.Body.Close()
if resp.StatusCode == http.StatusOK {
callDone = true
} else if resp.StatusCode == http.StatusForbidden && !callDone {
// we assume SID is expired, so retry login
} else {
return nil, errors.New(fmt.Sprintf("data.lua failed: %s", resp.Status))
}
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return nil, err
}
return body, nil
}
func getRenamedLabel(labelRenames *[]LabelRename, label string) string {
if labelRenames != nil {
for _, lblRen := range *labelRenames {
if lblRen.Pattern.MatchString(label) {
return lblRen.Name
}
}
}
return label
}
func GetMetrics(labelRenames *[]LabelRename, data map[string]interface{}, metricDef LuaMetricValueDefinition) ([]LuaMetricValue, error) {
var values []map[string]interface{}
var err error
if metricDef.Path != "" {
pathItems := strings.Split(metricDef.Path, ".")
values, err = _getValues(data, pathItems, "")
if err != nil {
return nil, err
}
} else {
values = make([]map[string]interface{}, 1)
values[0] = data
}
name := metricDef.Path
if name != "" {
name += "."
}
name += metricDef.Key
metrics := make([]LuaMetricValue, 0)
for _, valMap := range values {
value, exists := valMap[metricDef.Key]
if exists {
lmv := LuaMetricValue{
Name: name,
Value: toString(value),
Labels: make(map[string]string),
}
for _, l := range metricDef.Labels {
lv, exists := valMap[l]
if exists {
lmv.Labels[l] = getRenamedLabel(labelRenames, toString(lv))
}
}
metrics = append(metrics, lmv)
}
}
return metrics, nil
}
// from https://stackoverflow.com/questions/33710672/golang-encode-string-utf16-little-endian-and-hash-with-md5
func utf16leMd5(s string) []byte {
enc := unicode.UTF16(unicode.LittleEndian, unicode.IgnoreBOM).NewEncoder()
hasher := md5.New()
t := transform.NewWriter(hasher, enc)
t.Write([]byte(s))
return hasher.Sum(nil)
}
func ParseJSON(jsonData []byte) (map[string]interface{}, error) {
var data map[string]interface{}
// Unmarshal or Decode the JSON to the interface.
json.Unmarshal(jsonData, &data)
return data, nil
}
// helper for retrieving values from parsed JSON
func _getValues(data interface{}, pathItems []string, parentPath string) ([]map[string]interface{}, error) {
value := data
curPath := parentPath
for i, p := range pathItems {
switch vv := value.(type) {
case []interface{}:
if p == "*" {
values := make([]map[string]interface{}, 0, len(vv))
for index, u := range vv {
subvals, err := _getValues(u, pathItems[i+1:], fmt.Sprintf("%s.%d", curPath, index))
if err != nil {
return nil, err
}
values = append(values, subvals...)
}
return values, nil
} else {
index, err := strconv.Atoi(p)
if err != nil {
return nil, errors.New(fmt.Sprintf("item '%s' is an array, but path item '%s' is neither '*' nor a number", curPath, p))
}
if index < 0 {
// this is an index from the end of the values
index += len(vv)
}
if index >= 0 && index < len(vv) {
value = vv[index]
} else {
return nil, errors.New(fmt.Sprintf("index %d is invalid for array '%s' with length %d", index, curPath, len(vv)))
}
}
case map[string]interface{}:
var exits bool
value, exits = vv[p]
if !exits {
return nil, errors.New(fmt.Sprintf("key '%s' not existing in hash '%s'", p, curPath))
}
default:
return nil, errors.New(fmt.Sprintf("item '%s' is neither a hash or array", curPath))
}
if curPath == "" {
curPath = p
} else {
curPath += "." + p
}
}
vm, isType := value.(map[string]interface{})
if !isType {
return nil, errors.New(fmt.Sprintf("item '%s' is not a hash", curPath))
}
values := make([]map[string]interface{}, 1)
values[0] = vm
return values, nil
}
func toString(value interface{}) string {
// should we better check or simple convert everything ????
return fmt.Sprintf("%v", value)
}

298
main.go

@ -15,38 +15,43 @@ package main
// limitations under the License. // limitations under the License.
import ( import (
"bytes"
"encoding/json"
"errors"
"fmt" "fmt"
"io/ioutil"
"log" "log"
"net/url"
"net/http" "net/http"
"sync" "net/url"
"time" "regexp"
"encoding/json"
"io/ioutil"
"sort" "sort"
"bytes"
"errors"
"strings"
"strconv" "strconv"
"strings"
"sync"
"time"
"github.com/namsral/flag" "github.com/namsral/flag"
"github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promhttp" "github.com/prometheus/client_golang/prometheus/promhttp"
lua "github.com/sberk42/fritzbox_exporter/fritzbox_lua"
upnp "github.com/sberk42/fritzbox_exporter/fritzbox_upnp" upnp "github.com/sberk42/fritzbox_exporter/fritzbox_upnp"
) )
const serviceLoadRetryTime = 1 * time.Minute const serviceLoadRetryTime = 1 * time.Minute
var ( var (
flag_test = flag.Bool("test", false, "print all available metrics to stdout") flag_luacall = flag.Bool("testLua", false, "test LUA") // TODO cleanup once completed
flag_test = flag.Bool("test", false, "print all available metrics to stdout")
flag_collect = flag.Bool("collect", false, "print configured metrics to stdout and exit") flag_collect = flag.Bool("collect", false, "print configured metrics to stdout and exit")
flag_jsonout = flag.String("json-out", "", "store metrics also to JSON file when running test") flag_jsonout = flag.String("json-out", "", "store metrics also to JSON file when running test")
flag_addr = flag.String("listen-address", "127.0.0.1:9042", "The address to listen on for HTTP requests.") flag_addr = flag.String("listen-address", "127.0.0.1:9042", "The address to listen on for HTTP requests.")
flag_metrics_file = flag.String("metrics-file", "metrics.json", "The JSON file with the metric definitions.") flag_metrics_file = flag.String("metrics-file", "metrics.json", "The JSON file with the metric definitions.")
flag_gateway_url = flag.String("gateway-url", "http://fritz.box:49000", "The URL of the FRITZ!Box") flag_gateway_url = flag.String("gateway-url", "http://fritz.box:49000", "The URL of the FRITZ!Box")
flag_gateway_luaurl = flag.String("gateway-luaurl", "http://fritz.box", "The URL of the FRITZ!Box UI")
flag_gateway_username = flag.String("username", "", "The user for the FRITZ!Box UPnP service") flag_gateway_username = flag.String("username", "", "The user for the FRITZ!Box UPnP service")
flag_gateway_password = flag.String("password", "", "The password for the FRITZ!Box UPnP service") flag_gateway_password = flag.String("password", "", "The password for the FRITZ!Box UPnP service")
) )
@ -58,36 +63,35 @@ var (
}) })
) )
type JSON_PromDesc struct { type JSON_PromDesc struct {
FqName string `json:"fqName"` FqName string `json:"fqName"`
Help string `json:"help"` Help string `json:"help"`
VarLabels []string `json:"varLabels"` VarLabels []string `json:"varLabels"`
} }
type ActionArg struct { type ActionArg struct {
Name string `json:"Name"` Name string `json:"Name"`
IsIndex bool `json:"IsIndex"` IsIndex bool `json:"IsIndex"`
ProviderAction string `json:"ProviderAction"` ProviderAction string `json:"ProviderAction"`
Value string `json:"Value"` Value string `json:"Value"`
} }
type Metric struct { type Metric struct {
// initialized loading JSON // initialized loading JSON
Service string `json:"service"` Service string `json:"service"`
Action string `json:"action"` Action string `json:"action"`
ActionArgument *ActionArg `json:"actionArgument"` ActionArgument *ActionArg `json:"actionArgument"`
Result string `json:"result"` Result string `json:"result"`
OkValue string `json:"okValue"` OkValue string `json:"okValue"`
PromDesc JSON_PromDesc `json:"promDesc"` PromDesc JSON_PromDesc `json:"promDesc"`
PromType string `json:"promType"` PromType string `json:"promType"`
// initialized at startup // initialized at startup
Desc *prometheus.Desc Desc *prometheus.Desc
MetricType prometheus.ValueType MetricType prometheus.ValueType
} }
var metrics []*Metric; var metrics []*Metric
type FritzboxCollector struct { type FritzboxCollector struct {
Url string Url string
@ -101,9 +105,9 @@ type FritzboxCollector struct {
// simple ResponseWriter to collect output // simple ResponseWriter to collect output
type TestResponseWriter struct { type TestResponseWriter struct {
header http.Header header http.Header
statusCode int statusCode int
body bytes.Buffer body bytes.Buffer
} }
func (w *TestResponseWriter) Header() http.Header { func (w *TestResponseWriter) Header() http.Header {
@ -149,34 +153,34 @@ func (fc *FritzboxCollector) Describe(ch chan<- *prometheus.Desc) {
} }
func (fc *FritzboxCollector) ReportMetric(ch chan<- prometheus.Metric, m *Metric, result upnp.Result) { func (fc *FritzboxCollector) ReportMetric(ch chan<- prometheus.Metric, m *Metric, result upnp.Result) {
val, ok := result[m.Result] val, ok := result[m.Result]
if !ok { if !ok {
fmt.Printf("%s.%s has no result %s", m.Service, m.Action, m.Result) fmt.Printf("%s.%s has no result %s", m.Service, m.Action, m.Result)
collect_errors.Inc() collect_errors.Inc()
return return
} }
var floatval float64 var floatval float64
switch tval := val.(type) { switch tval := val.(type) {
case uint64: case uint64:
floatval = float64(tval) floatval = float64(tval)
case bool: case bool:
if tval { if tval {
floatval = 1 floatval = 1
} else { } else {
floatval = 0 floatval = 0
} }
case string: case string:
if tval == m.OkValue { if tval == m.OkValue {
floatval = 1 floatval = 1
} else { } else {
floatval = 0 floatval = 0
} }
default: default:
fmt.Println("unknown type", val) fmt.Println("unknown type", val)
collect_errors.Inc() collect_errors.Inc()
return return
} }
labels := make([]string, len(m.PromDesc.VarLabels)) labels := make([]string, len(m.PromDesc.VarLabels))
@ -188,54 +192,54 @@ func (fc *FritzboxCollector) ReportMetric(ch chan<- prometheus.Metric, m *Metric
if !ok { if !ok {
fmt.Printf("%s.%s has no resul for label %s", m.Service, m.Action, l) fmt.Printf("%s.%s has no resul for label %s", m.Service, m.Action, l)
lval = "" lval = ""
} }
// convert hostname and MAC tolower to avoid problems with labels // convert hostname and MAC tolower to avoid problems with labels
if(l == "HostName" || l == "MACAddress" ) { if l == "HostName" || l == "MACAddress" {
labels[i] = strings.ToLower(fmt.Sprintf("%v", lval)) labels[i] = strings.ToLower(fmt.Sprintf("%v", lval))
} else { } else {
labels[i] = fmt.Sprintf("%v", lval) labels[i] = fmt.Sprintf("%v", lval)
} }
} }
} }
ch <- prometheus.MustNewConstMetric( ch <- prometheus.MustNewConstMetric(
m.Desc, m.Desc,
m.MetricType, m.MetricType,
floatval, floatval,
labels...) labels...)
} }
func (fc *FritzboxCollector) GetActionResult(result_map map[string]upnp.Result, serviceType string, actionName string, actionArg *upnp.ActionArgument) (upnp.Result, error) { func (fc *FritzboxCollector) GetActionResult(result_map map[string]upnp.Result, serviceType string, actionName string, actionArg *upnp.ActionArgument) (upnp.Result, error) {
m_key := serviceType+"|"+actionName
// for calls with argument also add arguement name and value to key m_key := serviceType + "|" + actionName
// for calls with argument also add arguement name and value to key
if actionArg != nil { if actionArg != nil {
m_key += "|"+actionArg.Name+"|"+fmt.Sprintf("%v", actionArg.Value) m_key += "|" + actionArg.Name + "|" + fmt.Sprintf("%v", actionArg.Value)
} }
last_result := result_map[m_key]; last_result := result_map[m_key]
if last_result == nil { if last_result == nil {
service, ok := fc.Root.Services[serviceType] service, ok := fc.Root.Services[serviceType]
if !ok { if !ok {
return nil, errors.New(fmt.Sprintf("service %s not found", serviceType)) return nil, errors.New(fmt.Sprintf("service %s not found", serviceType))
} }
action, ok := service.Actions[actionName] action, ok := service.Actions[actionName]
if !ok { if !ok {
return nil, errors.New(fmt.Sprintf("action %s not found in service %s", actionName, serviceType)) return nil, errors.New(fmt.Sprintf("action %s not found in service %s", actionName, serviceType))
} }
var err error var err error
last_result, err = action.Call(actionArg); last_result, err = action.Call(actionArg)
if err != nil { if err != nil {
return nil, err return nil, err
} }
result_map[m_key]=last_result result_map[m_key] = last_result
} }
return last_result, nil return last_result, nil
@ -258,9 +262,9 @@ func (fc *FritzboxCollector) Collect(ch chan<- prometheus.Metric) {
var actArg *upnp.ActionArgument var actArg *upnp.ActionArgument
if m.ActionArgument != nil { if m.ActionArgument != nil {
aa := m.ActionArgument aa := m.ActionArgument
var value interface {} var value interface{}
value = aa.Value value = aa.Value
if aa.ProviderAction != "" { if aa.ProviderAction != "" {
provRes, err := fc.GetActionResult(result_map, m.Service, aa.ProviderAction, nil) provRes, err := fc.GetActionResult(result_map, m.Service, aa.ProviderAction, nil)
@ -269,9 +273,9 @@ func (fc *FritzboxCollector) Collect(ch chan<- prometheus.Metric) {
collect_errors.Inc() collect_errors.Inc()
continue continue
} }
var ok bool var ok bool
value, ok = provRes[aa.Value] // Value contains the result name for provider actions value, ok = provRes[aa.Value] // Value contains the result name for provider actions
if !ok { if !ok {
fmt.Printf("provider action %s for %s.%s has no result %s", m.Service, m.Action, aa.Value) fmt.Printf("provider action %s for %s.%s has no result %s", m.Service, m.Action, aa.Value)
collect_errors.Inc() collect_errors.Inc()
@ -285,34 +289,34 @@ func (fc *FritzboxCollector) Collect(ch chan<- prometheus.Metric) {
if err != nil { if err != nil {
fmt.Println(err.Error()) fmt.Println(err.Error())
collect_errors.Inc() collect_errors.Inc()
continue continue
} }
for i:=0; i<count; i++ { for i := 0; i < count; i++ {
actArg = &upnp.ActionArgument{Name: aa.Name, Value: i } actArg = &upnp.ActionArgument{Name: aa.Name, Value: i}
result, err := fc.GetActionResult(result_map, m.Service, m.Action, actArg) result, err := fc.GetActionResult(result_map, m.Service, m.Action, actArg)
if err != nil { if err != nil {
fmt.Println(err.Error()) fmt.Println(err.Error())
collect_errors.Inc() collect_errors.Inc()
continue continue
} }
fc.ReportMetric(ch, m, result) fc.ReportMetric(ch, m, result)
} }
continue continue
} else { } else {
actArg = &upnp.ActionArgument{Name: aa.Name, Value: value } actArg = &upnp.ActionArgument{Name: aa.Name, Value: value}
} }
} }
result, err := fc.GetActionResult(result_map, m.Service, m.Action, actArg) result, err := fc.GetActionResult(result_map, m.Service, m.Action, actArg)
if err != nil { if err != nil {
fmt.Println(err.Error()) fmt.Println(err.Error())
collect_errors.Inc() collect_errors.Inc()
continue continue
} }
fc.ReportMetric(ch, m, result) fc.ReportMetric(ch, m, result)
@ -324,7 +328,7 @@ func test() {
if err != nil { if err != nil {
panic(err) panic(err)
} }
var newEntry bool = false var newEntry bool = false
var json bytes.Buffer var json bytes.Buffer
json.WriteString("[\n") json.WriteString("[\n")
@ -337,7 +341,7 @@ func test() {
for _, k := range serviceKeys { for _, k := range serviceKeys {
s := root.Services[k] s := root.Services[k]
fmt.Printf("Service: %s (Url: %s)\n", k, s.ControlUrl) fmt.Printf("Service: %s (Url: %s)\n", k, s.ControlUrl)
actionKeys := []string{} actionKeys := []string{}
for l, _ := range s.Actions { for l, _ := range s.Actions {
actionKeys = append(actionKeys, l) actionKeys = append(actionKeys, l)
@ -350,22 +354,22 @@ func test() {
sv := arg.StateVariable sv := arg.StateVariable
fmt.Printf(" %s [%s] (%s, %s)\n", arg.RelatedStateVariable, arg.Direction, arg.Name, sv.DataType) fmt.Printf(" %s [%s] (%s, %s)\n", arg.RelatedStateVariable, arg.Direction, arg.Name, sv.DataType)
} }
if !a.IsGetOnly() { if !a.IsGetOnly() {
fmt.Printf(" %s - not calling, since arguments required or no output\n", a.Name) fmt.Printf(" %s - not calling, since arguments required or no output\n", a.Name)
continue continue
} }
// only create JSON for Get // only create JSON for Get
// TODO also create JSON templates for input actionParams // TODO also create JSON templates for input actionParams
for _, arg := range a.Arguments { for _, arg := range a.Arguments {
// create new json entry // create new json entry
if(newEntry) { if newEntry {
json.WriteString(",\n") json.WriteString(",\n")
} else { } else {
newEntry=true newEntry = true
} }
json.WriteString("\t{\n\t\t\"service\": \"") json.WriteString("\t{\n\t\t\"service\": \"")
json.WriteString(k) json.WriteString(k)
json.WriteString("\",\n\t\t\"action\": \"") json.WriteString("\",\n\t\t\"action\": \"")
@ -377,7 +381,7 @@ func test() {
fmt.Printf(" %s - calling - results: variable: value\n", a.Name) fmt.Printf(" %s - calling - results: variable: value\n", a.Name)
res, err := a.Call(nil) res, err := a.Call(nil)
if err != nil { if err != nil {
fmt.Printf(" FAILED:%s\n", err.Error()) fmt.Printf(" FAILED:%s\n", err.Error())
continue continue
@ -388,28 +392,90 @@ func test() {
} }
} }
} }
json.WriteString("\n]") json.WriteString("\n]")
if *flag_jsonout != "" { if *flag_jsonout != "" {
err := ioutil.WriteFile(*flag_jsonout, json.Bytes(), 0644) err := ioutil.WriteFile(*flag_jsonout, json.Bytes(), 0644)
if err != nil { if err != nil {
fmt.Printf("Failed writing JSON file '%s': %s\n", *flag_jsonout, err.Error()) fmt.Printf("Failed writing JSON file '%s': %s\n", *flag_jsonout, err.Error())
} }
}
}
func testLuaCall() {
var luaSession lua.LuaSession
luaSession.BaseUrl = *flag_gateway_luaurl
luaSession.Username = *flag_gateway_username
luaSession.Password = *flag_gateway_password
var jsonData []byte
var err error
fmt.Println("calling login")
page := lua.LuaPage{"data.lua", "page=energy"}
jsonData, err = luaSession.LoadData(page)
if err != nil {
fmt.Println(err.Error())
return
}
fmt.Println(fmt.Sprintf("JSON: %s", string(jsonData)))
data, err := lua.ParseJSON(jsonData)
if err != nil {
fmt.Println(err.Error())
return
}
labelRenames := make([]lua.LabelRename, 0)
labelRenames = addLabelRename(labelRenames, "(?i)prozessor", "CPU")
labelRenames = addLabelRename(labelRenames, "(?i)system", "System")
labelRenames = addLabelRename(labelRenames, "(?i)FON", "Phone")
labelRenames = addLabelRename(labelRenames, "(?i)WLAN", "WLAN")
labelRenames = addLabelRename(labelRenames, "(?i)USB", "USB")
pidMetric := lua.LuaMetricValueDefinition{"", "pid", nil}
powerMetric := lua.LuaMetricValueDefinition{"data.drain.*", "actPerc", []string{"name"}}
// fmt.Println(fmt.Sprintf("DATA: %v", data))
dumpMetric(&labelRenames, data, pidMetric)
dumpMetric(&labelRenames, data, powerMetric)
}
func dumpMetric(labelRenames *[]lua.LabelRename, data map[string]interface{}, metricDef lua.LuaMetricValueDefinition) {
metrics, err := lua.GetMetrics(labelRenames, data, metricDef)
if err != nil {
fmt.Println(err.Error())
return
} }
fmt.Println(fmt.Sprintf("Metrics: %v", metrics))
}
func addLabelRename(labelRenames []lua.LabelRename, pattern string, name string) []lua.LabelRename {
regex, err := regexp.Compile(pattern)
if err == nil {
return append(labelRenames, lua.LabelRename{*regex, name})
}
return labelRenames
} }
func getValueType(vt string) prometheus.ValueType { func getValueType(vt string) prometheus.ValueType {
switch vt { switch vt {
case "CounterValue": case "CounterValue":
return prometheus.CounterValue; return prometheus.CounterValue
case "GaugeValue": case "GaugeValue":
return prometheus.GaugeValue; return prometheus.GaugeValue
case "UntypedValue": case "UntypedValue":
return prometheus.UntypedValue; return prometheus.UntypedValue
} }
return prometheus.UntypedValue; return prometheus.UntypedValue
} }
func main() { func main() {
@ -426,7 +492,11 @@ func main() {
return return
} }
// read metrics if *flag_luacall {
testLuaCall()
return
}
// read metrics
jsonData, err := ioutil.ReadFile(*flag_metrics_file) jsonData, err := ioutil.ReadFile(*flag_metrics_file)
if err != nil { if err != nil {
fmt.Println("error reading metric file:", err) fmt.Println("error reading metric file:", err)
@ -442,24 +512,24 @@ func main() {
// init metrics // init metrics
for _, m := range metrics { for _, m := range metrics {
pd := m.PromDesc pd := m.PromDesc
// make labels lower case // make labels lower case
labels := make([]string, len(pd.VarLabels)) labels := make([]string, len(pd.VarLabels))
for i, l := range pd.VarLabels { for i, l := range pd.VarLabels {
labels[i] = strings.ToLower(l) labels[i] = strings.ToLower(l)
} }
m.Desc = prometheus.NewDesc(pd.FqName, pd.Help, labels, nil) m.Desc = prometheus.NewDesc(pd.FqName, pd.Help, labels, nil)
m.MetricType = getValueType(m.PromType) m.MetricType = getValueType(m.PromType)
} }
collector := &FritzboxCollector{ collector := &FritzboxCollector{
Url: *flag_gateway_url, Url: *flag_gateway_url,
Gateway: u.Hostname(), Gateway: u.Hostname(),
Username: *flag_gateway_username, Username: *flag_gateway_username,
Password: *flag_gateway_password, Password: *flag_gateway_password,
} }
if *flag_collect { if *flag_collect {
collector.LoadServices() collector.LoadServices()
@ -471,13 +541,13 @@ func main() {
// simulate HTTP request without starting actual http server // simulate HTTP request without starting actual http server
writer := TestResponseWriter{header: http.Header{}} writer := TestResponseWriter{header: http.Header{}}
request := http.Request{} request := http.Request{}
promhttp.Handler().ServeHTTP(&writer, &request) promhttp.Handler().ServeHTTP(&writer, &request)
fmt.Println(writer.String()) fmt.Println(writer.String())
return return
} }
go collector.LoadServices() go collector.LoadServices()
prometheus.MustRegister(collector) prometheus.MustRegister(collector)

Loading…
Cancel
Save