tplinkcmd/cmd/tpapi/httpTemplates.go

262 lines
6.9 KiB
Go

package main
import (
"fmt"
"io"
"log"
"net"
"strconv"
"strings"
"encoding/json"
"net/http"
)
func tmpltError(w http.ResponseWriter, s int, m string) {
var (
output []byte
o = struct {
Error bool `json:"error" yaml:"error"`
ErrorMsg string `json:"errorMessage" yaml:"errorMessage"`
}{
Error: true,
ErrorMsg: m,
}
err error
)
w.Header().Add("Content-Type", "application/json")
output, err = json.MarshalIndent(o, "", " ")
if err != nil {
log.Printf("[TRACE] Unable to marshal error message: %v.", err)
}
w.WriteHeader(s)
w.Write(output) //nolint:errcheck
}
func tmpltWebRoot(w http.ResponseWriter) {
o := struct {
Application string `json:"application" yaml:"application"`
Description string `json:"description" yaml:"description"`
Version string `json:"version" yaml:"version"`
}{
Application: "TP-Link Web API",
Description: "API for automated calls to TP-Link devices",
Version: "v1.0.0",
}
w.Header().Add("Content-Type", "application/json")
output, err := json.MarshalIndent(o, "", " ")
if err != nil {
log.Printf("[TRACE] Unable to marshal error message: %v.", err)
}
w.Write(output) //nolint:errcheck
}
func tmpltHealthCheck(w http.ResponseWriter) {
o := struct {
WebServer bool `json:"webServerActive" yaml:"webServerActive"`
Status string `json:"status" yaml:"status"`
}{
WebServer: true,
Status: "healthy",
}
output, err := json.MarshalIndent(o, "", " ")
if err != nil {
log.Printf("[TRACE] Unable to marshal status message: %v.", err)
}
w.Header().Add("Content-Type", "application/json")
w.Write(output) //nolint:errcheck
}
func tmpltTPlinkGet(w http.ResponseWriter, r *http.Request) {
keys := r.URL.Query()
if len(keys) == 0 {
log.Printf("[INFO] Required parameters missing: no host, state, or deviceid was provided.\n")
tmpltError(w, http.StatusBadRequest, "Required parameters missing: no parameter for host, state, or deviceid was provided.")
return
}
var (
host string
hostSet bool
state bool
stateSet bool
id int
)
for k, v := range keys {
switch strings.ToLower(k) {
case "host":
if len(v) != 1 {
log.Printf("[INFO] Host was defined multiple times.\n")
tmpltError(w, http.StatusBadRequest, "Required parameter host was defined multiple times.")
return
}
_, err := net.LookupHost(v[0])
if err != nil {
log.Printf("[INFO] Unable to resolve provided hostname: %s\n", v[0])
tmpltError(w, http.StatusBadRequest, "Unable to resolve provided hostname.")
return
}
hostSet = true
host = v[0]
case "state":
if len(v) != 1 {
log.Printf("[INFO] State was defined multiple times.\n")
tmpltError(w, http.StatusBadRequest, "Required parameter state was defined multiple times.")
return
}
s, err := strconv.ParseBool(v[0])
if err != nil {
log.Printf("[INFO] State is not boolean: %s\n", v[0])
tmpltError(w, http.StatusBadRequest, "Required parameter state is not boolean.")
return
}
stateSet = true
state = s
case "deviceid":
if len(v) != 1 {
log.Printf("[INFO] Deviceid was defined multiple times.\n")
tmpltError(w, http.StatusBadRequest, "Required parameter device was defined multiple times.")
return
}
d, err := strconv.Atoi(v[0])
if err != nil {
log.Printf("[INFO] Deviceid is not an integer: %s\n", v[0])
tmpltError(w, http.StatusBadRequest, "Required parameter deviceid is not an integer.")
return
}
id = d
}
}
if !hostSet {
log.Printf("[INFO] Required parameter missing: no host was provided.\n")
tmpltError(w, http.StatusBadRequest, "Required parameters missing: no parameter for host provided.")
return
}
if !stateSet {
log.Printf("[INFO] Required parameter missing: no state was provided.\n")
tmpltError(w, http.StatusBadRequest, "Required parameters missing: no parameter for state provided.")
return
}
fs, err := switchDevice(host, id, state)
if err != nil {
log.Printf("[INFO] %v\n", err)
tmpltError(w, http.StatusBadRequest, fmt.Sprintf("[INFO] %v\n", err))
return
}
o := struct {
Status string `json:"status" yaml:"status"`
Host string `json:"host" yaml:"host"`
DeviceID int `json:"device_id" yaml:"deviceID"`
}{
Status: fs,
Host: host,
DeviceID: id,
}
w.Header().Add("Content-Type", "application/json")
output, err := json.MarshalIndent(o, "", " ")
if err != nil {
log.Printf("[TRACE] Unable to marshal error message: %v.", err)
}
w.Write(output) //nolint:errcheck
}
func tmpltTPlinkPost(w http.ResponseWriter, r *http.Request) {
type outputStruct struct {
Status string `json:"status" yaml:"status"`
Host string `json:"host" yaml:"host"`
DeviceID int `json:"device_id" yaml:"deviceID"`
}
var (
payload alertmanagerStruct
o []outputStruct
)
// read body
defer r.Body.Close()
b, err := io.ReadAll(r.Body)
if err != nil {
log.Printf("[DEBUG] Unable to read post body from request: %v\n", err)
tmpltError(w, http.StatusInternalServerError, fmt.Sprintf("Unable to read post body from request: %v", err))
return
}
// unmarshal body
err = json.Unmarshal(b, &payload)
if err != nil {
log.Printf("[TRACE] Data received:\n--START--\n%s\n--END--\n", string(b))
log.Printf("[INFO] Unable to parse payload from API call: %v\n", err)
tmpltError(w, http.StatusInternalServerError, fmt.Sprintf("Unable to read info from device: %v\n", err))
return
}
for _, alert := range payload.Alerts {
var (
host string
id int
validKey bool
fs string
)
// validate keys exist
if host, validKey = alert.Annotations["host"]; !validKey {
log.Printf("[DEBUG] Unable to find host key annotation.")
tmpltError(w, http.StatusInternalServerError, "Unable to find host key annotation.")
return
}
if _, validKey = alert.Annotations["id"]; !validKey {
log.Printf("[DEBUG] Unable to find device id key annotation.")
tmpltError(w, http.StatusInternalServerError, "Unable to find device id key annotation.")
return
}
// convert annotation string to integer
id, err := strconv.Atoi(alert.Annotations["id"])
if err != nil {
log.Printf("[DEBUG] Unable to convert string to integer for provided device id: %v\n", err)
tmpltError(w, http.StatusInternalServerError, fmt.Sprintf("Unable to convert string to integer for provided device id: %v\n", err))
return
}
switch strings.ToLower(alert.Status) {
case "firing":
fs, err = switchDevice(host, id, true)
case "resolved":
fs, err = switchDevice(host, id, false)
default:
}
if err != nil {
log.Printf("[INFO] %v\n", err)
tmpltError(w, http.StatusBadRequest, fmt.Sprintf("[INFO] %v\n", err))
return
}
o = append(o, outputStruct{
Status: fs,
Host: host,
DeviceID: id,
})
}
w.Header().Add("Content-Type", "application/json")
output, err := json.MarshalIndent(o, "", " ")
if err != nil {
log.Printf("[TRACE] Unable to marshal error message: %v.", err)
}
w.Write(output) //nolint:errcheck
}