1
0
mirror of https://github.com/juanfont/headscale.git synced 2025-09-16 17:50:44 +02:00
juanfont.headscale/hscontrol/state/debug.go
Kristoffer Dalby 50ed24847b debug: add json and improve
Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com>
2025-09-09 09:40:00 +02:00

382 lines
9.8 KiB
Go

package state
import (
"fmt"
"strings"
"time"
"github.com/juanfont/headscale/hscontrol/routes"
"github.com/juanfont/headscale/hscontrol/types"
"tailscale.com/tailcfg"
)
// DebugOverviewInfo represents the state overview information in a structured format.
type DebugOverviewInfo struct {
Nodes struct {
Total int `json:"total"`
Online int `json:"online"`
Expired int `json:"expired"`
Ephemeral int `json:"ephemeral"`
} `json:"nodes"`
Users map[string]int `json:"users"` // username -> node count
TotalUsers int `json:"total_users"`
Policy struct {
Mode string `json:"mode"`
Path string `json:"path,omitempty"`
} `json:"policy"`
DERP struct {
Configured bool `json:"configured"`
Regions int `json:"regions"`
} `json:"derp"`
PrimaryRoutes int `json:"primary_routes"`
}
// DebugDERPInfo represents DERP map information in a structured format.
type DebugDERPInfo struct {
Configured bool `json:"configured"`
TotalRegions int `json:"total_regions"`
Regions map[int]*DebugDERPRegion `json:"regions,omitempty"`
}
// DebugDERPRegion represents a single DERP region.
type DebugDERPRegion struct {
RegionID int `json:"region_id"`
RegionName string `json:"region_name"`
Nodes []*DebugDERPNode `json:"nodes"`
}
// DebugDERPNode represents a single DERP node.
type DebugDERPNode struct {
Name string `json:"name"`
HostName string `json:"hostname"`
DERPPort int `json:"derp_port"`
STUNPort int `json:"stun_port,omitempty"`
}
// DebugStringInfo wraps a debug string for JSON serialization.
type DebugStringInfo struct {
Content string `json:"content"`
}
// DebugOverview returns a comprehensive overview of the current state for debugging.
func (s *State) DebugOverview() string {
s.mu.RLock()
defer s.mu.RUnlock()
allNodes := s.nodeStore.ListNodes()
users, _ := s.ListAllUsers()
var sb strings.Builder
sb.WriteString("=== Headscale State Overview ===\n\n")
// Node statistics
sb.WriteString(fmt.Sprintf("Nodes: %d total\n", allNodes.Len()))
userNodeCounts := make(map[string]int)
onlineCount := 0
expiredCount := 0
ephemeralCount := 0
now := time.Now()
for _, node := range allNodes.All() {
if node.Valid() {
userName := node.User().Name
userNodeCounts[userName]++
if node.IsOnline().Valid() && node.IsOnline().Get() {
onlineCount++
}
if node.Expiry().Valid() && node.Expiry().Get().Before(now) {
expiredCount++
}
if node.AuthKey().Valid() && node.AuthKey().Ephemeral() {
ephemeralCount++
}
}
}
sb.WriteString(fmt.Sprintf(" - Online: %d\n", onlineCount))
sb.WriteString(fmt.Sprintf(" - Expired: %d\n", expiredCount))
sb.WriteString(fmt.Sprintf(" - Ephemeral: %d\n", ephemeralCount))
sb.WriteString("\n")
// User statistics
sb.WriteString(fmt.Sprintf("Users: %d total\n", len(users)))
for userName, nodeCount := range userNodeCounts {
sb.WriteString(fmt.Sprintf(" - %s: %d nodes\n", userName, nodeCount))
}
sb.WriteString("\n")
// Policy information
sb.WriteString("Policy:\n")
sb.WriteString(fmt.Sprintf(" - Mode: %s\n", s.cfg.Policy.Mode))
if s.cfg.Policy.Mode == types.PolicyModeFile {
sb.WriteString(fmt.Sprintf(" - Path: %s\n", s.cfg.Policy.Path))
}
sb.WriteString("\n")
// DERP information
derpMap := s.derpMap.Load()
if derpMap != nil {
sb.WriteString(fmt.Sprintf("DERP: %d regions configured\n", len(derpMap.Regions)))
} else {
sb.WriteString("DERP: not configured\n")
}
sb.WriteString("\n")
// Route information
routeCount := len(strings.Split(strings.TrimSpace(s.primaryRoutes.String()), "\n"))
if s.primaryRoutes.String() == "" {
routeCount = 0
}
sb.WriteString(fmt.Sprintf("Primary Routes: %d active\n", routeCount))
sb.WriteString("\n")
// Registration cache
sb.WriteString("Registration Cache: active\n")
sb.WriteString("\n")
return sb.String()
}
// DebugNodeStore returns debug information about the NodeStore.
func (s *State) DebugNodeStore() string {
return s.nodeStore.DebugString()
}
// DebugDERPMap returns debug information about the DERP map configuration.
func (s *State) DebugDERPMap() string {
derpMap := s.derpMap.Load()
if derpMap == nil {
return "DERP Map: not configured\n"
}
var sb strings.Builder
sb.WriteString("=== DERP Map Configuration ===\n\n")
sb.WriteString(fmt.Sprintf("Total Regions: %d\n\n", len(derpMap.Regions)))
for regionID, region := range derpMap.Regions {
sb.WriteString(fmt.Sprintf("Region %d: %s\n", regionID, region.RegionName))
sb.WriteString(fmt.Sprintf(" - Nodes: %d\n", len(region.Nodes)))
for _, node := range region.Nodes {
sb.WriteString(fmt.Sprintf(" - %s (%s:%d)\n",
node.Name, node.HostName, node.DERPPort))
if node.STUNPort != 0 {
sb.WriteString(fmt.Sprintf(" STUN: %d\n", node.STUNPort))
}
}
sb.WriteString("\n")
}
return sb.String()
}
// DebugSSHPolicies returns debug information about SSH policies for all nodes.
func (s *State) DebugSSHPolicies() map[string]*tailcfg.SSHPolicy {
nodes := s.nodeStore.ListNodes()
sshPolicies := make(map[string]*tailcfg.SSHPolicy)
for _, node := range nodes.All() {
if !node.Valid() {
continue
}
pol, err := s.SSHPolicy(node)
if err != nil {
// Store the error information
continue
}
key := fmt.Sprintf("id:%d hostname:%s givenname:%s",
node.ID(), node.Hostname(), node.GivenName())
sshPolicies[key] = pol
}
return sshPolicies
}
// DebugRegistrationCache returns debug information about the registration cache.
func (s *State) DebugRegistrationCache() map[string]interface{} {
// The cache doesn't expose internal statistics, so we provide basic info
result := map[string]interface{}{
"type": "zcache",
"expiration": registerCacheExpiration.String(),
"cleanup": registerCacheCleanup.String(),
"status": "active",
}
return result
}
// DebugConfig returns debug information about the current configuration.
func (s *State) DebugConfig() *types.Config {
return s.cfg
}
// DebugPolicy returns the current policy data as a string.
func (s *State) DebugPolicy() (string, error) {
switch s.cfg.Policy.Mode {
case types.PolicyModeDB:
p, err := s.GetPolicy()
if err != nil {
return "", err
}
return p.Data, nil
case types.PolicyModeFile:
pol, err := policyBytes(s.db, s.cfg)
if err != nil {
return "", err
}
return string(pol), nil
default:
return "", fmt.Errorf("unsupported policy mode: %s", s.cfg.Policy.Mode)
}
}
// DebugFilter returns the current filter rules and matchers.
func (s *State) DebugFilter() ([]tailcfg.FilterRule, error) {
filter, _ := s.Filter()
return filter, nil
}
// DebugRoutes returns the current primary routes information as a structured object.
func (s *State) DebugRoutes() routes.DebugRoutes {
return s.primaryRoutes.DebugJSON()
}
// DebugRoutesString returns the current primary routes information as a string.
func (s *State) DebugRoutesString() string {
return s.PrimaryRoutesString()
}
// DebugPolicyManager returns the policy manager debug string.
func (s *State) DebugPolicyManager() string {
return s.PolicyDebugString()
}
// PolicyDebugString returns a debug representation of the current policy.
func (s *State) PolicyDebugString() string {
return s.polMan.DebugString()
}
// DebugOverviewJSON returns a structured overview of the current state for debugging.
func (s *State) DebugOverviewJSON() DebugOverviewInfo {
s.mu.RLock()
defer s.mu.RUnlock()
allNodes := s.nodeStore.ListNodes()
users, _ := s.ListAllUsers()
info := DebugOverviewInfo{
Users: make(map[string]int),
TotalUsers: len(users),
}
// Node statistics
info.Nodes.Total = allNodes.Len()
now := time.Now()
for _, node := range allNodes.All() {
if node.Valid() {
userName := node.User().Name
info.Users[userName]++
if node.IsOnline().Valid() && node.IsOnline().Get() {
info.Nodes.Online++
}
if node.Expiry().Valid() && node.Expiry().Get().Before(now) {
info.Nodes.Expired++
}
if node.AuthKey().Valid() && node.AuthKey().Ephemeral() {
info.Nodes.Ephemeral++
}
}
}
// Policy information
info.Policy.Mode = string(s.cfg.Policy.Mode)
if s.cfg.Policy.Mode == types.PolicyModeFile {
info.Policy.Path = s.cfg.Policy.Path
}
derpMap := s.derpMap.Load()
if derpMap != nil {
info.DERP.Configured = true
info.DERP.Regions = len(derpMap.Regions)
} else {
info.DERP.Configured = false
info.DERP.Regions = 0
}
// Route information
routeCount := len(strings.Split(strings.TrimSpace(s.primaryRoutes.String()), "\n"))
if s.primaryRoutes.String() == "" {
routeCount = 0
}
info.PrimaryRoutes = routeCount
return info
}
// DebugDERPJSON returns structured debug information about the DERP map configuration.
func (s *State) DebugDERPJSON() DebugDERPInfo {
derpMap := s.derpMap.Load()
info := DebugDERPInfo{
Configured: derpMap != nil,
Regions: make(map[int]*DebugDERPRegion),
}
if derpMap == nil {
return info
}
info.TotalRegions = len(derpMap.Regions)
for regionID, region := range derpMap.Regions {
debugRegion := &DebugDERPRegion{
RegionID: regionID,
RegionName: region.RegionName,
Nodes: make([]*DebugDERPNode, 0, len(region.Nodes)),
}
for _, node := range region.Nodes {
debugNode := &DebugDERPNode{
Name: node.Name,
HostName: node.HostName,
DERPPort: node.DERPPort,
STUNPort: node.STUNPort,
}
debugRegion.Nodes = append(debugRegion.Nodes, debugNode)
}
info.Regions[regionID] = debugRegion
}
return info
}
// DebugNodeStoreJSON returns the actual nodes map from the current NodeStore snapshot.
func (s *State) DebugNodeStoreJSON() map[types.NodeID]types.Node {
snapshot := s.nodeStore.data.Load()
return snapshot.nodesByID
}
// DebugPolicyManagerJSON returns structured debug information about the policy manager.
func (s *State) DebugPolicyManagerJSON() DebugStringInfo {
return DebugStringInfo{
Content: s.polMan.DebugString(),
}
}