mirror of
https://github.com/weaveworks/scope.git
synced 2026-03-04 18:51:17 +00:00
- App takes POST report on /api/report - Probe publishes to configured target(s) - Name resolution happens on probe-side - There's no longer an xfer.ProbePort - xfer.Collector responsibility is reduced - Fixes to remaining experimental components. - rm experimental/bridge: it's not being used, and by changing the app/probe comm model, it would require a complete refactor anyway. We can easily rebuild it when we need to. It will even be much simpler. - rm experimental/graphviz: it's broken for some time anyway, and we don't really need to play around with it as a rendering option anymore. - rm experimental/oneshot: we never use this anymore.
136 lines
3.8 KiB
Go
136 lines
3.8 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/gob"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/weaveworks/scope/render"
|
|
"github.com/weaveworks/scope/report"
|
|
"github.com/weaveworks/scope/xfer"
|
|
)
|
|
|
|
// 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
|
|
}
|
|
}
|
|
|
|
type collector interface {
|
|
xfer.Reporter
|
|
xfer.Adder
|
|
}
|
|
|
|
// Router returns the HTTP dispatcher, managing API and UI requests, and
|
|
// accepting reports from probes.. It will always use the embedded HTML
|
|
// resources for the UI.
|
|
func Router(c collector) *mux.Router {
|
|
router := mux.NewRouter()
|
|
router.HandleFunc("/api/report", makeReportPostHandler(c)).Methods("POST")
|
|
get := router.Methods("GET").Subrouter()
|
|
get.HandleFunc("/api", apiHandler)
|
|
get.HandleFunc("/api/topology", makeTopologyList(c))
|
|
get.HandleFunc("/api/topology/{topology}", captureTopology(c, handleTopology))
|
|
get.HandleFunc("/api/topology/{topology}/ws", captureTopology(c, handleWs))
|
|
get.MatcherFunc(URLMatcher("/api/topology/{topology}/{id}")).HandlerFunc(captureTopology(c, handleNode))
|
|
get.MatcherFunc(URLMatcher("/api/topology/{topology}/{local}/{remote}")).HandlerFunc(captureTopology(c, handleEdge))
|
|
get.MatcherFunc(URLMatcher("/api/origin/host/{id}")).HandlerFunc(makeOriginHostHandler(c))
|
|
get.HandleFunc("/api/report", makeRawReportHandler(c))
|
|
get.PathPrefix("/").Handler(http.FileServer(FS(false))) // everything else is static
|
|
return router
|
|
}
|
|
|
|
func makeReportPostHandler(a xfer.Adder) http.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
var rpt report.Report
|
|
if err := gob.NewDecoder(r.Body).Decode(&rpt); err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
a.Add(rpt)
|
|
w.WriteHeader(http.StatusOK)
|
|
}
|
|
}
|
|
|
|
func captureTopology(rep xfer.Reporter, f func(xfer.Reporter, topologyView, http.ResponseWriter, *http.Request)) http.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
topology, ok := topologyRegistry[mux.Vars(r)["topology"]]
|
|
if !ok {
|
|
http.NotFound(w, r)
|
|
return
|
|
}
|
|
f(rep, topology, w, r)
|
|
}
|
|
}
|
|
|
|
// APIDetails are some generic details that can be fetched from /api
|
|
type APIDetails struct {
|
|
Version string `json:"version"`
|
|
}
|
|
|
|
func apiHandler(w http.ResponseWriter, r *http.Request) {
|
|
respondWith(w, http.StatusOK, APIDetails{Version: version})
|
|
}
|
|
|
|
var topologyRegistry = map[string]topologyView{
|
|
"applications": {
|
|
human: "Applications",
|
|
parent: "",
|
|
renderer: render.FilterUnconnected{Renderer: render.ProcessRenderer},
|
|
},
|
|
"applications-by-name": {
|
|
human: "by name",
|
|
parent: "applications",
|
|
renderer: render.FilterUnconnected{Renderer: render.ProcessNameRenderer},
|
|
},
|
|
"containers": {
|
|
human: "Containers",
|
|
parent: "",
|
|
renderer: render.ContainerRenderer,
|
|
},
|
|
"containers-by-image": {
|
|
human: "by image",
|
|
parent: "containers",
|
|
renderer: render.ContainerImageRenderer,
|
|
},
|
|
"hosts": {
|
|
human: "Hosts",
|
|
parent: "",
|
|
renderer: render.HostRenderer,
|
|
},
|
|
}
|
|
|
|
type topologyView struct {
|
|
human string
|
|
parent string
|
|
renderer render.Renderer
|
|
}
|