| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | // Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved. | 
					
						
							|  |  |  | // Use of this source code is governed by a BSD-style | 
					
						
							|  |  |  | // license that can be found in the LICENSE file. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Package ipnstate captures the entire state of the Tailscale network. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It's a leaf package so ipn, wgengine, and magicsock can all depend on it. | 
					
						
							|  |  |  | package ipnstate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"html" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	"log" | 
					
						
							|  |  |  | 	"sort" | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	"sync" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-27 20:25:25 +00:00
										 |  |  | 	"inet.af/netaddr" | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	"tailscale.com/tailcfg" | 
					
						
							|  |  |  | 	"tailscale.com/types/key" | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 	"tailscale.com/util/dnsname" | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Status represents the entire state of the IPN network. | 
					
						
							|  |  |  | type Status struct { | 
					
						
							| 
									
										
										
										
											2021-03-18 21:07:58 -07:00
										 |  |  | 	// Version is the daemon's long version (see version.Long). | 
					
						
							|  |  |  | 	Version string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// BackendState is an ipn.State string value: | 
					
						
							|  |  |  | 	//  "NoState", "NeedsLogin", "NeedsMachineAuth", "Stopped", | 
					
						
							|  |  |  | 	//  "Starting", "Running". | 
					
						
							| 
									
										
										
										
											2021-01-22 14:28:44 -08:00
										 |  |  | 	BackendState string | 
					
						
							| 
									
										
										
										
											2021-03-18 21:07:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 06:43:33 -08:00
										 |  |  | 	AuthURL      string       // current URL provided by control to authorize client | 
					
						
							| 
									
										
										
										
											2021-01-22 14:28:44 -08:00
										 |  |  | 	TailscaleIPs []netaddr.IP // Tailscale IP(s) assigned to this node | 
					
						
							|  |  |  | 	Self         *PeerStatus | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// MagicDNSSuffix is the network's MagicDNS suffix for nodes | 
					
						
							|  |  |  | 	// in the network such as "userfoo.tailscale.net". | 
					
						
							|  |  |  | 	// There are no surrounding dots. | 
					
						
							|  |  |  | 	// MagicDNSSuffix should be populated regardless of whether a domain | 
					
						
							|  |  |  | 	// has MagicDNS enabled. | 
					
						
							|  |  |  | 	MagicDNSSuffix string | 
					
						
							| 
									
										
										
										
											2020-08-26 07:26:10 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Peer map[key.Public]*PeerStatus | 
					
						
							|  |  |  | 	User map[tailcfg.UserID]tailcfg.UserProfile | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (s *Status) Peers() []key.Public { | 
					
						
							|  |  |  | 	kk := make([]key.Public, 0, len(s.Peer)) | 
					
						
							|  |  |  | 	for k := range s.Peer { | 
					
						
							|  |  |  | 		kk = append(kk, k) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sort.Slice(kk, func(i, j int) bool { return bytes.Compare(kk[i][:], kk[j][:]) < 0 }) | 
					
						
							|  |  |  | 	return kk | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-04 13:12:42 -08:00
										 |  |  | type PeerStatusLite struct { | 
					
						
							|  |  |  | 	TxBytes, RxBytes int64 | 
					
						
							|  |  |  | 	LastHandshake    time.Time | 
					
						
							|  |  |  | 	NodeKey          tailcfg.NodeKey | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | type PeerStatus struct { | 
					
						
							| 
									
										
										
										
											2021-05-07 07:27:14 -07:00
										 |  |  | 	ID        tailcfg.StableNodeID | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	PublicKey key.Public | 
					
						
							|  |  |  | 	HostName  string // HostInfo's Hostname (not a DNS name or necessarily unique) | 
					
						
							| 
									
										
										
										
											2020-08-27 13:24:59 -07:00
										 |  |  | 	DNSName   string | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	OS        string // HostInfo.OS | 
					
						
							|  |  |  | 	UserID    tailcfg.UserID | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 07:20:27 -07:00
										 |  |  | 	TailAddrDeprecated string       `json:"TailAddr"` // Tailscale IP | 
					
						
							|  |  |  | 	TailscaleIPs       []netaddr.IP // Tailscale IP(s) assigned to this node | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Endpoints: | 
					
						
							|  |  |  | 	Addrs   []string | 
					
						
							|  |  |  | 	CurAddr string // one of Addrs, or unique if roaming | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 	Relay   string // DERP region | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	RxBytes       int64 | 
					
						
							|  |  |  | 	TxBytes       int64 | 
					
						
							|  |  |  | 	Created       time.Time // time registered with tailcontrol | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 	LastWrite     time.Time // time last packet sent | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	LastSeen      time.Time // last seen to tailcontrol | 
					
						
							|  |  |  | 	LastHandshake time.Time // with local wireguard | 
					
						
							|  |  |  | 	KeepAlive     bool | 
					
						
							| 
									
										
										
										
											2021-02-05 13:07:48 -08:00
										 |  |  | 	ExitNode      bool // true if this is the currently selected exit node. | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-16 10:57:46 -07:00
										 |  |  | 	PeerAPIURL   []string | 
					
						
							|  |  |  | 	Capabilities []string `json:",omitempty"` | 
					
						
							| 
									
										
										
										
											2021-03-25 15:38:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-30 18:05:51 -08:00
										 |  |  | 	// ShareeNode indicates this node exists in the netmap because | 
					
						
							|  |  |  | 	// it's owned by a shared-to user and that node might connect | 
					
						
							|  |  |  | 	// to us. These nodes should be hidden by "tailscale status" | 
					
						
							|  |  |  | 	// etc by default. | 
					
						
							|  |  |  | 	ShareeNode bool `json:",omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	// InNetworkMap means that this peer was seen in our latest network map. | 
					
						
							|  |  |  | 	// In theory, all of InNetworkMap and InMagicSock and InEngine should all be true. | 
					
						
							|  |  |  | 	InNetworkMap bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// InMagicSock means that this peer is being tracked by magicsock. | 
					
						
							|  |  |  | 	// In theory, all of InNetworkMap and InMagicSock and InEngine should all be true. | 
					
						
							|  |  |  | 	InMagicSock bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// InEngine means that this peer is tracked by the wireguard engine. | 
					
						
							|  |  |  | 	// In theory, all of InNetworkMap and InMagicSock and InEngine should all be true. | 
					
						
							|  |  |  | 	InEngine bool | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type StatusBuilder struct { | 
					
						
							|  |  |  | 	mu     sync.Mutex | 
					
						
							|  |  |  | 	locked bool | 
					
						
							|  |  |  | 	st     Status | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-25 15:38:40 -07:00
										 |  |  | // MutateStatus calls f with the status to mutate. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It may not assume other fields of status are already populated, and | 
					
						
							|  |  |  | // may not retain or write to the Status after f returns. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // MutateStatus acquires a lock so f must not call back into sb. | 
					
						
							|  |  |  | func (sb *StatusBuilder) MutateStatus(f func(*Status)) { | 
					
						
							| 
									
										
										
										
											2021-01-10 12:03:01 -08:00
										 |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2021-03-25 15:38:40 -07:00
										 |  |  | 	f(&sb.st) | 
					
						
							| 
									
										
										
										
											2021-01-10 12:03:01 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | func (sb *StatusBuilder) Status() *Status { | 
					
						
							|  |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							|  |  |  | 	sb.locked = true | 
					
						
							|  |  |  | 	return &sb.st | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-25 15:38:40 -07:00
										 |  |  | // MutateSelfStatus calls f with the PeerStatus of our own node to mutate. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It may not assume other fields of status are already populated, and | 
					
						
							|  |  |  | // may not retain or write to the Status after f returns. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // MutateStatus acquires a lock so f must not call back into sb. | 
					
						
							|  |  |  | func (sb *StatusBuilder) MutateSelfStatus(f func(*PeerStatus)) { | 
					
						
							| 
									
										
										
										
											2020-08-26 07:26:10 +08:00
										 |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2021-03-25 15:38:40 -07:00
										 |  |  | 	if sb.st.Self == nil { | 
					
						
							|  |  |  | 		sb.st.Self = new(PeerStatus) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f(sb.st.Self) | 
					
						
							| 
									
										
										
										
											2020-08-26 07:26:10 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | // AddUser adds a user profile to the status. | 
					
						
							| 
									
										
										
										
											2020-09-29 21:39:43 -07:00
										 |  |  | func (sb *StatusBuilder) AddUser(id tailcfg.UserID, up tailcfg.UserProfile) { | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							|  |  |  | 	if sb.locked { | 
					
						
							|  |  |  | 		log.Printf("[unexpected] ipnstate: AddUser after Locked") | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if sb.st.User == nil { | 
					
						
							|  |  |  | 		sb.st.User = make(map[tailcfg.UserID]tailcfg.UserProfile) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 21:39:43 -07:00
										 |  |  | 	sb.st.User[id] = up | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-27 20:25:25 +00:00
										 |  |  | // AddIP adds a Tailscale IP address to the status. | 
					
						
							|  |  |  | func (sb *StatusBuilder) AddTailscaleIP(ip netaddr.IP) { | 
					
						
							|  |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							|  |  |  | 	if sb.locked { | 
					
						
							|  |  |  | 		log.Printf("[unexpected] ipnstate: AddIP after Locked") | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sb.st.TailscaleIPs = append(sb.st.TailscaleIPs, ip) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | // AddPeer adds a peer node to the status. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Its PeerStatus is mixed with any previous status already added. | 
					
						
							|  |  |  | func (sb *StatusBuilder) AddPeer(peer key.Public, st *PeerStatus) { | 
					
						
							|  |  |  | 	if st == nil { | 
					
						
							|  |  |  | 		panic("nil PeerStatus") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sb.mu.Lock() | 
					
						
							|  |  |  | 	defer sb.mu.Unlock() | 
					
						
							|  |  |  | 	if sb.locked { | 
					
						
							|  |  |  | 		log.Printf("[unexpected] ipnstate: AddPeer after Locked") | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if sb.st.Peer == nil { | 
					
						
							|  |  |  | 		sb.st.Peer = make(map[key.Public]*PeerStatus) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	e, ok := sb.st.Peer[peer] | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		sb.st.Peer[peer] = st | 
					
						
							|  |  |  | 		st.PublicKey = peer | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-07 07:27:14 -07:00
										 |  |  | 	if v := st.ID; v != "" { | 
					
						
							|  |  |  | 		e.ID = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	if v := st.HostName; v != "" { | 
					
						
							|  |  |  | 		e.HostName = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-08-27 13:24:59 -07:00
										 |  |  | 	if v := st.DNSName; v != "" { | 
					
						
							|  |  |  | 		e.DNSName = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 	if v := st.Relay; v != "" { | 
					
						
							|  |  |  | 		e.Relay = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	if v := st.UserID; v != 0 { | 
					
						
							|  |  |  | 		e.UserID = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-14 07:20:27 -07:00
										 |  |  | 	if v := st.TailAddrDeprecated; v != "" { | 
					
						
							|  |  |  | 		e.TailAddrDeprecated = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.TailscaleIPs; v != nil { | 
					
						
							|  |  |  | 		e.TailscaleIPs = v | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.OS; v != "" { | 
					
						
							|  |  |  | 		e.OS = st.OS | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.Addrs; v != nil { | 
					
						
							|  |  |  | 		e.Addrs = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.CurAddr; v != "" { | 
					
						
							|  |  |  | 		e.CurAddr = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.RxBytes; v != 0 { | 
					
						
							|  |  |  | 		e.RxBytes = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.TxBytes; v != 0 { | 
					
						
							|  |  |  | 		e.TxBytes = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.LastHandshake; !v.IsZero() { | 
					
						
							|  |  |  | 		e.LastHandshake = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.Created; !v.IsZero() { | 
					
						
							|  |  |  | 		e.Created = v | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if v := st.LastSeen; !v.IsZero() { | 
					
						
							|  |  |  | 		e.LastSeen = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 	if v := st.LastWrite; !v.IsZero() { | 
					
						
							|  |  |  | 		e.LastWrite = v | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | 	if st.InNetworkMap { | 
					
						
							|  |  |  | 		e.InNetworkMap = true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if st.InMagicSock { | 
					
						
							|  |  |  | 		e.InMagicSock = true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if st.InEngine { | 
					
						
							|  |  |  | 		e.InEngine = true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if st.KeepAlive { | 
					
						
							|  |  |  | 		e.KeepAlive = true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-02-05 13:07:48 -08:00
										 |  |  | 	if st.ExitNode { | 
					
						
							|  |  |  | 		e.ExitNode = true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-11-30 18:05:51 -08:00
										 |  |  | 	if st.ShareeNode { | 
					
						
							|  |  |  | 		e.ShareeNode = true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-25 22:57:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type StatusUpdater interface { | 
					
						
							|  |  |  | 	UpdateStatus(*StatusBuilder) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (st *Status) WriteHTML(w io.Writer) { | 
					
						
							|  |  |  | 	f := func(format string, args ...interface{}) { fmt.Fprintf(w, format, args...) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 	f(`<!DOCTYPE html> | 
					
						
							|  |  |  | <html lang="en"> | 
					
						
							|  |  |  | <head> | 
					
						
							|  |  |  | <title>Tailscale State</title> | 
					
						
							|  |  |  | <style> | 
					
						
							|  |  |  | body { font-family: monospace; } | 
					
						
							|  |  |  | .owner { text-decoration: underline; } | 
					
						
							|  |  |  | .tailaddr { font-style: italic; } | 
					
						
							|  |  |  | .acenter { text-align: center; } | 
					
						
							|  |  |  | .aright { text-align: right; } | 
					
						
							|  |  |  | table, th, td { border: 1px solid black; border-spacing : 0; border-collapse : collapse; } | 
					
						
							|  |  |  | thead { background-color: #FFA500; } | 
					
						
							|  |  |  | th, td { padding: 5px; } | 
					
						
							|  |  |  | td { vertical-align: top; } | 
					
						
							|  |  |  | table tbody tr:nth-child(even) td { background-color: #f5f5f5; } | 
					
						
							|  |  |  | </style> | 
					
						
							|  |  |  | </head> | 
					
						
							|  |  |  | <body> | 
					
						
							|  |  |  | <h1>Tailscale State</h1> | 
					
						
							|  |  |  | `) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 	//f("<p><b>logid:</b> %s</p>\n", logid) | 
					
						
							|  |  |  | 	//f("<p><b>opts:</b> <code>%s</code></p>\n", html.EscapeString(fmt.Sprintf("%+v", opts))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-27 20:25:25 +00:00
										 |  |  | 	ips := make([]string, 0, len(st.TailscaleIPs)) | 
					
						
							|  |  |  | 	for _, ip := range st.TailscaleIPs { | 
					
						
							|  |  |  | 		ips = append(ips, ip.String()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f("<p>Tailscale IP: %s", strings.Join(ips, ", ")) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 	f("<table>\n<thead>\n") | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 	f("<tr><th>Peer</th><th>OS</th><th>Node</th><th>Owner</th><th>Rx</th><th>Tx</th><th>Activity</th><th>Connection</th></tr>\n") | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 	f("</thead>\n<tbody>\n") | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 	var peers []*PeerStatus | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 	for _, peer := range st.Peers() { | 
					
						
							|  |  |  | 		ps := st.Peer[peer] | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 		if ps.ShareeNode { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		peers = append(peers, ps) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	SortPeers(peers) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, ps := range peers { | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 		var actAgo string | 
					
						
							|  |  |  | 		if !ps.LastWrite.IsZero() { | 
					
						
							|  |  |  | 			ago := now.Sub(ps.LastWrite) | 
					
						
							|  |  |  | 			actAgo = ago.Round(time.Second).String() + " ago" | 
					
						
							|  |  |  | 			if ago < 5*time.Minute { | 
					
						
							|  |  |  | 				actAgo = "<b>" + actAgo + "</b>" | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var owner string | 
					
						
							|  |  |  | 		if up, ok := st.User[ps.UserID]; ok { | 
					
						
							|  |  |  | 			owner = up.LoginName | 
					
						
							|  |  |  | 			if i := strings.Index(owner, "@"); i != -1 { | 
					
						
							|  |  |  | 				owner = owner[:i] | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-18 17:15:38 -05:00
										 |  |  | 		hostName := dnsname.SanitizeHostname(ps.HostName) | 
					
						
							|  |  |  | 		dnsName := dnsname.TrimSuffix(ps.DNSName, st.MagicDNSSuffix) | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 		if strings.EqualFold(dnsName, hostName) || ps.UserID != st.Self.UserID { | 
					
						
							|  |  |  | 			hostName = "" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var hostNameHTML string | 
					
						
							|  |  |  | 		if hostName != "" { | 
					
						
							|  |  |  | 			hostNameHTML = "<br>" + html.EscapeString(hostName) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 07:20:27 -07:00
										 |  |  | 		var tailAddr string | 
					
						
							|  |  |  | 		if len(ps.TailscaleIPs) > 0 { | 
					
						
							|  |  |  | 			tailAddr = ps.TailscaleIPs[0].String() | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 		f("<tr><td>%s</td><td class=acenter>%s</td>"+ | 
					
						
							|  |  |  | 			"<td><b>%s</b>%s<div class=\"tailaddr\">%s</div></td><td class=\"acenter owner\">%s</td><td class=\"aright\">%v</td><td class=\"aright\">%v</td><td class=\"aright\">%v</td>", | 
					
						
							|  |  |  | 			ps.PublicKey.ShortString(), | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 			osEmoji(ps.OS), | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 			html.EscapeString(dnsName), | 
					
						
							|  |  |  | 			hostNameHTML, | 
					
						
							| 
									
										
										
										
											2021-04-14 07:20:27 -07:00
										 |  |  | 			tailAddr, | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 			html.EscapeString(owner), | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 			ps.RxBytes, | 
					
						
							|  |  |  | 			ps.TxBytes, | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 			actAgo, | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 		) | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 		f("<td>") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 		// TODO: let server report this active bool instead | 
					
						
							|  |  |  | 		active := !ps.LastWrite.IsZero() && time.Since(ps.LastWrite) < 2*time.Minute | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 		if active { | 
					
						
							|  |  |  | 			if ps.Relay != "" && ps.CurAddr == "" { | 
					
						
							|  |  |  | 				f("relay <b>%s</b>", html.EscapeString(ps.Relay)) | 
					
						
							|  |  |  | 			} else if ps.CurAddr != "" { | 
					
						
							|  |  |  | 				f("direct <b>%s</b>", html.EscapeString(ps.CurAddr)) | 
					
						
							| 
									
										
										
										
											2020-07-03 13:44:22 -07:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 		f("</td>") // end Addrs | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		f("</tr>\n") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-05-01 23:55:16 +02:00
										 |  |  | 	f("</tbody>\n</table>\n") | 
					
						
							|  |  |  | 	f("</body>\n</html>\n") | 
					
						
							| 
									
										
										
										
											2020-03-27 13:26:35 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func osEmoji(os string) string { | 
					
						
							|  |  |  | 	switch os { | 
					
						
							|  |  |  | 	case "linux": | 
					
						
							|  |  |  | 		return "🐧" | 
					
						
							|  |  |  | 	case "macOS": | 
					
						
							|  |  |  | 		return "🍎" | 
					
						
							|  |  |  | 	case "windows": | 
					
						
							|  |  |  | 		return "🖥️" | 
					
						
							|  |  |  | 	case "iOS": | 
					
						
							|  |  |  | 		return "📱" | 
					
						
							|  |  |  | 	case "android": | 
					
						
							|  |  |  | 		return "🤖" | 
					
						
							|  |  |  | 	case "freebsd": | 
					
						
							|  |  |  | 		return "👿" | 
					
						
							|  |  |  | 	case "openbsd": | 
					
						
							|  |  |  | 		return "🐡" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return "👽" | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-09 14:49:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // PingResult contains response information for the "tailscale ping" subcommand, | 
					
						
							|  |  |  | // saying how Tailscale can reach a Tailscale IP or subnet-routed IP. | 
					
						
							|  |  |  | type PingResult struct { | 
					
						
							|  |  |  | 	IP       string // ping destination | 
					
						
							|  |  |  | 	NodeIP   string // Tailscale IP of node handling IP (different for subnet routers) | 
					
						
							|  |  |  | 	NodeName string // DNS name base or (possibly not unique) hostname | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Err            string | 
					
						
							|  |  |  | 	LatencySeconds float64 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 15:16:15 -07:00
										 |  |  | 	// Endpoint is the ip:port if direct UDP was used. | 
					
						
							|  |  |  | 	// It is not currently set for TSMP pings. | 
					
						
							|  |  |  | 	Endpoint string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// DERPRegionID is non-zero DERP region ID if DERP was used. | 
					
						
							|  |  |  | 	// It is not currently set for TSMP pings. | 
					
						
							|  |  |  | 	DERPRegionID int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// DERPRegionCode is the three-letter region code | 
					
						
							|  |  |  | 	// corresponding to DERPRegionID. | 
					
						
							|  |  |  | 	// It is not currently set for TSMP pings. | 
					
						
							|  |  |  | 	DERPRegionCode string | 
					
						
							| 
									
										
										
										
											2020-08-09 14:49:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-29 15:17:05 -07:00
										 |  |  | 	// PeerAPIPort is set by TSMP ping responses for peers that | 
					
						
							|  |  |  | 	// are running a peerapi server. This is the port they're | 
					
						
							|  |  |  | 	// running the server on. | 
					
						
							|  |  |  | 	PeerAPIPort uint16 `json:",omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-09 14:49:42 -07:00
										 |  |  | 	// TODO(bradfitz): details like whether port mapping was used on either side? (Once supported) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func SortPeers(peers []*PeerStatus) { | 
					
						
							|  |  |  | 	sort.Slice(peers, func(i, j int) bool { return sortKey(peers[i]) < sortKey(peers[j]) }) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func sortKey(ps *PeerStatus) string { | 
					
						
							|  |  |  | 	if ps.DNSName != "" { | 
					
						
							|  |  |  | 		return ps.DNSName | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if ps.HostName != "" { | 
					
						
							|  |  |  | 		return ps.HostName | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-14 07:20:27 -07:00
										 |  |  | 	// TODO(bradfitz): add PeerStatus.Less and avoid these allocs in a Less func. | 
					
						
							|  |  |  | 	if len(ps.TailscaleIPs) > 0 { | 
					
						
							|  |  |  | 		return ps.TailscaleIPs[0].String() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return string(ps.PublicKey[:]) | 
					
						
							| 
									
										
										
										
											2021-01-26 08:28:34 -08:00
										 |  |  | } |