2015-01-21 11:07:45 -08:00
|
|
|
// Copyright 2013 The Prometheus Authors
|
2013-02-08 05:49:55 -08:00
|
|
|
// 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 (
|
2013-02-08 06:38:50 -08:00
|
|
|
"flag"
|
2013-04-05 04:41:52 -07:00
|
|
|
"fmt"
|
2013-03-27 09:40:01 -07:00
|
|
|
"html/template"
|
2014-07-29 09:28:48 -07:00
|
|
|
"io/ioutil"
|
2013-08-09 09:09:44 -07:00
|
|
|
"net"
|
2013-02-08 05:49:55 -08:00
|
|
|
"net/http"
|
2013-08-09 09:09:44 -07:00
|
|
|
"os"
|
2015-03-24 14:04:38 -07:00
|
|
|
"strings"
|
2013-08-29 06:15:22 -07:00
|
|
|
"time"
|
|
|
|
|
|
|
|
pprof_runtime "runtime/pprof"
|
2013-06-28 01:19:16 -07:00
|
|
|
|
2013-08-12 08:18:02 -07:00
|
|
|
"github.com/golang/glog"
|
2013-06-28 01:19:16 -07:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
|
2015-05-18 03:16:25 -07:00
|
|
|
clientmodel "github.com/prometheus/client_golang/model"
|
|
|
|
|
2013-06-28 01:19:16 -07:00
|
|
|
"github.com/prometheus/prometheus/web/api"
|
|
|
|
"github.com/prometheus/prometheus/web/blob"
|
2013-02-08 05:49:55 -08:00
|
|
|
)
|
|
|
|
|
2015-05-18 03:16:25 -07:00
|
|
|
var localhostRepresentations = []string{"127.0.0.1", "localhost"}
|
|
|
|
|
2013-02-08 06:38:50 -08:00
|
|
|
// Commandline flags.
|
|
|
|
var (
|
2015-02-08 14:29:57 -08:00
|
|
|
listenAddress = flag.String("web.listen-address", ":9090", "Address to listen on for the web interface, API, and telemetry.")
|
|
|
|
metricsPath = flag.String("web.telemetry-path", "/metrics", "Path under which to expose metrics.")
|
2014-11-20 12:03:51 -08:00
|
|
|
useLocalAssets = flag.Bool("web.use-local-assets", false, "Read assets/templates from file instead of binary.")
|
|
|
|
userAssetsPath = flag.String("web.user-assets", "", "Path to static asset directory, available at /user.")
|
|
|
|
enableQuit = flag.Bool("web.enable-remote-shutdown", false, "Enable remote service shutdown.")
|
2013-02-08 06:38:50 -08:00
|
|
|
)
|
|
|
|
|
2014-12-10 07:16:49 -08:00
|
|
|
// WebService handles the HTTP endpoints with the exception of /api.
|
2013-05-05 10:32:04 -07:00
|
|
|
type WebService struct {
|
2014-06-06 02:55:53 -07:00
|
|
|
StatusHandler *PrometheusStatusHandler
|
|
|
|
MetricsHandler *api.MetricsService
|
|
|
|
AlertsHandler *AlertsHandler
|
|
|
|
ConsolesHandler *ConsolesHandler
|
2015-03-24 14:04:38 -07:00
|
|
|
GraphsHandler *GraphsHandler
|
2014-04-14 16:02:15 -07:00
|
|
|
|
2015-03-14 19:36:15 -07:00
|
|
|
QuitChan chan struct{}
|
2013-05-05 10:32:04 -07:00
|
|
|
}
|
|
|
|
|
2014-12-10 07:16:49 -08:00
|
|
|
// ServeForever serves the HTTP endpoints and only returns upon errors.
|
2015-05-19 05:38:50 -07:00
|
|
|
func (ws WebService) ServeForever(pathPrefix string) {
|
2015-03-24 14:04:38 -07:00
|
|
|
|
2014-06-18 10:43:15 -07:00
|
|
|
http.Handle("/favicon.ico", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2013-05-06 00:56:32 -07:00
|
|
|
http.Error(w, "", 404)
|
|
|
|
}))
|
|
|
|
|
2015-03-24 14:04:38 -07:00
|
|
|
http.Handle(pathPrefix, prometheus.InstrumentHandler(
|
|
|
|
pathPrefix, ws.StatusHandler,
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"alerts", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"alerts", ws.AlertsHandler,
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"consoles/", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"consoles/", http.StripPrefix(pathPrefix+"consoles/", ws.ConsolesHandler),
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"graph", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"graph", ws.GraphsHandler,
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"heap", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"heap", http.HandlerFunc(dumpHeap),
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2013-04-02 10:14:02 -07:00
|
|
|
|
2015-03-24 14:04:38 -07:00
|
|
|
ws.MetricsHandler.RegisterHandler(pathPrefix)
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+strings.TrimLeft(*metricsPath, "/"), prometheus.Handler())
|
2013-03-21 07:55:48 -07:00
|
|
|
if *useLocalAssets {
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"static/", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"static/", http.StripPrefix(pathPrefix+"static/", http.FileServer(http.Dir("web/static"))),
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2013-03-21 05:55:59 -07:00
|
|
|
} else {
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"static/", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"static/", http.StripPrefix(pathPrefix+"static/", new(blob.Handler)),
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2013-03-21 05:55:59 -07:00
|
|
|
}
|
2013-02-08 05:49:55 -08:00
|
|
|
|
2013-05-23 06:47:00 -07:00
|
|
|
if *userAssetsPath != "" {
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"user/", prometheus.InstrumentHandler(
|
|
|
|
pathPrefix+"user/", http.StripPrefix(pathPrefix+"user/", http.FileServer(http.Dir(*userAssetsPath))),
|
2014-06-18 10:43:15 -07:00
|
|
|
))
|
2013-05-23 06:47:00 -07:00
|
|
|
}
|
|
|
|
|
2014-04-14 16:02:15 -07:00
|
|
|
if *enableQuit {
|
2015-04-16 16:12:04 -07:00
|
|
|
http.Handle(pathPrefix+"-/quit", http.HandlerFunc(ws.quitHandler))
|
2015-03-24 14:04:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if pathPrefix != "/" {
|
|
|
|
http.Handle("/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
http.Redirect(w, r, pathPrefix, http.StatusFound)
|
|
|
|
}))
|
2014-04-14 16:02:15 -07:00
|
|
|
}
|
|
|
|
|
2015-05-19 05:38:50 -07:00
|
|
|
glog.Infof("Listening on %s", *listenAddress)
|
2013-05-05 10:32:04 -07:00
|
|
|
|
2015-05-19 05:38:50 -07:00
|
|
|
// If we cannot bind to a port, retry after 30 seconds.
|
|
|
|
for {
|
|
|
|
err := http.ListenAndServe(*listenAddress, nil)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Could not listen on %s: %s", *listenAddress, err)
|
|
|
|
}
|
|
|
|
time.Sleep(30 * time.Second)
|
|
|
|
}
|
2013-02-08 05:49:55 -08:00
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
|
2014-12-10 07:16:49 -08:00
|
|
|
func (ws WebService) quitHandler(w http.ResponseWriter, r *http.Request) {
|
2014-04-14 16:02:15 -07:00
|
|
|
if r.Method != "POST" {
|
|
|
|
w.Header().Add("Allow", "POST")
|
|
|
|
w.WriteHeader(http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "Requesting termination... Goodbye!")
|
|
|
|
|
2015-03-14 19:36:15 -07:00
|
|
|
close(ws.QuitChan)
|
2014-04-14 16:02:15 -07:00
|
|
|
}
|
|
|
|
|
2014-07-29 09:28:48 -07:00
|
|
|
func getTemplateFile(name string) (string, error) {
|
|
|
|
if *useLocalAssets {
|
|
|
|
file, err := ioutil.ReadFile(fmt.Sprintf("web/templates/%s.html", name))
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Could not read %s template: %s", name, err)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(file), nil
|
|
|
|
}
|
2014-12-10 07:16:49 -08:00
|
|
|
file, err := blob.GetFile(blob.TemplateFiles, name+".html")
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Could not read %s template: %s", name, err)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(file), nil
|
2013-06-07 02:17:17 -07:00
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
|
2015-03-24 14:04:38 -07:00
|
|
|
func getConsoles(pathPrefix string) string {
|
2015-04-06 08:30:13 -07:00
|
|
|
if _, err := os.Stat(*consoleTemplatesPath + "/index.html"); !os.IsNotExist(err) {
|
2015-03-24 14:04:38 -07:00
|
|
|
return pathPrefix + "consoles/index.html"
|
2014-10-02 07:44:47 -07:00
|
|
|
}
|
|
|
|
if *userAssetsPath != "" {
|
2015-04-06 08:30:13 -07:00
|
|
|
if _, err := os.Stat(*userAssetsPath + "/index.html"); !os.IsNotExist(err) {
|
2015-03-24 14:04:38 -07:00
|
|
|
return pathPrefix + "user/index.html"
|
2014-10-02 07:44:47 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2015-05-18 03:16:25 -07:00
|
|
|
func getTemplate(name string, pathPrefix string) (*template.Template, error) {
|
|
|
|
t := template.New("_base")
|
|
|
|
var err error
|
2015-03-24 14:04:38 -07:00
|
|
|
|
2014-07-29 09:28:48 -07:00
|
|
|
t.Funcs(template.FuncMap{
|
2014-10-02 07:44:47 -07:00
|
|
|
"since": time.Since,
|
2015-03-24 14:04:38 -07:00
|
|
|
"getConsoles": func() string { return getConsoles(pathPrefix) },
|
|
|
|
"pathPrefix": func() string { return pathPrefix },
|
2015-05-18 03:16:25 -07:00
|
|
|
"stripLabels": func(lset clientmodel.LabelSet, labels ...clientmodel.LabelName) clientmodel.LabelSet {
|
|
|
|
for _, ln := range labels {
|
|
|
|
delete(lset, ln)
|
|
|
|
}
|
|
|
|
return lset
|
|
|
|
},
|
|
|
|
"globalURL": func(url string) string {
|
|
|
|
hostname, err := os.Hostname()
|
|
|
|
if err != nil {
|
|
|
|
glog.Warningf("Couldn't get hostname: %s, returning target.URL()", err)
|
|
|
|
return url
|
|
|
|
}
|
|
|
|
for _, localhostRepresentation := range localhostRepresentations {
|
|
|
|
url = strings.Replace(url, "//"+localhostRepresentation, "//"+hostname, 1)
|
|
|
|
}
|
|
|
|
return url
|
|
|
|
},
|
2014-07-29 09:28:48 -07:00
|
|
|
})
|
2015-05-18 03:16:25 -07:00
|
|
|
|
2014-07-29 09:28:48 -07:00
|
|
|
file, err := getTemplateFile("_base")
|
2013-03-27 09:40:01 -07:00
|
|
|
if err != nil {
|
2015-05-18 03:16:25 -07:00
|
|
|
glog.Errorln("Could not read base template:", err)
|
2013-03-27 09:40:01 -07:00
|
|
|
return nil, err
|
|
|
|
}
|
2015-05-18 03:16:25 -07:00
|
|
|
t, err = t.Parse(file)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorln("Could not parse base template:", err)
|
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
|
2014-07-29 09:28:48 -07:00
|
|
|
file, err = getTemplateFile(name)
|
2013-03-27 09:40:01 -07:00
|
|
|
if err != nil {
|
2015-05-18 04:14:41 -07:00
|
|
|
glog.Error("Could not read template %s: %s", name, err)
|
2013-03-27 09:40:01 -07:00
|
|
|
return nil, err
|
|
|
|
}
|
2015-05-18 03:16:25 -07:00
|
|
|
t, err = t.Parse(file)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Could not parse template %s: %s", name, err)
|
|
|
|
}
|
|
|
|
return t, err
|
2013-03-27 09:40:01 -07:00
|
|
|
}
|
|
|
|
|
2015-03-24 14:04:38 -07:00
|
|
|
func executeTemplate(w http.ResponseWriter, name string, data interface{}, pathPrefix string) {
|
|
|
|
tpl, err := getTemplate(name, pathPrefix)
|
2013-03-27 09:40:01 -07:00
|
|
|
if err != nil {
|
2013-08-12 09:22:48 -07:00
|
|
|
glog.Error("Error preparing layout template: ", err)
|
2013-03-27 09:40:01 -07:00
|
|
|
return
|
|
|
|
}
|
2013-04-28 10:01:56 -07:00
|
|
|
err = tpl.Execute(w, data)
|
|
|
|
if err != nil {
|
2013-08-12 09:22:48 -07:00
|
|
|
glog.Error("Error executing template: ", err)
|
2013-04-28 10:01:56 -07:00
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
}
|
2013-08-09 09:09:44 -07:00
|
|
|
|
2013-08-29 06:15:22 -07:00
|
|
|
func dumpHeap(w http.ResponseWriter, r *http.Request) {
|
|
|
|
target := fmt.Sprintf("/tmp/%d.heap", time.Now().Unix())
|
|
|
|
f, err := os.Create(target)
|
|
|
|
if err != nil {
|
|
|
|
glog.Error("Could not dump heap: ", err)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "Writing to %s...", target)
|
|
|
|
defer f.Close()
|
|
|
|
pprof_runtime.WriteHeapProfile(f)
|
|
|
|
fmt.Fprintf(w, "Done")
|
|
|
|
}
|
|
|
|
|
2014-12-10 07:16:49 -08:00
|
|
|
// MustBuildServerURL returns the server URL and panics in case an error occurs.
|
2015-03-24 14:04:38 -07:00
|
|
|
func MustBuildServerURL(pathPrefix string) string {
|
2013-08-09 09:09:44 -07:00
|
|
|
_, port, err := net.SplitHostPort(*listenAddress)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
hostname, err := os.Hostname()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2015-03-24 14:04:38 -07:00
|
|
|
return fmt.Sprintf("http://%s:%s%s", hostname, port, pathPrefix)
|
2013-08-09 09:09:44 -07:00
|
|
|
}
|