package main

import (
	"fmt"
	"io/ioutil"
	"net/http"
	"os"
	"time"

	"gopkg.in/alecthomas/kingpin.v2"
	"gopkg.in/yaml.v2"

	hue "github.com/collinux/gohue"
	"github.com/prometheus/client_golang/prometheus"
	"github.com/prometheus/client_golang/prometheus/promhttp"
	"github.com/prometheus/common/log"
	"github.com/prometheus/common/version"
)

const namespace = "hue"

var (
	app         = kingpin.New("hue_exporter", "A Prometheus exporter for Philips Hue.")
	showVersion = app.Flag("version", "Print the version and exit.").Short('V').Bool()
	run         = app.Command("run", "Run the exporter.").Default()
	// TODO: update https://github.com/prometheus/prometheus/wiki/Default-port-allocations
	addr     = run.Flag("listen.address", "The address to listen on for HTTP requests.").Short('l').Default(":9366").TCP()
	config   = run.Flag("config.file", "The config file to use.").Short('c').Default("hue_exporter.yml").ExistingFile()
	generate = app.Command("generate", "Generate configuration for Hue exporter.")
	output   = generate.Flag("output.file", "The output file to use.").Short('o').Default("hue_exporter.yml").String()
)

type Config struct {
	IPAddr       string `yaml:"ip_address"`
	APIKey       string `yaml:"api_key"`
	SensorConfig struct {
		IgnoreTypes []string `yaml:"ignore_types"`
		MatchNames  bool     `yaml:"match_names"`
	} `yaml:"sensors"`
}

// Bridge is an interface for the bridge struct from Collinux/gohue to allow stubbing in tests
type Bridge interface {
	Login(string) error
	GetAllSensors() ([]hue.Sensor, error)
	GetAllLights() ([]hue.Light, error)
	GetAllGroups() ([]hue.Group, error)
}

func readConfig(raw []byte, cfg *Config) {
	err := yaml.Unmarshal(raw, cfg)
	if err != nil {
		log.Fatalf("Error parsing config file: %v\n", err)
	}
}

func newBridge(ipAddr string) Bridge {
	bridge, err := hue.NewBridge(ipAddr)
	if err != nil {
		log.Fatalf("Error connecting to Hue bridge at %v: %v\n", ipAddr, err)
	}
	return bridge
}

func setupPrometheus(bridge Bridge, cfg *Config) {
	err := bridge.Login((*cfg).APIKey)
	if err != nil {
		log.Fatalf("Error authenticating with Hue bridge at %v: %v\n", (*cfg).IPAddr, err)
	}

	prometheus.MustRegister(NewGroupCollector(namespace, bridge))
	prometheus.MustRegister(NewLightCollector(namespace, bridge))
	prometheus.MustRegister(NewSensorCollector(namespace, bridge, (*cfg).SensorConfig.IgnoreTypes, (*cfg).SensorConfig.MatchNames))
	prometheus.MustRegister(version.NewCollector("hue_exporter"))
}

func listen() {
	http.Handle("/metrics", promhttp.Handler())
	http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
		w.Write([]byte(`<html>
            <head><title>Hue Exporter</title></head>
            <body>
            <h1>Hue Exporter</h1>
            <p><a href="/metrics">Metrics</a></p>
            </body>
            </html>`))
	})
	srv := &http.Server{
		Addr:         (*addr).String(),
		ReadTimeout:  5 * time.Second,
		WriteTimeout: 10 * time.Second,
		ErrorLog:     log.NewErrorLogger(),
	}
	log.Infoln("Listening on", (*addr).String())
	log.Fatal(srv.ListenAndServe())
}

func runServer() {
	var cfg Config

	raw, err := ioutil.ReadFile(*config)
	if err != nil {
		log.Fatalf("Error reading config file: %v\n", err)
	}
	readConfig(raw, &cfg)
	bridge := newBridge(cfg.IPAddr)
	setupPrometheus(bridge, &cfg)
	listen()
}

func main() {
	log.AddFlags(app)
	command := kingpin.MustParse(app.Parse(os.Args[1:]))
	if *showVersion {
		fmt.Fprintln(os.Stdout, version.Print("hue_exporter"))
	} else {
		log.Infoln("Starting hue_exporter", version.Info())
		log.Infoln("Build context", version.BuildContext())
		switch command {
		case run.FullCommand():
			runServer()
		case generate.FullCommand():
			generateConfig(output)
		}
	}
}