mirror of
				https://github.com/juanfont/headscale.git
				synced 2025-10-28 10:51:44 +01:00 
			
		
		
		
	Some endpoints in /debug send JSON data as string. Set the Content-Type header to "application/json" which renders nicely in Firefox. Mention the /debug route in the example configuration.
		
			
				
	
	
		
			131 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			131 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package hscontrol
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 
 | |
| 	"github.com/arl/statsviz"
 | |
| 	"github.com/juanfont/headscale/hscontrol/types"
 | |
| 	"github.com/prometheus/client_golang/prometheus/promhttp"
 | |
| 	"tailscale.com/tailcfg"
 | |
| 	"tailscale.com/tsweb"
 | |
| )
 | |
| 
 | |
| func (h *Headscale) debugHTTPServer() *http.Server {
 | |
| 	debugMux := http.NewServeMux()
 | |
| 	debug := tsweb.Debugger(debugMux)
 | |
| 	debug.Handle("notifier", "Connected nodes in notifier", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write([]byte(h.nodeNotifier.String()))
 | |
| 	}))
 | |
| 	debug.Handle("config", "Current configuration", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		config, err := json.MarshalIndent(h.cfg, "", "  ")
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(config)
 | |
| 	}))
 | |
| 	debug.Handle("policy", "Current policy", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		pol, err := h.policyBytes()
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(pol)
 | |
| 	}))
 | |
| 	debug.Handle("filter", "Current filter", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		filter := h.polMan.Filter()
 | |
| 
 | |
| 		filterJSON, err := json.MarshalIndent(filter, "", "  ")
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(filterJSON)
 | |
| 	}))
 | |
| 	debug.Handle("ssh", "SSH Policy per node", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		nodes, err := h.db.ListNodes()
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 
 | |
| 		sshPol := make(map[string]*tailcfg.SSHPolicy)
 | |
| 		for _, node := range nodes {
 | |
| 			pol, err := h.polMan.SSHPolicy(node)
 | |
| 			if err != nil {
 | |
| 				httpError(w, err)
 | |
| 				return
 | |
| 			}
 | |
| 
 | |
| 			sshPol[fmt.Sprintf("id:%d  hostname:%s givenname:%s", node.ID, node.Hostname, node.GivenName)] = pol
 | |
| 		}
 | |
| 
 | |
| 		sshJSON, err := json.MarshalIndent(sshPol, "", "  ")
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(sshJSON)
 | |
| 	}))
 | |
| 	debug.Handle("derpmap", "Current DERPMap", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		dm := h.DERPMap
 | |
| 
 | |
| 		dmJSON, err := json.MarshalIndent(dm, "", "  ")
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(dmJSON)
 | |
| 	}))
 | |
| 	debug.Handle("registration-cache", "Pending registrations", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		registrationsJSON, err := json.MarshalIndent(h.registrationCache.Items(), "", "  ")
 | |
| 		if err != nil {
 | |
| 			httpError(w, err)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Header().Set("Content-Type", "application/json")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write(registrationsJSON)
 | |
| 	}))
 | |
| 	debug.Handle("routes", "Routes", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		w.Header().Set("Content-Type", "text/plain")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write([]byte(h.primaryRoutes.String()))
 | |
| 	}))
 | |
| 	debug.Handle("policy-manager", "Policy Manager", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		w.Header().Set("Content-Type", "text/plain")
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 		w.Write([]byte(h.polMan.DebugString()))
 | |
| 	}))
 | |
| 
 | |
| 	err := statsviz.Register(debugMux)
 | |
| 	if err == nil {
 | |
| 		debug.URL("/debug/statsviz", "Statsviz (visualise go metrics)")
 | |
| 	}
 | |
| 
 | |
| 	debug.URL("/metrics", "Prometheus metrics")
 | |
| 	debugMux.Handle("/metrics", promhttp.Handler())
 | |
| 
 | |
| 	debugHTTPServer := &http.Server{
 | |
| 		Addr:         h.cfg.MetricsAddr,
 | |
| 		Handler:      debugMux,
 | |
| 		ReadTimeout:  types.HTTPTimeout,
 | |
| 		WriteTimeout: 0,
 | |
| 	}
 | |
| 
 | |
| 	return debugHTTPServer
 | |
| }
 |