2013-02-08 05:49:55 -08: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 (
|
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"
|
2013-08-09 09:09:44 -07:00
|
|
|
"net"
|
2013-02-08 05:49:55 -08:00
|
|
|
"net/http"
|
2013-04-22 04:11:21 -07:00
|
|
|
"net/http/pprof"
|
2013-08-09 09:09:44 -07:00
|
|
|
"os"
|
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"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/exp"
|
|
|
|
|
|
|
|
"github.com/prometheus/prometheus/web/api"
|
|
|
|
"github.com/prometheus/prometheus/web/blob"
|
2013-02-08 05:49:55 -08:00
|
|
|
)
|
|
|
|
|
2013-02-08 06:38:50 -08:00
|
|
|
// Commandline flags.
|
|
|
|
var (
|
2013-03-21 10:11:34 -07:00
|
|
|
listenAddress = flag.String("listenAddress", ":9090", "Address to listen on for web interface.")
|
2013-03-25 04:09:39 -07:00
|
|
|
useLocalAssets = flag.Bool("useLocalAssets", false, "Read assets/templates from file instead of binary.")
|
2013-05-23 06:47:00 -07:00
|
|
|
userAssetsPath = flag.String("userAssets", "", "Path to static asset directory, available at /user")
|
2014-04-14 16:02:15 -07:00
|
|
|
enableQuit = flag.Bool("web.enableRemoteShutdown", false, "Enable remote service shutdown")
|
2013-02-08 06:38:50 -08:00
|
|
|
)
|
|
|
|
|
2013-05-05 10:32:04 -07:00
|
|
|
type WebService struct {
|
2013-08-13 08:19:13 -07:00
|
|
|
StatusHandler *PrometheusStatusHandler
|
2013-05-14 02:21:27 -07:00
|
|
|
DatabasesHandler *DatabasesHandler
|
|
|
|
MetricsHandler *api.MetricsService
|
2013-06-13 07:10:05 -07:00
|
|
|
AlertsHandler *AlertsHandler
|
2014-05-28 10:44:54 -07:00
|
|
|
ConsolesHandler *ConsolesHandler
|
2014-04-14 16:02:15 -07:00
|
|
|
|
|
|
|
QuitDelegate func()
|
2013-05-05 10:32:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (w WebService) ServeForever() error {
|
2013-05-06 00:56:32 -07:00
|
|
|
exp.Handle("/favicon.ico", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
http.Error(w, "", 404)
|
|
|
|
}))
|
|
|
|
|
2013-04-22 04:11:21 -07:00
|
|
|
// TODO(julius): This will need to be rewritten once the exp package provides
|
|
|
|
// the coarse mux behaviors via a wrapper function.
|
|
|
|
exp.Handle("/debug/pprof/", http.HandlerFunc(pprof.Index))
|
|
|
|
exp.Handle("/debug/pprof/cmdline", http.HandlerFunc(pprof.Cmdline))
|
|
|
|
exp.Handle("/debug/pprof/profile", http.HandlerFunc(pprof.Profile))
|
|
|
|
exp.Handle("/debug/pprof/symbol", http.HandlerFunc(pprof.Symbol))
|
|
|
|
|
2013-05-05 10:32:04 -07:00
|
|
|
exp.Handle("/", w.StatusHandler)
|
2013-05-14 02:21:27 -07:00
|
|
|
exp.Handle("/databases", w.DatabasesHandler)
|
2013-06-13 07:10:05 -07:00
|
|
|
exp.Handle("/alerts", w.AlertsHandler)
|
2014-05-28 10:44:54 -07:00
|
|
|
exp.Handle("/consoles/", http.StripPrefix("/consoles/", w.ConsolesHandler))
|
2013-04-05 04:24:50 -07:00
|
|
|
exp.HandleFunc("/graph", graphHandler)
|
2013-08-29 06:15:22 -07:00
|
|
|
exp.HandleFunc("/heap", dumpHeap)
|
2013-04-02 10:14:02 -07:00
|
|
|
|
2013-10-22 11:31:52 -07:00
|
|
|
w.MetricsHandler.RegisterHandler()
|
2013-08-09 09:09:44 -07:00
|
|
|
exp.Handle("/metrics", prometheus.DefaultHandler)
|
2013-03-21 07:55:48 -07:00
|
|
|
if *useLocalAssets {
|
2013-04-05 04:24:50 -07:00
|
|
|
exp.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("web/static"))))
|
2013-03-21 05:55:59 -07:00
|
|
|
} else {
|
2013-04-05 04:24:50 -07:00
|
|
|
exp.Handle("/static/", http.StripPrefix("/static/", new(blob.Handler)))
|
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 != "" {
|
|
|
|
exp.Handle("/user/", http.StripPrefix("/user/", http.FileServer(http.Dir(*userAssetsPath))))
|
|
|
|
}
|
|
|
|
|
2014-04-14 16:02:15 -07:00
|
|
|
if *enableQuit {
|
|
|
|
exp.HandleFunc("/-/quit", w.quitHandler)
|
|
|
|
}
|
|
|
|
|
2013-08-12 09:22:48 -07:00
|
|
|
glog.Info("listening on ", *listenAddress)
|
2013-05-05 10:32:04 -07:00
|
|
|
|
|
|
|
return http.ListenAndServe(*listenAddress, exp.DefaultCoarseMux)
|
2013-02-08 05:49:55 -08:00
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
|
2014-04-14 16:02:15 -07:00
|
|
|
func (s WebService) quitHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != "POST" {
|
|
|
|
w.Header().Add("Allow", "POST")
|
|
|
|
w.WriteHeader(http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "Requesting termination... Goodbye!")
|
|
|
|
|
|
|
|
s.QuitDelegate()
|
|
|
|
}
|
|
|
|
|
2013-06-07 02:17:17 -07:00
|
|
|
func getLocalTemplate(name string) (*template.Template, error) {
|
|
|
|
return template.ParseFiles(
|
|
|
|
"web/templates/_base.html",
|
|
|
|
fmt.Sprintf("web/templates/%s.html", name),
|
|
|
|
)
|
|
|
|
}
|
2013-03-27 09:40:01 -07:00
|
|
|
|
2013-06-07 02:17:17 -07:00
|
|
|
func getEmbeddedTemplate(name string) (*template.Template, error) {
|
|
|
|
t := template.New("_base")
|
2013-03-27 09:40:01 -07:00
|
|
|
|
|
|
|
file, err := blob.GetFile(blob.TemplateFiles, "_base.html")
|
|
|
|
if err != nil {
|
2013-08-12 09:22:48 -07:00
|
|
|
glog.Error("Could not read base template: ", err)
|
2013-03-27 09:40:01 -07:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t.Parse(string(file))
|
|
|
|
|
|
|
|
file, err = blob.GetFile(blob.TemplateFiles, name+".html")
|
|
|
|
if err != nil {
|
2013-08-12 08:18:02 -07:00
|
|
|
glog.Errorf("Could not read %s template: %s", name, err)
|
2013-03-27 09:40:01 -07:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t.Parse(string(file))
|
|
|
|
|
2013-06-07 02:17:17 -07:00
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getTemplate(name string) (t *template.Template, err error) {
|
|
|
|
if *useLocalAssets {
|
|
|
|
t, err = getLocalTemplate(name)
|
|
|
|
} else {
|
|
|
|
t, err = getEmbeddedTemplate(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if *userAssetsPath != "" {
|
|
|
|
// replace "user_dashboard_link" template
|
|
|
|
t.Parse(`{{define "user_dashboard_link"}}<a href="/user">User Dashboard{{end}}`)
|
|
|
|
}
|
|
|
|
|
2013-03-27 09:40:01 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func executeTemplate(w http.ResponseWriter, name string, data interface{}) {
|
|
|
|
tpl, err := getTemplate(name)
|
|
|
|
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")
|
|
|
|
}
|
|
|
|
|
2013-08-09 09:09:44 -07:00
|
|
|
func MustBuildServerUrl() string {
|
|
|
|
_, port, err := net.SplitHostPort(*listenAddress)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
hostname, err := os.Hostname()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("http://%s:%s", hostname, port)
|
|
|
|
}
|