2013-07-17 15:45:01 +00:00
|
|
|
// Copyright 2013 Prometheus Team
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
package web
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"html/template"
|
|
|
|
"net/http"
|
2014-10-27 13:53:31 +00:00
|
|
|
_ "net/http/pprof"
|
2015-05-23 10:32:24 +00:00
|
|
|
"strings"
|
2013-07-17 15:45:01 +00:00
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
2015-05-23 20:34:45 +00:00
|
|
|
"github.com/prometheus/log"
|
2013-07-17 15:45:01 +00:00
|
|
|
|
2013-08-05 09:49:56 +00:00
|
|
|
"github.com/prometheus/alertmanager/web/api"
|
|
|
|
"github.com/prometheus/alertmanager/web/blob"
|
2013-07-17 15:45:01 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Commandline flags.
|
|
|
|
var (
|
2015-02-08 22:20:09 +00:00
|
|
|
listenAddress = flag.String("web.listen-address", ":9093", "Address to listen on for the web interface and API.")
|
|
|
|
useLocalAssets = flag.Bool("web.use-local-assets", false, "Serve assets and templates from local files instead of from the binary.")
|
2013-07-17 15:45:01 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type WebService struct {
|
|
|
|
AlertManagerService *api.AlertManagerService
|
2013-07-18 12:49:37 +00:00
|
|
|
AlertsHandler *AlertsHandler
|
|
|
|
SilencesHandler *SilencesHandler
|
|
|
|
StatusHandler *StatusHandler
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 19:39:59 +00:00
|
|
|
func (w WebService) ServeForever(pathPrefix string) error {
|
2013-07-17 15:45:01 +00:00
|
|
|
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"favicon.ico", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2013-07-17 15:45:01 +00:00
|
|
|
http.Error(w, "", 404)
|
|
|
|
}))
|
|
|
|
|
2015-03-23 18:39:31 +00:00
|
|
|
http.HandleFunc("/", prometheus.InstrumentHandlerFunc("index", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
// The "/" pattern matches everything, so we need to check
|
|
|
|
// that we're at the root here.
|
2015-04-29 19:39:59 +00:00
|
|
|
if req.URL.Path == pathPrefix {
|
|
|
|
w.AlertsHandler.ServeHTTP(rw, req)
|
2015-05-23 10:32:24 +00:00
|
|
|
} else if req.URL.Path == strings.TrimRight(pathPrefix, "/") {
|
2015-04-29 19:39:59 +00:00
|
|
|
http.Redirect(rw, req, pathPrefix, http.StatusFound)
|
2015-05-23 10:32:24 +00:00
|
|
|
} else if !strings.HasPrefix(req.URL.Path, pathPrefix) {
|
|
|
|
// We're running under a prefix but the user requested something
|
|
|
|
// outside of it. Let's see if this page exists under the prefix.
|
|
|
|
http.Redirect(rw, req, pathPrefix+strings.TrimLeft(req.URL.Path, "/"), http.StatusFound)
|
2015-04-29 19:39:59 +00:00
|
|
|
} else {
|
2015-03-23 18:39:31 +00:00
|
|
|
http.NotFound(rw, req)
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"alerts", prometheus.InstrumentHandler("alerts", w.AlertsHandler))
|
|
|
|
http.Handle(pathPrefix+"silences", prometheus.InstrumentHandler("silences", w.SilencesHandler))
|
|
|
|
http.Handle(pathPrefix+"status", prometheus.InstrumentHandler("status", w.StatusHandler))
|
2013-07-17 15:45:01 +00:00
|
|
|
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"metrics", prometheus.Handler())
|
2013-07-17 15:45:01 +00:00
|
|
|
if *useLocalAssets {
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"static/", http.StripPrefix(pathPrefix+"static/", http.FileServer(http.Dir("web/static"))))
|
2013-07-17 15:45:01 +00:00
|
|
|
} else {
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"static/", http.StripPrefix(pathPrefix+"static/", new(blob.Handler)))
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
2015-05-23 10:32:24 +00:00
|
|
|
http.Handle(pathPrefix+"api/", w.AlertManagerService.Handler())
|
2013-07-17 15:45:01 +00:00
|
|
|
|
2015-05-23 20:34:45 +00:00
|
|
|
log.Info("listening on ", *listenAddress)
|
2013-07-17 15:45:01 +00:00
|
|
|
|
2014-10-27 13:53:31 +00:00
|
|
|
return http.ListenAndServe(*listenAddress, nil)
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 19:39:59 +00:00
|
|
|
func getLocalTemplate(name string, pathPrefix string) (*template.Template, error) {
|
2013-07-22 09:11:18 +00:00
|
|
|
t := template.New("_base.html")
|
|
|
|
t.Funcs(webHelpers)
|
2015-04-29 19:39:59 +00:00
|
|
|
t.Funcs(template.FuncMap{"pathPrefix": func() string { return pathPrefix }})
|
|
|
|
|
2013-07-22 09:11:18 +00:00
|
|
|
return t.ParseFiles(
|
2013-07-17 15:45:01 +00:00
|
|
|
"web/templates/_base.html",
|
|
|
|
fmt.Sprintf("web/templates/%s.html", name),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-04-29 19:39:59 +00:00
|
|
|
func getEmbeddedTemplate(name string, pathPrefix string) (*template.Template, error) {
|
2013-07-22 09:11:18 +00:00
|
|
|
t := template.New("_base.html")
|
|
|
|
t.Funcs(webHelpers)
|
2015-04-29 19:39:59 +00:00
|
|
|
t.Funcs(template.FuncMap{"pathPrefix": func() string { return pathPrefix }})
|
2013-07-17 15:45:01 +00:00
|
|
|
|
|
|
|
file, err := blob.GetFile(blob.TemplateFiles, "_base.html")
|
|
|
|
if err != nil {
|
2015-05-23 20:34:45 +00:00
|
|
|
log.Error("Could not read base template: ", err)
|
2013-07-17 15:45:01 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t.Parse(string(file))
|
|
|
|
|
|
|
|
file, err = blob.GetFile(blob.TemplateFiles, name+".html")
|
|
|
|
if err != nil {
|
2015-05-23 20:34:45 +00:00
|
|
|
log.Errorf("Could not read %s template: %s", name, err)
|
2013-07-17 15:45:01 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t.Parse(string(file))
|
|
|
|
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
2015-04-29 19:39:59 +00:00
|
|
|
func getTemplate(name string, pathPrefix string) (t *template.Template, err error) {
|
2013-07-17 15:45:01 +00:00
|
|
|
if *useLocalAssets {
|
2015-04-29 19:39:59 +00:00
|
|
|
t, err = getLocalTemplate(name, pathPrefix)
|
2013-07-17 15:45:01 +00:00
|
|
|
} else {
|
2015-04-29 19:39:59 +00:00
|
|
|
t, err = getEmbeddedTemplate(name, pathPrefix)
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2013-07-22 09:11:18 +00:00
|
|
|
return nil, err
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
|
2013-07-22 09:11:18 +00:00
|
|
|
return t, nil
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 19:39:59 +00:00
|
|
|
func executeTemplate(w http.ResponseWriter, name string, data interface{}, pathPrefix string) {
|
|
|
|
tpl, err := getTemplate(name, pathPrefix)
|
2013-07-17 15:45:01 +00:00
|
|
|
if err != nil {
|
2015-05-23 20:34:45 +00:00
|
|
|
log.Error("Error preparing layout template: ", err)
|
2013-07-17 15:45:01 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
err = tpl.Execute(w, data)
|
|
|
|
if err != nil {
|
2015-05-23 20:34:45 +00:00
|
|
|
log.Error("Error executing template: ", err)
|
2013-07-17 15:45:01 +00:00
|
|
|
}
|
|
|
|
}
|