mirror of
https://github.com/weaveworks/scope.git
synced 2026-03-03 18:20:27 +00:00
116 lines
3.3 KiB
Go
116 lines
3.3 KiB
Go
package app
|
|
|
|
import (
|
|
"compress/gzip"
|
|
"encoding/gob"
|
|
"encoding/json"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
|
|
"github.com/PuerkitoBio/ghost/handlers"
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/weaveworks/scope/common/hostname"
|
|
"github.com/weaveworks/scope/report"
|
|
"github.com/weaveworks/scope/xfer"
|
|
)
|
|
|
|
var (
|
|
// Version - set at buildtime.
|
|
Version = "dev"
|
|
|
|
// UniqueID - set at runtime.
|
|
UniqueID = "0"
|
|
)
|
|
|
|
// URLMatcher uses request.RequestURI (the raw, unparsed request) to attempt
|
|
// to match pattern. It does this as go's URL.Parse method is broken, and
|
|
// mistakenly unescapes the Path before parsing it. This breaks %2F (encoded
|
|
// forward slashes) in the paths.
|
|
func URLMatcher(pattern string) mux.MatcherFunc {
|
|
matchParts := strings.Split(pattern, "/")
|
|
|
|
return func(r *http.Request, rm *mux.RouteMatch) bool {
|
|
path := strings.SplitN(r.RequestURI, "?", 2)[0]
|
|
parts := strings.Split(path, "/")
|
|
if len(parts) != len(matchParts) {
|
|
return false
|
|
}
|
|
|
|
rm.Vars = map[string]string{}
|
|
for i, part := range parts {
|
|
unescaped, err := url.QueryUnescape(part)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
match := matchParts[i]
|
|
if strings.HasPrefix(match, "{") && strings.HasSuffix(match, "}") {
|
|
rm.Vars[strings.Trim(match, "{}")] = unescaped
|
|
} else if matchParts[i] != unescaped {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
}
|
|
|
|
func gzipHandler(h http.HandlerFunc) http.HandlerFunc {
|
|
return handlers.GZIPHandlerFunc(h, nil)
|
|
}
|
|
|
|
// RegisterTopologyRoutes registers the various topology routes with a http mux.
|
|
func RegisterTopologyRoutes(c Reporter, router *mux.Router) {
|
|
get := router.Methods("GET").Subrouter()
|
|
get.HandleFunc("/api", gzipHandler(apiHandler))
|
|
get.HandleFunc("/api/topology", gzipHandler(topologyRegistry.makeTopologyList(c)))
|
|
get.HandleFunc("/api/topology/{topology}",
|
|
gzipHandler(topologyRegistry.captureRenderer(c, handleTopology)))
|
|
get.HandleFunc("/api/topology/{topology}/ws",
|
|
topologyRegistry.captureRenderer(c, handleWs)) // NB not gzip!
|
|
get.MatcherFunc(URLMatcher("/api/topology/{topology}/{id}")).HandlerFunc(
|
|
gzipHandler(topologyRegistry.captureRendererWithoutFilters(c, handleNode)))
|
|
get.HandleFunc("/api/report", gzipHandler(makeRawReportHandler(c)))
|
|
}
|
|
|
|
// RegisterReportPostHandler registers the handler for report submission
|
|
func RegisterReportPostHandler(a Adder, router *mux.Router) {
|
|
post := router.Methods("POST").Subrouter()
|
|
post.HandleFunc("/api/report", func(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
rpt report.Report
|
|
reader = r.Body
|
|
err error
|
|
)
|
|
if strings.Contains(r.Header.Get("Content-Encoding"), "gzip") {
|
|
reader, err = gzip.NewReader(r.Body)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
}
|
|
|
|
decoder := gob.NewDecoder(reader).Decode
|
|
if strings.HasPrefix(r.Header.Get("Content-Type"), "application/json") {
|
|
decoder = json.NewDecoder(reader).Decode
|
|
}
|
|
if err := decoder(&rpt); err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
a.Add(rpt)
|
|
if len(rpt.Pod.Nodes) > 0 {
|
|
topologyRegistry.enableKubernetesTopologies()
|
|
}
|
|
w.WriteHeader(http.StatusOK)
|
|
})
|
|
}
|
|
|
|
func apiHandler(w http.ResponseWriter, r *http.Request) {
|
|
respondWith(w, http.StatusOK, xfer.Details{
|
|
ID: UniqueID,
|
|
Version: Version,
|
|
Hostname: hostname.Get(),
|
|
})
|
|
}
|