mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-26 13:11:11 -08:00
a081c9436b
Main changes: - Switched to using `go-bindata` in place of `scripts/embed-static.sh`. - Support for building Prometheus without a `Makefile`. - Minor typo fix to make Prometheus build on Windows (without Makefiles). Please note that this does not mean that prometheus will work on Windows. There are still failing tests!
258 lines
7.1 KiB
Go
258 lines
7.1 KiB
Go
// Copyright 2013 The Prometheus Authors
|
|
// 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"
|
|
"io/ioutil"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
|
|
pprof_runtime "runtime/pprof"
|
|
|
|
clientmodel "github.com/prometheus/client_golang/model"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/log"
|
|
|
|
"github.com/prometheus/prometheus/web/api/legacy"
|
|
"github.com/prometheus/prometheus/web/api/v1"
|
|
|
|
"github.com/prometheus/prometheus/util/route"
|
|
"github.com/prometheus/prometheus/web/blob"
|
|
)
|
|
|
|
var localhostRepresentations = []string{"127.0.0.1", "localhost"}
|
|
|
|
// Commandline flags.
|
|
var (
|
|
listenAddress = flag.String("web.listen-address", ":9090", "Address to listen on for the web interface, API, and telemetry.")
|
|
hostname = flag.String("web.hostname", "", "Hostname on which the server is available.")
|
|
metricsPath = flag.String("web.telemetry-path", "/metrics", "Path under which to expose metrics.")
|
|
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.")
|
|
)
|
|
|
|
// WebService handles the HTTP endpoints with the exception of /api.
|
|
type WebService struct {
|
|
QuitChan chan struct{}
|
|
router *route.Router
|
|
}
|
|
|
|
type WebServiceOptions struct {
|
|
PathPrefix string
|
|
StatusHandler *PrometheusStatusHandler
|
|
APILegacy *legacy.API
|
|
APIv1 *v1.API
|
|
AlertsHandler *AlertsHandler
|
|
ConsolesHandler *ConsolesHandler
|
|
GraphsHandler *GraphsHandler
|
|
}
|
|
|
|
// NewWebService returns a new WebService.
|
|
func NewWebService(o *WebServiceOptions) *WebService {
|
|
router := route.New()
|
|
|
|
ws := &WebService{
|
|
router: router,
|
|
QuitChan: make(chan struct{}),
|
|
}
|
|
|
|
if o.PathPrefix != "" {
|
|
// If the prefix is missing for the root path, append it.
|
|
router.Get("/", func(w http.ResponseWriter, r *http.Request) {
|
|
http.Redirect(w, r, o.PathPrefix, 302)
|
|
})
|
|
router = router.WithPrefix(o.PathPrefix)
|
|
}
|
|
|
|
instr := prometheus.InstrumentHandler
|
|
|
|
router.Get("/", instr("status", o.StatusHandler))
|
|
router.Get("/alerts", instr("alerts", o.AlertsHandler))
|
|
router.Get("/graph", instr("graph", o.GraphsHandler))
|
|
router.Get("/heap", instr("heap", http.HandlerFunc(dumpHeap)))
|
|
|
|
router.Get(*metricsPath, prometheus.Handler().ServeHTTP)
|
|
|
|
o.APILegacy.Register(router.WithPrefix("/api"))
|
|
|
|
o.APIv1.Register(router.WithPrefix("/api/v1"))
|
|
|
|
router.Get("/consoles/*filepath", instr("consoles", o.ConsolesHandler))
|
|
|
|
if *useLocalAssets {
|
|
router.Get("/static/*filepath", instr("static", route.FileServe("web/blob/static")))
|
|
} else {
|
|
router.Get("/static/*filepath", instr("static", blob.Handler{}))
|
|
}
|
|
|
|
if *userAssetsPath != "" {
|
|
router.Get("/user/*filepath", instr("user", route.FileServe(*userAssetsPath)))
|
|
}
|
|
|
|
if *enableQuit {
|
|
router.Post("/-/quit", ws.quitHandler)
|
|
}
|
|
|
|
return ws
|
|
}
|
|
|
|
// Run serves the HTTP endpoints.
|
|
func (ws *WebService) Run() {
|
|
log.Infof("Listening on %s", *listenAddress)
|
|
|
|
// If we cannot bind to a port, retry after 30 seconds.
|
|
for {
|
|
err := http.ListenAndServe(*listenAddress, ws.router)
|
|
if err != nil {
|
|
log.Errorf("Could not listen on %s: %s", *listenAddress, err)
|
|
}
|
|
time.Sleep(30 * time.Second)
|
|
}
|
|
}
|
|
|
|
func (ws *WebService) quitHandler(w http.ResponseWriter, r *http.Request) {
|
|
fmt.Fprintf(w, "Requesting termination... Goodbye!")
|
|
|
|
close(ws.QuitChan)
|
|
}
|
|
|
|
func getTemplateFile(name string) (string, error) {
|
|
if *useLocalAssets {
|
|
file, err := ioutil.ReadFile(fmt.Sprintf("web/blob/templates/%s.html", name))
|
|
if err != nil {
|
|
log.Errorf("Could not read %s template: %s", name, err)
|
|
return "", err
|
|
}
|
|
return string(file), nil
|
|
}
|
|
file, err := blob.GetFile(blob.TemplateFiles, name+".html")
|
|
if err != nil {
|
|
log.Errorf("Could not read %s template: %s", name, err)
|
|
return "", err
|
|
}
|
|
return string(file), nil
|
|
}
|
|
|
|
func getConsoles(pathPrefix string) string {
|
|
if _, err := os.Stat(*consoleTemplatesPath + "/index.html"); !os.IsNotExist(err) {
|
|
return pathPrefix + "/consoles/index.html"
|
|
}
|
|
if *userAssetsPath != "" {
|
|
if _, err := os.Stat(*userAssetsPath + "/index.html"); !os.IsNotExist(err) {
|
|
return pathPrefix + "/user/index.html"
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func getTemplate(name string, pathPrefix string) (*template.Template, error) {
|
|
t := template.New("_base")
|
|
var err error
|
|
|
|
t.Funcs(template.FuncMap{
|
|
"since": time.Since,
|
|
"getConsoles": func() string { return getConsoles(pathPrefix) },
|
|
"pathPrefix": func() string { return pathPrefix },
|
|
"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 := getHostname()
|
|
if err != nil {
|
|
log.Warnf("Couldn't get hostname: %s, returning target.URL()", err)
|
|
return url
|
|
}
|
|
for _, localhostRepresentation := range localhostRepresentations {
|
|
url = strings.Replace(url, "//"+localhostRepresentation, "//"+hostname, 1)
|
|
}
|
|
return url
|
|
},
|
|
})
|
|
|
|
file, err := getTemplateFile("_base")
|
|
if err != nil {
|
|
log.Errorln("Could not read base template:", err)
|
|
return nil, err
|
|
}
|
|
t, err = t.Parse(file)
|
|
if err != nil {
|
|
log.Errorln("Could not parse base template:", err)
|
|
}
|
|
|
|
file, err = getTemplateFile(name)
|
|
if err != nil {
|
|
log.Error("Could not read template %s: %s", name, err)
|
|
return nil, err
|
|
}
|
|
t, err = t.Parse(file)
|
|
if err != nil {
|
|
log.Errorf("Could not parse template %s: %s", name, err)
|
|
}
|
|
return t, err
|
|
}
|
|
|
|
func executeTemplate(w http.ResponseWriter, name string, data interface{}, pathPrefix string) {
|
|
tpl, err := getTemplate(name, pathPrefix)
|
|
if err != nil {
|
|
log.Error("Error preparing layout template: ", err)
|
|
return
|
|
}
|
|
err = tpl.Execute(w, data)
|
|
if err != nil {
|
|
log.Error("Error executing template: ", err)
|
|
}
|
|
}
|
|
|
|
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 {
|
|
log.Error("Could not dump heap: ", err)
|
|
}
|
|
fmt.Fprintf(w, "Writing to %s...", target)
|
|
defer f.Close()
|
|
pprof_runtime.WriteHeapProfile(f)
|
|
fmt.Fprintf(w, "Done")
|
|
}
|
|
|
|
// MustBuildServerURL returns the server URL and panics in case an error occurs.
|
|
func MustBuildServerURL(pathPrefix string) string {
|
|
_, port, err := net.SplitHostPort(*listenAddress)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
hostname, err := getHostname()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return fmt.Sprintf("http://%s:%s%s/", hostname, port, pathPrefix)
|
|
}
|
|
|
|
func getHostname() (string, error) {
|
|
if *hostname != "" {
|
|
return *hostname, nil
|
|
}
|
|
return os.Hostname()
|
|
}
|