mirror of
				https://github.com/prometheus/prometheus.git
				synced 2025-10-26 05:51:01 +01:00 
			
		
		
		
	Essentially: - Remove unused code. - Make it 'go vet' clean. The only remaining warnings are in generated code. - Make it 'golint' clean. The only remaining warnings are in gerenated code. - Smoothed out same minor things. Change-Id: I3fe5c1fbead27b0e7a9c247fee2f5a45bc2d42c6
		
			
				
	
	
		
			201 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			201 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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"
 | |
| 	"io/ioutil"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"os"
 | |
| 	"time"
 | |
| 
 | |
| 	pprof_runtime "runtime/pprof"
 | |
| 
 | |
| 	"github.com/golang/glog"
 | |
| 	"github.com/prometheus/client_golang/prometheus"
 | |
| 
 | |
| 	"github.com/prometheus/prometheus/web/api"
 | |
| 	"github.com/prometheus/prometheus/web/blob"
 | |
| )
 | |
| 
 | |
| // Commandline flags.
 | |
| var (
 | |
| 	listenAddress  = flag.String("web.listen-address", ":9090", "Address to listen on for web interface.")
 | |
| 	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 {
 | |
| 	StatusHandler   *PrometheusStatusHandler
 | |
| 	MetricsHandler  *api.MetricsService
 | |
| 	AlertsHandler   *AlertsHandler
 | |
| 	ConsolesHandler *ConsolesHandler
 | |
| 
 | |
| 	QuitDelegate func()
 | |
| }
 | |
| 
 | |
| // ServeForever serves the HTTP endpoints and only returns upon errors.
 | |
| func (ws WebService) ServeForever() error {
 | |
| 	http.Handle("/favicon.ico", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		http.Error(w, "", 404)
 | |
| 	}))
 | |
| 
 | |
| 	http.Handle("/", prometheus.InstrumentHandler(
 | |
| 		"/", ws.StatusHandler,
 | |
| 	))
 | |
| 	http.Handle("/alerts", prometheus.InstrumentHandler(
 | |
| 		"/alerts", ws.AlertsHandler,
 | |
| 	))
 | |
| 	http.Handle("/consoles/", prometheus.InstrumentHandler(
 | |
| 		"/consoles/", http.StripPrefix("/consoles/", ws.ConsolesHandler),
 | |
| 	))
 | |
| 	http.Handle("/graph", prometheus.InstrumentHandler(
 | |
| 		"/graph", http.HandlerFunc(graphHandler),
 | |
| 	))
 | |
| 	http.Handle("/heap", prometheus.InstrumentHandler(
 | |
| 		"/heap", http.HandlerFunc(dumpHeap),
 | |
| 	))
 | |
| 
 | |
| 	ws.MetricsHandler.RegisterHandler()
 | |
| 	http.Handle("/metrics", prometheus.Handler())
 | |
| 	if *useLocalAssets {
 | |
| 		http.Handle("/static/", prometheus.InstrumentHandler(
 | |
| 			"/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("web/static"))),
 | |
| 		))
 | |
| 	} else {
 | |
| 		http.Handle("/static/", prometheus.InstrumentHandler(
 | |
| 			"/static/", http.StripPrefix("/static/", new(blob.Handler)),
 | |
| 		))
 | |
| 	}
 | |
| 
 | |
| 	if *userAssetsPath != "" {
 | |
| 		http.Handle("/user/", prometheus.InstrumentHandler(
 | |
| 			"/user/", http.StripPrefix("/user/", http.FileServer(http.Dir(*userAssetsPath))),
 | |
| 		))
 | |
| 	}
 | |
| 
 | |
| 	if *enableQuit {
 | |
| 		http.Handle("/-/quit", http.HandlerFunc(ws.quitHandler))
 | |
| 	}
 | |
| 
 | |
| 	glog.Info("listening on ", *listenAddress)
 | |
| 
 | |
| 	return http.ListenAndServe(*listenAddress, nil)
 | |
| }
 | |
| 
 | |
| func (ws 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!")
 | |
| 
 | |
| 	ws.QuitDelegate()
 | |
| }
 | |
| 
 | |
| 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
 | |
| 	}
 | |
| 	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
 | |
| }
 | |
| 
 | |
| func getConsoles() string {
 | |
| 	if _, err := os.Stat(*consoleTemplatesPath + "/index.html"); !os.IsNotExist(err) {
 | |
| 		return "/consoles/index.html"
 | |
| 	}
 | |
| 	if *userAssetsPath != "" {
 | |
| 		if _, err := os.Stat(*userAssetsPath + "/index.html"); !os.IsNotExist(err) {
 | |
| 			return "/user/index.html"
 | |
| 		}
 | |
| 	}
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| func getTemplate(name string) (t *template.Template, err error) {
 | |
| 	t = template.New("_base")
 | |
| 	t.Funcs(template.FuncMap{
 | |
| 		"since":       time.Since,
 | |
| 		"getConsoles": getConsoles,
 | |
| 	})
 | |
| 	file, err := getTemplateFile("_base")
 | |
| 	if err != nil {
 | |
| 		glog.Error("Could not read base template: ", err)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	t.Parse(file)
 | |
| 
 | |
| 	file, err = getTemplateFile(name)
 | |
| 	if err != nil {
 | |
| 		glog.Error("Could not read base template: ", err)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	t.Parse(file)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func executeTemplate(w http.ResponseWriter, name string, data interface{}) {
 | |
| 	tpl, err := getTemplate(name)
 | |
| 	if err != nil {
 | |
| 		glog.Error("Error preparing layout template: ", err)
 | |
| 		return
 | |
| 	}
 | |
| 	err = tpl.Execute(w, data)
 | |
| 	if err != nil {
 | |
| 		glog.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 {
 | |
| 		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")
 | |
| }
 | |
| 
 | |
| // MustBuildServerURL returns the server URL and panics in case an error occurs.
 | |
| 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)
 | |
| }
 |