Compare commits

..

8 Commits

Author SHA1 Message Date
e4a372e29a Merge pull request 'Added Config file' (#10) from config-file into master
All checks were successful
continuous-integration/drone/push Build is passing
Reviewed-on: #10
2021-02-11 19:58:27 +00:00
Stefan Hoffmann
831ce76e79 use global variable for config
All checks were successful
continuous-integration/drone/pr Build is passing
continuous-integration/drone/push Build is passing
2021-02-10 19:31:31 +01:00
e292e304c3
made callUnifiAPI, processAPIs, processUNMSAPI functions for config type
All checks were successful
continuous-integration/drone/pr Build is passing
continuous-integration/drone/push Build is passing
2021-02-07 21:30:40 +01:00
e98b5f1267 Merge branch 'master' into config-file
All checks were successful
continuous-integration/drone/pr Build is passing
continuous-integration/drone/push Build is passing
2021-02-07 19:02:09 +00:00
Stefan Hoffmann
861d1c5d5e Added enbled flag to APIs in config file
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/pr Build is passing
2021-02-07 10:46:39 +01:00
Stefan Hoffmann
e133a1e95d Ignore config files!
All checks were successful
continuous-integration/drone/push Build is passing
2021-02-06 22:45:10 +01:00
Stefan Hoffmann
a1bb128291 fixed devices url
All checks were successful
continuous-integration/drone/push Build is passing
2021-02-06 22:44:05 +01:00
Stefan Hoffmann
737e4268fc added first support for config file
All checks were successful
continuous-integration/drone/push Build is passing
2021-02-06 22:32:24 +01:00
4 changed files with 109 additions and 75 deletions

1
.gitignore vendored
View File

@ -14,3 +14,4 @@
output/* output/*
ubnt-freifunk-map-api ubnt-freifunk-map-api
config.json

15
example.config.json Normal file
View File

@ -0,0 +1,15 @@
{
"unms": {
"enabled": true,
"unmsAPIUrl": "https://unifi.freifunk-troisdorf.de/v2.1",
"APItoken": "API TOKEN",
"devicesURL": "https://git.freifunk-rhein-sieg.net/Freifunk-Troisdorf/ubnt-freifunk-map-api/raw/branch/master/"
},
"unifi": {
"enabled": true,
"APIUrl": "https://unifi.freifunk-troisdorf.de:8443",
"user": "APIuser",
"password": "PASSWORD",
"ucDevicesURL": "https://git.freifunk-rhein-sieg.net/Freifunk-Troisdorf/ubnt-freifunk-map-api/raw/branch/master/"
}
}

120
main.go
View File

@ -10,6 +10,7 @@ import (
"log" "log"
"net/http" "net/http"
"net/http/cookiejar" "net/http/cookiejar"
"os"
"strconv" "strconv"
"strings" "strings"
"time" "time"
@ -17,20 +18,15 @@ import (
_ "git.nils.zone/nils/prettify" _ "git.nils.zone/nils/prettify"
) )
// types
const ( const (
baseURL = "https://unifi.freifunk-troisdorf.de/v2.1" iso8601 = "2006-01-02T15:04:05-0700"
ucBaseURL = "https://unifi.freifunk-troisdorf.de:8443"
iso8601 = "2006-01-02T15:04:05-0700"
) )
// flags // flags
var token = flag.String("token", "", "Defines the x-auth-token") var configPath = flag.String("configPath", "config.json", "Path to config.json")
var ucUser = flag.String("ucUser", "", "Defines the Unifi API User")
var ucPass = flag.String("ucPass", "", "Defines the Unifi API Password")
var version = "development" var version = "development"
var delay time.Duration = 60 * time.Second var delay time.Duration = 60 * time.Second
var conf = loadconfig(*configPath)
func main() { func main() {
log.Printf("starting version %s...\n", version) log.Printf("starting version %s...\n", version)
@ -38,16 +34,9 @@ func main() {
flag.Parse() flag.Parse()
// check if flags are set // check if flags are set
if *token == "" { if *configPath == "" {
log.Fatalln("Please specify an API token via the flag '-token'") log.Fatalln("Please specify path to config.json flag '-configPath'")
} }
if *ucPass == "" {
log.Fatalln("Please specify an API Password via the flag '-ucPass'")
}
if *ucUser == "" {
log.Fatalln("Please specify an API User via the flag '-ucUser'")
}
// start API processing (runs in a loop) // start API processing (runs in a loop)
go processAPIs() go processAPIs()
@ -55,36 +44,15 @@ func main() {
serveJSON() serveJSON()
} }
//switch Unifi AP Mod IDs to Names func loadconfig(file string) config {
func lookupModels(model string) string { var config config
switch model { configFile, err := os.Open(file)
case "BZ2", "U2S48", "U2Sv2": if err != nil {
return "Unifi AP" log.Fatalln(err)
case "BZ2LR", "U2L48", "U2Lv2":
return "UniFi AP-LR"
case "U7E", "U7Ev2":
return "UniFi AP-AC"
case "U7HD", "U7SHD":
return "UniFi AP-HD"
case "UXSDM":
return "UniFi AP-BaseStationXG"
case "UCMSH":
return "AP-MeshXG"
case "U7MP":
return "AP-AC-Mesh-Pro"
case "U7LR":
return "UniFi AP-AC-LR"
case "U7LT":
return "UniFi AP-AC-Lite"
case "U7P":
return "UniFi AP-Pro"
case "U7MSH":
return "UniFi AP-AC-Mesh"
case "U7PG2":
return "UniFi AP-AC-Pro"
default:
return "Unifi Gerät"
} }
jsonParse := json.NewDecoder(configFile)
jsonParse.Decode(&config)
return config
} }
//int to bool converter //int to bool converter
@ -100,23 +68,23 @@ func processUcAPIs() ([]node, []link) {
//get list of Unifi devices to display //get list of Unifi devices to display
var nodes []node var nodes []node
var links []link var links []link
d, err := getDevices("ucDevices.json") d, err := getDevices(conf.Unifi.UCDevicesURL)
if err != nil { if err != nil {
log.Fatalln(err) log.Fatalln(err)
} }
//call Unifi Controller //call Unifi Controller
ucAPI := newAPI(*ucUser, *ucPass, ucBaseURL) ucAPI := newAPI(conf.Unifi.User, conf.Unifi.Password, conf.Unifi.APIURL)
//login //login
ucAPI.ucLogin() ucAPI.ucLogin()
//get all Sites from Controller //get all Sites from Controller
sites, err := ucAPI.ucGetSites() sites, err := ucAPI.ucGetSites()
if err != nil { if err != nil {
panic(err) log.Fatalln(err)
} }
//get all devices in all sites //get all devices in all sites
devices, err := ucAPI.ucGetDevices(sites) devices, err := ucAPI.ucGetDevices(sites)
if err != nil { if err != nil {
panic(err) log.Fatalln(err)
} }
//build nodes struct //build nodes struct
@ -135,11 +103,11 @@ func processUcAPIs() ([]node, []link) {
load, err := strconv.ParseFloat(currentDevice.Sysstats.CPU, 64) load, err := strconv.ParseFloat(currentDevice.Sysstats.CPU, 64)
if err != nil { if err != nil {
panic(err) log.Fatalln(err)
} }
mem, err := strconv.ParseFloat(currentDevice.Sysstats.Memory, 64) mem, err := strconv.ParseFloat(currentDevice.Sysstats.Memory, 64)
if err != nil { if err != nil {
panic(err) log.Fatalln(err)
} }
nodes = append(nodes, node{ nodes = append(nodes, node{
@ -185,7 +153,7 @@ func processUNMSAPI() ([]node, []link) {
var links []link var links []link
var nodes []node var nodes []node
d, err := getDevices("devices.json") d, err := getDevices(conf.Unms.DevicesURL)
if err != nil { if err != nil {
log.Fatalln(err) log.Fatalln(err)
} }
@ -276,12 +244,16 @@ func processAPIs() {
var nodes []node var nodes []node
var links []link var links []link
unmsNodes, unmsLinks := processUNMSAPI() if conf.Unms.Enabled == true {
ucNodes, ucLinks := processUcAPIs() unmsNodes, unmsLinks := processUNMSAPI()
nodes = append(nodes, unmsNodes...) nodes = append(nodes, unmsNodes...)
nodes = append(nodes, ucNodes...) links = append(links, unmsLinks...)
links = append(links, unmsLinks...) }
links = append(links, ucLinks...) if conf.Unifi.Enabled == true {
ucNodes, ucLinks := processUcAPIs()
nodes = append(nodes, ucNodes...)
links = append(links, ucLinks...)
}
// assemble final struct // assemble final struct
o := output{ o := output{
@ -311,43 +283,41 @@ func getFile(url string) []byte {
return byteValue return byteValue
} }
func getDevices(file string) (devices, error) { func getDevices(url string) (devices, error) {
// get devices from JSON file // get devices from JSON file
jsonFile := getFile("https://git.freifunk-rhein-sieg.net/Freifunk-Troisdorf/ubnt-freifunk-map-api/raw/branch/master/" + file) jsonFile := getFile(url)
// read file to bytes // read file to bytes
// variable for d // variable for d
var d devices var d devices
// unmarshal to struct // unmarshal to struct
json.Unmarshal(jsonFile, &d) err := json.Unmarshal(jsonFile, &d)
if err != nil {
fmt.Println("can´t get devices file from " + url)
log.Fatal(err)
}
return d, nil return d, nil
} }
func callUnifiAPI(url string, i interface{}) error { func callUnifiAPI(url string, i interface{}) error {
request, err := http.NewRequest(http.MethodGet, baseURL+url, nil) request, err := http.NewRequest(http.MethodGet, conf.Unms.UnmsAPIURL+url, nil)
if err != nil { if err != nil {
return errors.New(fmt.Sprint("can't set request", baseURL+url)) return errors.New(fmt.Sprint("can't set request", conf.Unms.UnmsAPIURL+url))
} }
request.Header.Set("x-auth-token", *token) request.Header.Set("x-auth-token", conf.Unms.APItoken)
client := &http.Client{} client := &http.Client{}
response, err := client.Do(request) response, err := client.Do(request)
if err != nil { if err != nil {
return fmt.Errorf("can't get request %s with x-auth-token %s", baseURL+url, *token) return fmt.Errorf("can't get request %s with x-auth-token %s", conf.Unms.UnmsAPIURL+url, conf.Unms.APItoken)
}
if response.StatusCode != 200 {
log.Fatalln("Can´t call UNMS API, check token and URL. HTTP Status: ", response.StatusCode)
} }
data, err := ioutil.ReadAll(response.Body) data, err := ioutil.ReadAll(response.Body)
defer response.Body.Close() defer response.Body.Close()
if err != nil { if err != nil {
return fmt.Errorf("can't read response body: %+v", response.Body) return fmt.Errorf("can't read response body: %+v", response.Body)
} }
// try to fetch errors
var a apiResponse
json.Unmarshal(data, &a)
if a.StatusCode != 0 {
return fmt.Errorf("got following errorcode from API: %d %s %s", a.StatusCode, a.Error, a.Message)
}
// no error occurred, unmarshal to struct // no error occurred, unmarshal to struct
json.Unmarshal(data, &i) json.Unmarshal(data, &i)
return nil return nil

View File

@ -10,6 +10,22 @@ import (
"time" "time"
) )
type config struct {
Unms struct {
Enabled bool `json:"enabled"`
UnmsAPIURL string `json:"unmsAPIUrl"`
APItoken string `json:"APItoken"`
DevicesURL string `json:"devicesURL"`
} `json:"unms"`
Unifi struct {
Enabled bool `json:"enabled"`
APIURL string `json:"APIUrl"`
User string `json:"user"`
Password string `json:"password"`
UCDevicesURL string `json:"ucDevicesURL"`
} `json:"unifi"`
}
type device struct { type device struct {
Name string `json:"name"` Name string `json:"name"`
MAC string `json:"mac"` MAC string `json:"mac"`
@ -178,3 +194,35 @@ type ucAPIData struct {
baseURL string baseURL string
client *http.Client client *http.Client
} }
//switch Unifi AP Mod IDs to Names
func lookupModels(model string) string {
switch model {
case "BZ2", "U2S48", "U2Sv2":
return "Unifi AP"
case "BZ2LR", "U2L48", "U2Lv2":
return "UniFi AP-LR"
case "U7E", "U7Ev2":
return "UniFi AP-AC"
case "U7HD", "U7SHD":
return "UniFi AP-HD"
case "UXSDM":
return "UniFi AP-BaseStationXG"
case "UCMSH":
return "AP-MeshXG"
case "U7MP":
return "AP-AC-Mesh-Pro"
case "U7LR":
return "UniFi AP-AC-LR"
case "U7LT":
return "UniFi AP-AC-Lite"
case "U7P":
return "UniFi AP-Pro"
case "U7MSH":
return "UniFi AP-AC-Mesh"
case "U7PG2":
return "UniFi AP-AC-Pro"
default:
return "Unifi Gerät"
}
}