2020-02-05 22:16:58 +00: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.
2021-02-04 21:12:42 +00:00
package ipnlocal
2020-02-05 22:16:58 +00:00
import (
2020-09-28 22:28:26 +00:00
"bytes"
2020-03-14 03:53:58 +00:00
"context"
2020-02-03 18:35:52 +00:00
"errors"
2020-02-05 22:16:58 +00:00
"fmt"
2021-03-30 19:56:00 +00:00
"io"
2021-03-26 20:44:55 +00:00
"net"
2021-04-07 15:39:08 +00:00
"net/http"
2020-09-17 14:59:55 +00:00
"os"
2021-03-31 18:55:21 +00:00
"os/exec"
2021-03-29 17:42:33 +00:00
"path/filepath"
2020-10-21 19:55:03 +00:00
"runtime"
2021-04-08 21:54:25 +00:00
"sort"
2021-03-26 20:44:55 +00:00
"strconv"
2020-02-05 22:16:58 +00:00
"strings"
"sync"
2021-04-02 15:21:40 +00:00
"sync/atomic"
2021-04-06 20:38:47 +00:00
"syscall"
2020-02-05 22:16:58 +00:00
"time"
2020-05-11 21:02:12 +00:00
"inet.af/netaddr"
2020-02-05 22:16:58 +00:00
"tailscale.com/control/controlclient"
2021-02-25 05:29:51 +00:00
"tailscale.com/health"
2020-07-29 01:47:23 +00:00
"tailscale.com/internal/deepprint"
2021-02-04 21:12:42 +00:00
"tailscale.com/ipn"
2020-03-26 05:57:46 +00:00
"tailscale.com/ipn/ipnstate"
2020-04-01 04:48:33 +00:00
"tailscale.com/ipn/policy"
2021-03-25 21:50:21 +00:00
"tailscale.com/net/dns"
2020-08-28 04:25:17 +00:00
"tailscale.com/net/interfaces"
2020-07-31 20:27:09 +00:00
"tailscale.com/net/tsaddr"
2021-03-29 17:42:33 +00:00
"tailscale.com/paths"
2020-02-05 22:16:58 +00:00
"tailscale.com/portlist"
"tailscale.com/tailcfg"
2020-02-14 21:09:19 +00:00
"tailscale.com/types/empty"
2020-03-26 05:57:46 +00:00
"tailscale.com/types/key"
2020-02-15 03:23:16 +00:00
"tailscale.com/types/logger"
2021-02-05 23:44:46 +00:00
"tailscale.com/types/netmap"
2021-02-05 23:23:01 +00:00
"tailscale.com/types/persist"
2020-12-30 01:22:56 +00:00
"tailscale.com/types/wgkey"
2021-04-09 22:24:47 +00:00
"tailscale.com/util/dnsname"
2020-11-24 23:35:04 +00:00
"tailscale.com/util/systemd"
2020-02-05 22:16:58 +00:00
"tailscale.com/version"
"tailscale.com/wgengine"
"tailscale.com/wgengine/filter"
2020-05-11 21:02:12 +00:00
"tailscale.com/wgengine/router"
2021-01-29 20:16:36 +00:00
"tailscale.com/wgengine/wgcfg"
2021-02-05 20:44:43 +00:00
"tailscale.com/wgengine/wgcfg/nmcfg"
2020-02-05 22:16:58 +00:00
)
2020-10-20 17:40:12 +00:00
var controlDebugFlags = getControlDebugFlags ( )
func getControlDebugFlags ( ) [ ] string {
if e := os . Getenv ( "TS_DEBUG_CONTROL_FLAGS" ) ; e != "" {
return strings . Split ( e , "," )
}
return nil
}
2020-10-19 22:56:59 +00:00
2020-05-19 02:32:20 +00:00
// LocalBackend is the glue between the major pieces of the Tailscale
// network software: the cloud control plane (via controlclient), the
// network data plane (via wgengine), and the user-facing UIs and CLIs
// (collectively called "frontends", via LocalBackend's implementation
// of the Backend interface).
//
// LocalBackend implements the overall state machine for the Tailscale
// application. Frontends, controlclient and wgengine can feed events
// into LocalBackend to advance the state machine, and advancing the
// state machine generates events back out to zero or more components.
2020-02-05 22:16:58 +00:00
type LocalBackend struct {
2020-05-19 02:32:20 +00:00
// Elements that are thread-safe or constant after construction.
2021-03-16 05:20:48 +00:00
ctx context . Context // canceled by Close
ctxCancel context . CancelFunc // cancels ctx
logf logger . Logf // general logging
keyLogf logger . Logf // for printing list of peers on change
statsLogf logger . Logf // for printing peers stats on change
e wgengine . Engine
store ipn . StateStore
backendLogID string
unregisterLinkMon func ( )
unregisterHealthWatch func ( )
portpoll * portlist . Poller // may be nil
portpollOnce sync . Once // guards starting readPoller
gotPortPollRes chan struct { } // closed upon first readPoller result
serverURL string // tailcontrol URL
newDecompressor func ( ) ( controlclient . Decompressor , error )
2020-05-19 02:32:20 +00:00
2020-07-29 01:47:23 +00:00
filterHash string
2020-02-05 22:16:58 +00:00
// The mutex protects the following elements.
2020-09-28 22:28:26 +00:00
mu sync . Mutex
2021-04-07 15:39:08 +00:00
httpTestClient * http . Client // for controlclient. nil by default, used by tests.
2021-02-04 21:12:42 +00:00
notify func ( ipn . Notify )
2021-04-08 04:12:16 +00:00
cc * controlclient . Client
2021-02-04 21:12:42 +00:00
stateKey ipn . StateKey // computed in part from user-provided value
userID string // current controlling user ID (for Windows, primarily)
prefs * ipn . Prefs
2020-11-02 17:52:59 +00:00
inServerMode bool
2020-12-30 01:22:56 +00:00
machinePrivKey wgkey . Private
2021-02-04 21:12:42 +00:00
state ipn . State
2020-06-15 23:04:12 +00:00
// hostinfo is mutated in-place while mu is held.
hostinfo * tailcfg . Hostinfo
// netMap is not mutated in-place once set.
2021-03-25 22:38:40 +00:00
netMap * netmap . NetworkMap
nodeByAddr map [ netaddr . IP ] * tailcfg . Node
activeLogin string // last logged LoginName from netMap
engineStatus ipn . EngineStatus
endpoints [ ] string
blocked bool
authURL string
interact bool
prevIfState * interfaces . State
2021-03-30 18:19:42 +00:00
peerAPIServer * peerAPIServer // or nil
2021-03-25 22:38:40 +00:00
peerAPIListeners [ ] * peerAPIListener
2021-04-08 21:54:25 +00:00
incomingFiles map [ * incomingFile ] bool
2021-04-12 21:05:44 +00:00
// directFileRoot, if non-empty, means to write received files
// directly to this directory, without staging them in an
// intermediate buffered directory for "pick-up" later. If
// empty, the files are received in a daemon-owned location
// and the localapi is used to enumerate, download, and delete
// them. This is used on macOS where the GUI lifetime is the
// same as the Network Extension lifetime and we can thus avoid
// double-copying files by writing them to the right location
// immediately.
directFileRoot string
2020-02-05 22:16:58 +00:00
2020-05-19 02:32:20 +00:00
// statusLock must be held before calling statusChanged.Wait() or
2020-02-05 22:16:58 +00:00
// statusChanged.Broadcast().
statusLock sync . Mutex
statusChanged * sync . Cond
}
2020-02-03 18:35:52 +00:00
// NewLocalBackend returns a new LocalBackend that is ready to run,
// but is not actually running.
2021-02-04 21:12:42 +00:00
func NewLocalBackend ( logf logger . Logf , logid string , store ipn . StateStore , e wgengine . Engine ) ( * LocalBackend , error ) {
2020-02-05 22:16:58 +00:00
if e == nil {
panic ( "ipn.NewLocalBackend: wgengine must not be nil" )
}
2021-03-10 00:10:30 +00:00
// Default filter blocks everything and logs nothing, until Start() is called.
e . SetFilter ( filter . NewAllowNone ( logf , & netaddr . IPSet { } ) )
2020-02-05 22:16:58 +00:00
2020-03-14 03:53:58 +00:00
ctx , cancel := context . WithCancel ( context . Background ( ) )
2020-02-05 22:16:58 +00:00
portpoll , err := portlist . NewPoller ( )
if err != nil {
2020-04-11 15:35:34 +00:00
logf ( "skipping portlist: %s" , err )
2020-02-05 22:16:58 +00:00
}
2020-02-25 15:36:32 +00:00
b := & LocalBackend {
2020-10-14 21:07:40 +00:00
ctx : ctx ,
ctxCancel : cancel ,
logf : logf ,
keyLogf : logger . LogOnChange ( logf , 5 * time . Minute , time . Now ) ,
2020-10-29 22:26:10 +00:00
statsLogf : logger . LogOnChange ( logf , 5 * time . Minute , time . Now ) ,
2020-10-14 21:07:40 +00:00
e : e ,
store : store ,
backendLogID : logid ,
2021-02-04 21:12:42 +00:00
state : ipn . NoState ,
2020-10-14 21:07:40 +00:00
portpoll : portpoll ,
gotPortPollRes : make ( chan struct { } ) ,
2020-02-05 22:16:58 +00:00
}
b . statusChanged = sync . NewCond ( & b . statusLock )
2021-03-02 04:45:30 +00:00
linkMon := e . GetLinkMonitor ( )
2021-03-26 04:41:37 +00:00
b . prevIfState = linkMon . InterfaceState ( )
2021-03-02 04:45:30 +00:00
// Call our linkChange code once with the current state, and
// then also whenever it changes:
b . linkChange ( false , linkMon . InterfaceState ( ) )
b . unregisterLinkMon = linkMon . RegisterChangeCallback ( b . linkChange )
2021-03-16 05:20:48 +00:00
b . unregisterHealthWatch = health . RegisterWatcher ( b . onHealthChange )
2021-03-29 22:17:05 +00:00
wiredPeerAPIPort := false
if ig , ok := e . ( wgengine . InternalsGetter ) ; ok {
if tunWrap , _ , ok := ig . GetInternals ( ) ; ok {
tunWrap . PeerAPIPort = b . getPeerAPIPortForTSMPPing
wiredPeerAPIPort = true
}
}
if ! wiredPeerAPIPort {
b . logf ( "[unexpected] failed to wire up peer API port for engine %T" , e )
}
2020-02-25 15:36:32 +00:00
return b , nil
2020-02-05 22:16:58 +00:00
}
2021-04-12 21:05:44 +00:00
// SetDirectFileRoot sets the directory to download files to directly,
// without buffering them through an intermediate daemon-owned
// tailcfg.UserID-specific directory.
//
// This must be called before the LocalBackend starts being used.
func ( b * LocalBackend ) SetDirectFileRoot ( dir string ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
b . directFileRoot = dir
}
2021-03-02 04:45:30 +00:00
// linkChange is our link monitor callback, called whenever the network changes.
// major is whether ifst is different than earlier.
2020-08-28 04:25:17 +00:00
func ( b * LocalBackend ) linkChange ( major bool , ifst * interfaces . State ) {
2020-10-02 05:03:25 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2020-10-05 22:12:35 +00:00
hadPAC := b . prevIfState . HasPAC ( )
b . prevIfState = ifst
2020-10-06 22:22:46 +00:00
networkUp := ifst . AnyInterfaceUp ( )
2021-04-08 04:12:16 +00:00
if b . cc != nil {
go b . cc . SetPaused ( b . state == ipn . Stopped || ! networkUp )
2020-10-06 22:22:46 +00:00
}
2020-10-05 22:12:35 +00:00
// If the PAC-ness of the network changed, reconfig wireguard+route to
// add/remove subnets.
if hadPAC != ifst . HasPAC ( ) {
b . logf ( "linkChange: in state %v; PAC changed from %v->%v" , b . state , hadPAC , ifst . HasPAC ( ) )
switch b . state {
2021-02-04 21:12:42 +00:00
case ipn . NoState , ipn . Stopped :
2020-10-05 22:12:35 +00:00
// Do nothing.
default :
go b . authReconfig ( )
2020-10-02 05:03:25 +00:00
}
}
2021-02-23 04:43:35 +00:00
// If the local network configuration has changed, our filter may
// need updating to tweak default routes.
b . updateFilter ( b . netMap , b . prefs )
2021-03-30 20:49:08 +00:00
if runtime . GOOS == "windows" && b . netMap != nil {
want := len ( b . netMap . Addresses )
b . logf ( "linkChange: peerAPIListeners too low; trying again" )
if len ( b . peerAPIListeners ) < want {
go b . initPeerAPIListener ( )
}
}
2020-08-28 04:25:17 +00:00
}
2021-03-16 05:20:48 +00:00
func ( b * LocalBackend ) onHealthChange ( sys health . Subsystem , err error ) {
if err == nil {
b . logf ( "health(%q): ok" , sys )
} else {
b . logf ( "health(%q): error: %v" , sys , err )
}
}
2020-05-19 02:32:20 +00:00
// Shutdown halts the backend and all its sub-components. The backend
// can no longer be used after Shutdown returns.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) Shutdown ( ) {
2020-05-19 02:32:20 +00:00
b . mu . Lock ( )
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-05-19 02:32:20 +00:00
b . mu . Unlock ( )
2020-05-21 20:30:20 +00:00
2021-03-01 20:56:03 +00:00
b . unregisterLinkMon ( )
2021-03-16 05:20:48 +00:00
b . unregisterHealthWatch ( )
2021-04-08 04:12:16 +00:00
if cc != nil {
cc . Shutdown ( )
2020-05-19 02:32:20 +00:00
}
2020-05-21 20:30:20 +00:00
b . ctxCancel ( )
2020-02-05 22:16:58 +00:00
b . e . Close ( )
b . e . Wait ( )
}
2021-04-07 15:27:35 +00:00
// Prefs returns a copy of b's current prefs, with any private keys removed.
func ( b * LocalBackend ) Prefs ( ) * ipn . Prefs {
b . mu . Lock ( )
defer b . mu . Unlock ( )
p := b . prefs . Clone ( )
if p != nil && p . Persist != nil {
p . Persist . LegacyFrontendPrivateMachineKey = wgkey . Private { }
p . Persist . PrivateNodeKey = wgkey . Private { }
p . Persist . OldPrivateNodeKey = wgkey . Private { }
}
return p
}
2020-05-19 02:32:20 +00:00
// Status returns the latest status of the backend and its
// sub-components.
2020-03-26 05:57:46 +00:00
func ( b * LocalBackend ) Status ( ) * ipnstate . Status {
sb := new ( ipnstate . StatusBuilder )
b . UpdateStatus ( sb )
return sb . Status ( )
}
2021-03-19 04:07:58 +00:00
// StatusWithoutPeers is like Status but omits any details
// of peers.
func ( b * LocalBackend ) StatusWithoutPeers ( ) * ipnstate . Status {
sb := new ( ipnstate . StatusBuilder )
b . updateStatus ( sb , nil )
return sb . Status ( )
}
2020-05-19 02:32:20 +00:00
// UpdateStatus implements ipnstate.StatusUpdater.
2020-03-26 05:57:46 +00:00
func ( b * LocalBackend ) UpdateStatus ( sb * ipnstate . StatusBuilder ) {
b . e . UpdateStatus ( sb )
2021-03-19 04:07:58 +00:00
b . updateStatus ( sb , b . populatePeerStatusLocked )
}
2020-03-26 05:57:46 +00:00
2021-03-19 04:07:58 +00:00
// updateStatus populates sb with status.
//
// extraLocked, if non-nil, is called while b.mu is still held.
func ( b * LocalBackend ) updateStatus ( sb * ipnstate . StatusBuilder , extraLocked func ( * ipnstate . StatusBuilder ) ) {
2020-03-26 05:57:46 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2021-03-25 22:38:40 +00:00
sb . MutateStatus ( func ( s * ipnstate . Status ) {
s . Version = version . Long
s . BackendState = b . state . String ( )
s . AuthURL = b . authURL
if b . netMap != nil {
s . MagicDNSSuffix = b . netMap . MagicDNSSuffix ( )
}
} )
sb . MutateSelfStatus ( func ( ss * ipnstate . PeerStatus ) {
for _ , pln := range b . peerAPIListeners {
2021-03-26 20:44:55 +00:00
ss . PeerAPIURL = append ( ss . PeerAPIURL , pln . urlStr )
2021-03-25 22:38:40 +00:00
}
} )
2020-03-26 05:57:46 +00:00
// TODO: hostinfo, and its networkinfo
// TODO: EngineStatus copy (and deprecate it?)
2021-03-19 04:07:58 +00:00
if extraLocked != nil {
extraLocked ( sb )
}
}
func ( b * LocalBackend ) populatePeerStatusLocked ( sb * ipnstate . StatusBuilder ) {
if b . netMap == nil {
return
}
for id , up := range b . netMap . UserProfiles {
sb . AddUser ( id , up )
}
for _ , p := range b . netMap . Peers {
var lastSeen time . Time
if p . LastSeen != nil {
lastSeen = * p . LastSeen
2020-03-26 05:57:46 +00:00
}
2021-03-19 04:07:58 +00:00
var tailAddr string
for _ , addr := range p . Addresses {
// The peer struct currently only allows a single
// Tailscale IP address. For compatibility with the
// old display, make sure it's the IPv4 address.
if addr . IP . Is4 ( ) && addr . IsSingleIP ( ) && tsaddr . IsTailscaleIP ( addr . IP ) {
tailAddr = addr . IP . String ( )
break
2020-03-26 05:57:46 +00:00
}
}
2021-03-19 04:07:58 +00:00
sb . AddPeer ( key . Public ( p . Key ) , & ipnstate . PeerStatus {
InNetworkMap : true ,
UserID : p . User ,
TailAddr : tailAddr ,
HostName : p . Hostinfo . Hostname ,
DNSName : p . Name ,
OS : p . Hostinfo . OS ,
KeepAlive : p . KeepAlive ,
Created : p . Created ,
LastSeen : lastSeen ,
ShareeNode : p . Hostinfo . ShareeNode ,
ExitNode : p . StableID != "" && p . StableID == b . prefs . ExitNodeID ,
} )
2020-03-26 05:57:46 +00:00
}
2021-01-28 23:29:17 +00:00
}
2020-03-26 05:57:46 +00:00
2021-03-15 21:59:35 +00:00
// WhoIs reports the node and user who owns the node with the given IP:port.
// If the IP address is a Tailscale IP, the provided port may be 0.
2021-01-28 23:29:17 +00:00
// If ok == true, n and u are valid.
2021-03-15 21:59:35 +00:00
func ( b * LocalBackend ) WhoIs ( ipp netaddr . IPPort ) ( n * tailcfg . Node , u tailcfg . UserProfile , ok bool ) {
2021-01-28 23:29:17 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2021-03-15 21:59:35 +00:00
n , ok = b . nodeByAddr [ ipp . IP ]
2021-01-28 23:29:17 +00:00
if ! ok {
2021-03-15 21:59:35 +00:00
var ip netaddr . IP
if ipp . Port != 0 {
ip , ok = b . e . WhoIsIPPort ( ipp )
}
if ! ok {
return nil , u , false
}
n , ok = b . nodeByAddr [ ip ]
if ! ok {
return nil , u , false
}
2021-01-28 23:29:17 +00:00
}
u , ok = b . netMap . UserProfiles [ n . User ]
if ! ok {
return nil , u , false
}
return n , u , true
2020-03-26 05:57:46 +00:00
}
2020-02-05 22:16:58 +00:00
// SetDecompressor sets a decompression function, which must be a zstd
// reader.
//
// This exists because the iOS/Mac NetworkExtension is very resource
// constrained, and the zstd package is too heavy to fit in the
// constrained RSS limit.
func ( b * LocalBackend ) SetDecompressor ( fn func ( ) ( controlclient . Decompressor , error ) ) {
b . newDecompressor = fn
}
2020-06-15 23:04:12 +00:00
// setClientStatus is the callback invoked by the control client whenever it posts a new status.
// Among other things, this is where we update the netmap, packet filters, DNS and DERP maps.
func ( b * LocalBackend ) setClientStatus ( st controlclient . Status ) {
2020-07-29 01:47:23 +00:00
// The following do not depend on any data for which we need to lock b.
if st . Err != "" {
// TODO(crawshaw): display in the UI.
2020-12-21 18:58:06 +00:00
if st . Err == "EOF" {
b . logf ( "[v1] Received error: EOF" )
} else {
b . logf ( "Received error: %v" , st . Err )
}
2020-07-29 01:47:23 +00:00
return
}
2020-06-15 23:04:12 +00:00
if st . LoginFinished != nil {
// Auth completed, unblock the engine
b . blockEngineUpdates ( false )
b . authReconfig ( )
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { LoginFinished : & empty . Message { } } )
2020-06-15 23:04:12 +00:00
}
2020-07-29 01:47:23 +00:00
prefsChanged := false
// Lock b once and do only the things that require locking.
b . mu . Lock ( )
prefs := b . prefs
stateKey := b . stateKey
netMap := b . netMap
interact := b . interact
2020-06-15 23:04:12 +00:00
if st . Persist != nil {
2020-07-30 08:36:06 +00:00
if ! b . prefs . Persist . Equals ( st . Persist ) {
2020-07-29 01:47:23 +00:00
prefsChanged = true
b . prefs . Persist = st . Persist . Clone ( )
}
}
if st . NetMap != nil {
2021-02-25 05:15:14 +00:00
if b . findExitNodeIDLocked ( st . NetMap ) {
2021-01-21 01:24:16 +00:00
prefsChanged = true
}
2020-10-27 19:51:48 +00:00
b . setNetMapLocked ( st . NetMap )
2020-07-29 01:47:23 +00:00
}
if st . URL != "" {
b . authURL = st . URL
}
2021-02-04 21:12:42 +00:00
if b . state == ipn . NeedsLogin {
2020-07-29 01:47:23 +00:00
if ! b . prefs . WantRunning {
prefsChanged = true
}
b . prefs . WantRunning = true
}
// Prefs will be written out; this is not safe unless locked or cloned.
if prefsChanged {
prefs = b . prefs . Clone ( )
}
2020-06-15 23:04:12 +00:00
2020-07-29 01:47:23 +00:00
b . mu . Unlock ( )
2020-06-15 23:04:12 +00:00
2020-07-29 01:47:23 +00:00
// Now complete the lock-free parts of what we started while locked.
if prefsChanged {
2020-06-15 23:04:12 +00:00
if stateKey != "" {
if err := b . store . WriteState ( stateKey , prefs . ToBytes ( ) ) ; err != nil {
b . logf ( "Failed to save new controlclient state: %v" , err )
}
}
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { Prefs : prefs } )
2020-06-15 23:04:12 +00:00
}
if st . NetMap != nil {
2020-07-29 01:47:23 +00:00
if netMap != nil {
diff := st . NetMap . ConciseDiffFrom ( netMap )
2020-06-15 23:04:12 +00:00
if strings . TrimSpace ( diff ) == "" {
2020-12-21 18:58:06 +00:00
b . logf ( "[v1] netmap diff: (none)" )
2020-06-15 23:04:12 +00:00
} else {
b . logf ( "netmap diff:\n%v" , diff )
}
}
2020-07-29 01:47:23 +00:00
b . updateFilter ( st . NetMap , prefs )
b . e . SetNetworkMap ( st . NetMap )
2020-09-18 14:44:01 +00:00
b . e . SetDERPMap ( st . NetMap . DERPMap )
2020-07-29 01:47:23 +00:00
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { NetMap : st . NetMap } )
2020-06-15 23:04:12 +00:00
}
if st . URL != "" {
b . logf ( "Received auth URL: %.20v..." , st . URL )
2020-10-27 20:57:10 +00:00
if interact {
2020-06-15 23:04:12 +00:00
b . popBrowserAuthNow ( )
}
}
b . stateMachine ( )
2020-07-29 01:47:23 +00:00
// This is currently (2020-07-28) necessary; conditionally disabling it is fragile!
// This is where netmap information gets propagated to router and magicsock.
b . authReconfig ( )
2020-06-15 23:04:12 +00:00
}
2021-02-25 05:15:14 +00:00
// findExitNodeIDLocked updates b.prefs to reference an exit node by ID,
2021-02-25 04:05:23 +00:00
// rather than by IP. It returns whether prefs was mutated.
2021-02-25 05:15:14 +00:00
func ( b * LocalBackend ) findExitNodeIDLocked ( nm * netmap . NetworkMap ) ( prefsChanged bool ) {
2021-01-21 01:24:16 +00:00
// If we have a desired IP on file, try to find the corresponding
// node.
2021-02-25 04:05:23 +00:00
if b . prefs . ExitNodeIP . IsZero ( ) {
return false
}
2021-01-21 01:24:16 +00:00
2021-02-25 04:05:23 +00:00
// IP takes precedence over ID, so if both are set, clear ID.
if b . prefs . ExitNodeID != "" {
b . prefs . ExitNodeID = ""
prefsChanged = true
2021-01-21 01:24:16 +00:00
}
for _ , peer := range nm . Peers {
2021-02-25 04:05:23 +00:00
for _ , addr := range peer . Addresses {
if ! addr . IsSingleIP ( ) || addr . IP != b . prefs . ExitNodeIP {
2021-01-21 01:24:16 +00:00
continue
}
2021-02-25 04:05:23 +00:00
// Found the node being referenced, upgrade prefs to
// reference it directly for next time.
b . prefs . ExitNodeID = peer . StableID
b . prefs . ExitNodeIP = netaddr . IP { }
return true
2021-01-21 01:24:16 +00:00
}
}
2021-02-25 04:05:23 +00:00
return false
2021-01-21 01:24:16 +00:00
}
2020-06-15 23:04:12 +00:00
// setWgengineStatus is the callback by the wireguard engine whenever it posts a new status.
// This updates the endpoints both in the backend and in the control client.
func ( b * LocalBackend ) setWgengineStatus ( s * wgengine . Status , err error ) {
if err != nil {
2021-01-07 04:18:29 +00:00
b . logf ( "wgengine status error: %v" , err )
2020-06-15 23:04:12 +00:00
return
}
if s == nil {
b . logf ( "[unexpected] non-error wgengine update with status=nil: %v" , s )
return
}
b . mu . Lock ( )
2020-10-29 22:26:10 +00:00
es := b . parseWgStatusLocked ( s )
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-06-15 23:04:12 +00:00
b . engineStatus = es
b . endpoints = append ( [ ] string { } , s . LocalAddrs ... )
b . mu . Unlock ( )
2021-04-08 04:12:16 +00:00
if cc != nil {
cc . UpdateEndpoints ( 0 , s . LocalAddrs )
2020-06-15 23:04:12 +00:00
}
b . stateMachine ( )
b . statusLock . Lock ( )
b . statusChanged . Broadcast ( )
b . statusLock . Unlock ( )
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { Engine : & es } )
2020-06-15 23:04:12 +00:00
}
2021-04-07 05:11:50 +00:00
func ( b * LocalBackend ) SetNotifyCallback ( notify func ( ipn . Notify ) ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
b . notify = notify
}
2021-04-07 15:39:08 +00:00
// SetHTTPTestClient sets an alternate HTTP client to use with
// connections to the coordination server. It exists for
// testing. Using nil means to use the default.
func ( b * LocalBackend ) SetHTTPTestClient ( c * http . Client ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
b . httpTestClient = c
}
2020-05-19 02:32:20 +00:00
// Start applies the configuration specified in opts, and starts the
// state machine.
//
// TODO(danderson): this function is trying to do too many things at
// once: it loads state, or imports it, or updates prefs sometimes,
// contains some settings that are one-shot things done by `tailscale
// up` because we had nowhere else to put them, and there's no clear
// guarantee that switching from one user's state to another is
// actually a supported operation (it should be, but it's very unclear
// from the following whether or not that is a safe transition).
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) Start ( opts ipn . Options ) error {
2020-02-03 18:35:52 +00:00
if opts . Prefs == nil && opts . StateKey == "" {
return errors . New ( "no state key or prefs provided" )
2020-02-03 23:58:40 +00:00
}
2020-02-03 18:35:52 +00:00
if opts . Prefs != nil {
2020-04-11 15:35:34 +00:00
b . logf ( "Start: %v" , opts . Prefs . Pretty ( ) )
2020-02-03 18:35:52 +00:00
} else {
2020-04-11 15:35:34 +00:00
b . logf ( "Start" )
2020-02-03 18:35:52 +00:00
}
2020-02-05 22:16:58 +00:00
2020-06-15 23:04:12 +00:00
hostinfo := controlclient . NewHostinfo ( )
hostinfo . BackendLogID = b . backendLogID
hostinfo . FrontendLogID = opts . FrontendLogID
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
2020-02-25 20:30:28 +00:00
2021-04-08 04:12:16 +00:00
if b . cc != nil {
2020-02-25 20:30:28 +00:00
// TODO(apenwarr): avoid the need to reinit controlclient.
// This will trigger a full relogin/reconfigure cycle every
// time a Handle reconnects to the backend. Ideally, we
// would send the new Prefs and everything would get back
// into sync with the minimal changes. But that's not how it
// is right now, which is a sign that the code is still too
// complicated.
2021-04-08 04:12:16 +00:00
b . cc . Shutdown ( )
2020-02-25 20:30:28 +00:00
}
2021-04-07 15:39:08 +00:00
httpTestClient := b . httpTestClient
2020-02-25 20:30:28 +00:00
2020-06-15 23:04:12 +00:00
if b . hostinfo != nil {
hostinfo . Services = b . hostinfo . Services // keep any previous session and netinfo
hostinfo . NetInfo = b . hostinfo . NetInfo
2020-02-25 18:04:20 +00:00
}
2020-06-15 23:04:12 +00:00
b . hostinfo = hostinfo
2021-02-04 21:12:42 +00:00
b . state = ipn . NoState
2020-02-03 18:35:52 +00:00
2021-04-07 16:33:14 +00:00
if err := b . loadStateLocked ( opts . StateKey , opts . Prefs ) ; err != nil {
2020-02-03 18:35:52 +00:00
b . mu . Unlock ( )
return fmt . Errorf ( "loading requested state: %v" , err )
2020-02-03 23:58:40 +00:00
}
2020-02-03 18:35:52 +00:00
2021-04-02 15:21:40 +00:00
wantRunning := b . prefs . WantRunning
if wantRunning {
if err := b . initMachineKeyLocked ( ) ; err != nil {
return fmt . Errorf ( "initMachineKeyLocked: %w" , err )
}
}
2020-11-03 05:11:20 +00:00
b . inServerMode = b . prefs . ForceDaemon
2020-02-19 05:03:22 +00:00
b . serverURL = b . prefs . ControlURL
2020-06-15 23:04:12 +00:00
hostinfo . RoutableIPs = append ( hostinfo . RoutableIPs , b . prefs . AdvertiseRoutes ... )
hostinfo . RequestTags = append ( hostinfo . RequestTags , b . prefs . AdvertiseTags ... )
2020-11-04 18:24:33 +00:00
if b . inServerMode || runtime . GOOS == "windows" {
b . logf ( "Start: serverMode=%v" , b . inServerMode )
}
2020-07-24 21:05:04 +00:00
applyPrefsToHostinfo ( hostinfo , b . prefs )
2020-02-18 03:33:01 +00:00
2020-10-27 19:51:48 +00:00
b . setNetMapLocked ( nil )
2021-02-05 23:23:01 +00:00
persistv := b . prefs . Persist
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
2020-07-29 01:47:23 +00:00
b . updateFilter ( nil , nil )
2020-02-05 22:16:58 +00:00
2020-10-14 21:07:40 +00:00
if b . portpoll != nil {
b . portpollOnce . Do ( func ( ) {
go b . portpoll . Run ( b . ctx )
go b . readPoller ( )
// Give the poller a second to get results to
// prevent it from restarting our map poll
// HTTP request (via doSetHostinfoFilterServices >
// cli.SetHostinfo). In practice this is very quick.
t0 := time . Now ( )
timer := time . NewTimer ( time . Second )
select {
case <- b . gotPortPollRes :
b . logf ( "got initial portlist info in %v" , time . Since ( t0 ) . Round ( time . Millisecond ) )
timer . Stop ( )
case <- timer . C :
b . logf ( "timeout waiting for initial portlist" )
}
} )
}
2020-06-20 17:18:13 +00:00
var discoPublic tailcfg . DiscoKey
2020-06-28 18:53:37 +00:00
if controlclient . Debug . Disco {
2020-07-06 19:10:39 +00:00
discoPublic = b . e . DiscoPublicKey ( )
2020-06-20 17:18:13 +00:00
}
2020-06-19 19:06:49 +00:00
2020-02-05 22:16:58 +00:00
var err error
2021-02-05 23:23:01 +00:00
if persistv == nil {
2020-02-05 22:16:58 +00:00
// let controlclient initialize it
2021-02-05 23:23:01 +00:00
persistv = & persist . Persist { }
2020-02-05 22:16:58 +00:00
}
2021-04-08 04:17:33 +00:00
cc , err := controlclient . New ( controlclient . Options {
2021-04-02 15:21:40 +00:00
GetMachinePrivateKey : b . createGetMachinePrivateKeyFunc ( ) ,
Logf : logger . WithPrefix ( b . logf , "control: " ) ,
Persist : * persistv ,
ServerURL : b . serverURL ,
AuthKey : opts . AuthKey ,
Hostinfo : hostinfo ,
KeepAlive : true ,
NewDecompressor : b . newDecompressor ,
2021-04-07 15:39:08 +00:00
HTTPTestClient : httpTestClient ,
2021-04-02 15:21:40 +00:00
DiscoPublicKey : discoPublic ,
DebugFlags : controlDebugFlags ,
LinkMonitor : b . e . GetLinkMonitor ( ) ,
2021-03-31 18:55:21 +00:00
// Don't warn about broken Linux IP forwading when
// netstack is being used.
2021-04-01 16:35:41 +00:00
SkipIPForwardingCheck : wgengine . IsNetstackRouter ( b . e ) ,
2020-02-05 22:16:58 +00:00
} )
if err != nil {
return err
}
b . mu . Lock ( )
2021-04-08 04:17:33 +00:00
b . cc = cc
2020-02-28 20:12:49 +00:00
endpoints := b . endpoints
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
2020-02-28 20:12:49 +00:00
if endpoints != nil {
2021-04-08 04:17:33 +00:00
cc . UpdateEndpoints ( 0 , endpoints )
2020-02-05 22:16:58 +00:00
}
2021-04-08 04:17:33 +00:00
cc . SetStatusFunc ( b . setClientStatus )
2020-06-15 23:04:12 +00:00
b . e . SetStatusCallback ( b . setWgengineStatus )
2020-05-19 02:32:20 +00:00
b . e . SetNetInfoCallback ( b . setNetInfo )
2020-03-24 05:16:16 +00:00
2020-02-29 02:34:56 +00:00
b . mu . Lock ( )
prefs := b . prefs . Clone ( )
b . mu . Unlock ( )
2020-02-05 22:16:58 +00:00
blid := b . backendLogID
2020-04-11 15:35:34 +00:00
b . logf ( "Backend: logs: be:%v fe:%v" , blid , opts . FrontendLogID )
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { BackendLogID : & blid } )
b . send ( ipn . Notify { Prefs : prefs } )
2020-02-05 22:16:58 +00:00
2021-04-02 15:21:40 +00:00
if wantRunning {
2021-04-08 04:17:33 +00:00
cc . Login ( nil , controlclient . LoginDefault )
2021-04-02 15:21:40 +00:00
}
2020-02-05 22:16:58 +00:00
return nil
}
2020-05-19 02:32:20 +00:00
// updateFilter updates the packet filter in wgengine based on the
// given netMap and user preferences.
2021-02-05 23:44:46 +00:00
func ( b * LocalBackend ) updateFilter ( netMap * netmap . NetworkMap , prefs * ipn . Prefs ) {
2020-07-31 20:03:00 +00:00
// NOTE(danderson): keep change detection as the first thing in
// this function. Don't try to optimize by returning early, more
// likely than not you'll just end up breaking the change
// detection and end up with the wrong filter installed. This is
// quite hard to debug, so save yourself the trouble.
var (
haveNetmap = netMap != nil
2020-12-24 20:33:55 +00:00
addrs [ ] netaddr . IPPrefix
2020-11-10 05:33:41 +00:00
packetFilter [ ] filter . Match
2021-02-22 22:34:15 +00:00
localNetsB netaddr . IPSetBuilder
2021-03-10 00:10:30 +00:00
logNetsB netaddr . IPSetBuilder
2020-07-31 20:03:00 +00:00
shieldsUp = prefs == nil || prefs . ShieldsUp // Be conservative when not ready
)
2021-03-10 00:10:30 +00:00
// Log traffic for Tailscale IPs.
logNetsB . AddPrefix ( tsaddr . CGNATRange ( ) )
logNetsB . AddPrefix ( tsaddr . TailscaleULARange ( ) )
logNetsB . RemovePrefix ( tsaddr . ChromeOSVMRange ( ) )
2020-07-31 20:03:00 +00:00
if haveNetmap {
addrs = netMap . Addresses
2021-02-22 22:34:15 +00:00
for _ , p := range addrs {
localNetsB . AddPrefix ( p )
}
2020-07-31 20:03:00 +00:00
packetFilter = netMap . PacketFilter
2020-05-22 02:41:18 +00:00
}
2020-07-29 01:47:23 +00:00
if prefs != nil {
2021-02-22 22:34:15 +00:00
for _ , r := range prefs . AdvertiseRoutes {
2021-02-23 04:43:35 +00:00
if r . Bits == 0 {
// When offering a default route to the world, we
// filter out locally reachable LANs, so that the
// default route effectively appears to be a "guest
// wifi": you get internet access, but to additionally
// get LAN access the LAN(s) need to be offered
// explicitly as well.
s , err := shrinkDefaultRoute ( r )
if err != nil {
b . logf ( "computing default route filter: %v" , err )
continue
}
localNetsB . AddSet ( s )
} else {
localNetsB . AddPrefix ( r )
2021-03-10 00:10:30 +00:00
// When advertising a non-default route, we assume
// this is a corporate subnet that should be present
// in the audit logs.
logNetsB . AddPrefix ( r )
2021-02-23 04:43:35 +00:00
}
2021-02-22 22:34:15 +00:00
}
2020-07-29 01:47:23 +00:00
}
2021-02-22 22:34:15 +00:00
localNets := localNetsB . IPSet ( )
2021-03-10 00:10:30 +00:00
logNets := logNetsB . IPSet ( )
2020-07-29 01:47:23 +00:00
2021-03-10 00:10:30 +00:00
changed := deepprint . UpdateHash ( & b . filterHash , haveNetmap , addrs , packetFilter , localNets . Ranges ( ) , logNets . Ranges ( ) , shieldsUp )
2020-07-29 01:47:23 +00:00
if ! changed {
return
}
2020-05-22 02:41:18 +00:00
2020-08-01 02:07:14 +00:00
if ! haveNetmap {
2020-07-31 20:03:00 +00:00
b . logf ( "netmap packet filter: (not ready yet)" )
2021-03-10 00:10:30 +00:00
b . e . SetFilter ( filter . NewAllowNone ( b . logf , logNets ) )
2020-08-01 02:07:14 +00:00
return
}
2021-01-22 21:39:53 +00:00
oldFilter := b . e . GetFilter ( )
2020-08-01 02:07:14 +00:00
if shieldsUp {
2020-04-29 06:37:35 +00:00
b . logf ( "netmap packet filter: (shields up)" )
2021-03-10 00:10:30 +00:00
b . e . SetFilter ( filter . NewShieldsUpFilter ( localNets , logNets , oldFilter , b . logf ) )
2020-08-01 02:07:14 +00:00
} else {
2020-07-31 20:03:00 +00:00
b . logf ( "netmap packet filter: %v" , packetFilter )
2021-03-10 00:10:30 +00:00
b . e . SetFilter ( filter . New ( packetFilter , localNets , logNets , oldFilter , b . logf ) )
2020-05-22 02:41:18 +00:00
}
2020-07-29 01:47:23 +00:00
}
2021-02-23 04:43:35 +00:00
var removeFromDefaultRoute = [ ] netaddr . IPPrefix {
// RFC1918 LAN ranges
netaddr . MustParseIPPrefix ( "192.168.0.0/16" ) ,
netaddr . MustParseIPPrefix ( "172.16.0.0/12" ) ,
netaddr . MustParseIPPrefix ( "10.0.0.0/8" ) ,
2021-03-18 00:04:32 +00:00
// IPv4 link-local
netaddr . MustParseIPPrefix ( "169.254.0.0/16" ) ,
// IPv4 multicast
netaddr . MustParseIPPrefix ( "224.0.0.0/4" ) ,
2021-02-23 04:43:35 +00:00
// Tailscale IPv4 range
tsaddr . CGNATRange ( ) ,
// IPv6 Link-local addresses
netaddr . MustParseIPPrefix ( "fe80::/10" ) ,
2021-03-18 00:04:32 +00:00
// IPv6 multicast
netaddr . MustParseIPPrefix ( "ff00::/8" ) ,
2021-02-23 04:43:35 +00:00
// Tailscale IPv6 range
tsaddr . TailscaleULARange ( ) ,
}
// shrinkDefaultRoute returns an IPSet representing the IPs in route,
// minus those in removeFromDefaultRoute and local interface subnets.
func shrinkDefaultRoute ( route netaddr . IPPrefix ) ( * netaddr . IPSet , error ) {
var b netaddr . IPSetBuilder
b . AddPrefix ( route )
2021-03-18 00:04:32 +00:00
var hostIPs [ ] netaddr . IP
2021-02-23 04:43:35 +00:00
err := interfaces . ForeachInterfaceAddress ( func ( _ interfaces . Interface , pfx netaddr . IPPrefix ) {
if tsaddr . IsTailscaleIP ( pfx . IP ) {
return
}
if pfx . IsSingleIP ( ) {
return
}
2021-03-18 00:04:32 +00:00
hostIPs = append ( hostIPs , pfx . IP )
2021-02-23 04:43:35 +00:00
b . RemovePrefix ( pfx )
} )
if err != nil {
return nil , err
}
2021-03-18 00:04:32 +00:00
// Having removed all the LAN subnets, re-add the hosts's own
// IPs. It's fine for clients to connect to an exit node's public
// IP address, just not the attached subnet.
//
// Truly forbidden subnets (in removeFromDefaultRoute) will still
// be stripped back out by the next step.
for _ , ip := range hostIPs {
if route . Contains ( ip ) {
b . Add ( ip )
}
}
2021-02-23 04:43:35 +00:00
for _ , pfx := range removeFromDefaultRoute {
b . RemovePrefix ( pfx )
}
return b . IPSet ( ) , nil
}
2020-07-29 01:47:23 +00:00
// dnsCIDRsEqual determines whether two CIDR lists are equal
// for DNS map construction purposes (that is, only the first entry counts).
2020-12-24 20:33:55 +00:00
func dnsCIDRsEqual ( newAddr , oldAddr [ ] netaddr . IPPrefix ) bool {
2020-07-29 01:47:23 +00:00
if len ( newAddr ) != len ( oldAddr ) {
return false
2020-02-05 22:16:58 +00:00
}
2020-07-29 01:47:23 +00:00
if len ( newAddr ) == 0 || newAddr [ 0 ] == oldAddr [ 0 ] {
return true
}
return false
}
// dnsMapsEqual determines whether the new and the old network map
// induce the same DNS map. It does so without allocating memory,
// at the expense of giving false negatives if peers are reordered.
2021-02-05 23:44:46 +00:00
func dnsMapsEqual ( new , old * netmap . NetworkMap ) bool {
2020-07-29 01:47:23 +00:00
if ( old == nil ) != ( new == nil ) {
return false
}
if old == nil && new == nil {
return true
}
if len ( new . Peers ) != len ( old . Peers ) {
return false
}
if new . Name != old . Name {
return false
}
if ! dnsCIDRsEqual ( new . Addresses , old . Addresses ) {
return false
}
for i , newPeer := range new . Peers {
oldPeer := old . Peers [ i ]
if newPeer . Name != oldPeer . Name {
return false
}
if ! dnsCIDRsEqual ( newPeer . Addresses , oldPeer . Addresses ) {
return false
}
}
return true
2020-02-05 22:16:58 +00:00
}
2020-05-19 02:32:20 +00:00
// readPoller is a goroutine that receives service lists from
// b.portpoll and propagates them into the controlclient's HostInfo.
2020-04-29 09:23:29 +00:00
func ( b * LocalBackend ) readPoller ( ) {
2020-10-14 21:07:40 +00:00
n := 0
2020-02-05 22:16:58 +00:00
for {
2020-03-14 03:53:58 +00:00
ports , ok := <- b . portpoll . C
if ! ok {
return
2020-02-05 22:16:58 +00:00
}
sl := [ ] tailcfg . Service { }
for _ , p := range ports {
s := tailcfg . Service {
2020-04-01 04:48:33 +00:00
Proto : tailcfg . ServiceProto ( p . Proto ) ,
2020-02-05 22:16:58 +00:00
Port : p . Port ,
Description : p . Process ,
}
2020-04-01 04:48:33 +00:00
if policy . IsInterestingService ( s , version . OS ( ) ) {
sl = append ( sl , s )
}
2020-02-05 22:16:58 +00:00
}
b . mu . Lock ( )
2020-06-15 23:04:12 +00:00
if b . hostinfo == nil {
b . hostinfo = new ( tailcfg . Hostinfo )
2020-02-25 19:01:20 +00:00
}
2020-06-15 23:04:12 +00:00
b . hostinfo . Services = sl
hi := b . hostinfo
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
2020-04-29 09:23:29 +00:00
b . doSetHostinfoFilterServices ( hi )
2020-10-14 21:07:40 +00:00
n ++
if n == 1 {
close ( b . gotPortPollRes )
}
2020-02-05 22:16:58 +00:00
}
}
2020-05-19 02:32:20 +00:00
// send delivers n to the connected frontend. If no frontend is
// connected, the notification is dropped without being delivered.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) send ( n ipn . Notify ) {
2020-02-25 20:30:28 +00:00
b . mu . Lock ( )
2021-03-30 18:19:42 +00:00
notifyFunc := b . notify
2021-04-09 14:57:32 +00:00
apiSrv := b . peerAPIServer
2020-02-25 20:30:28 +00:00
b . mu . Unlock ( )
2021-03-30 18:19:42 +00:00
if notifyFunc == nil {
return
}
2021-04-09 14:57:32 +00:00
if apiSrv != nil && apiSrv . hasFilesWaiting ( ) {
n . FilesWaiting = & empty . Message { }
}
2021-03-30 18:19:42 +00:00
n . Version = version . Long
2021-04-08 21:54:25 +00:00
notifyFunc ( n )
}
func ( b * LocalBackend ) sendFileNotify ( ) {
var n ipn . Notify
b . mu . Lock ( )
notifyFunc := b . notify
apiSrv := b . peerAPIServer
if notifyFunc == nil || apiSrv == nil {
b . mu . Unlock ( )
return
}
2021-04-09 14:57:32 +00:00
// Make sure we always set n.IncomingFiles non-nil so it gets encoded
// in JSON to clients. They distinguish between empty and non-nil
// to know whether a Notify should be able about files.
n . IncomingFiles = make ( [ ] ipn . PartialFile , 0 )
2021-04-08 21:54:25 +00:00
for f := range b . incomingFiles {
n . IncomingFiles = append ( n . IncomingFiles , f . PartialFile ( ) )
}
b . mu . Unlock ( )
sort . Slice ( n . IncomingFiles , func ( i , j int ) bool {
return n . IncomingFiles [ i ] . Started . Before ( n . IncomingFiles [ j ] . Started )
} )
b . send ( n )
2020-02-05 22:16:58 +00:00
}
2020-05-19 02:32:20 +00:00
// popBrowserAuthNow shuts down the data plane and sends an auth URL
// to the connected frontend, if any.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) popBrowserAuthNow ( ) {
b . mu . Lock ( )
url := b . authURL
2020-10-27 20:57:10 +00:00
b . interact = false
2020-02-05 22:16:58 +00:00
b . authURL = ""
b . mu . Unlock ( )
2020-02-25 22:05:17 +00:00
2020-04-11 15:35:34 +00:00
b . logf ( "popBrowserAuthNow: url=%v" , url != "" )
2020-02-05 22:16:58 +00:00
b . blockEngineUpdates ( true )
b . stopEngineAndWait ( )
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { BrowseToURL : & url } )
if b . State ( ) == ipn . Running {
b . enterState ( ipn . Starting )
2020-02-05 22:16:58 +00:00
}
}
2021-04-02 15:21:40 +00:00
// For testing lazy machine key generation.
var panicOnMachineKeyGeneration , _ = strconv . ParseBool ( os . Getenv ( "TS_DEBUG_PANIC_MACHINE_KEY" ) )
func ( b * LocalBackend ) createGetMachinePrivateKeyFunc ( ) func ( ) ( wgkey . Private , error ) {
var cache atomic . Value
return func ( ) ( wgkey . Private , error ) {
if panicOnMachineKeyGeneration {
panic ( "machine key generated" )
}
if v , ok := cache . Load ( ) . ( wgkey . Private ) ; ok {
return v , nil
}
b . mu . Lock ( )
defer b . mu . Unlock ( )
if v , ok := cache . Load ( ) . ( wgkey . Private ) ; ok {
return v , nil
}
if err := b . initMachineKeyLocked ( ) ; err != nil {
return wgkey . Private { } , err
}
cache . Store ( b . machinePrivKey )
return b . machinePrivKey , nil
}
}
2020-09-28 22:28:26 +00:00
// initMachineKeyLocked is called to initialize b.machinePrivKey.
//
// b.prefs must already be initialized.
2020-11-04 17:35:58 +00:00
// b.stateKey should be set too, but just for nicer log messages.
2020-09-28 22:28:26 +00:00
// b.mu must be held.
2020-10-21 19:55:03 +00:00
func ( b * LocalBackend ) initMachineKeyLocked ( ) ( err error ) {
2020-09-28 22:28:26 +00:00
if ! b . machinePrivKey . IsZero ( ) {
// Already set.
return nil
}
2020-12-30 01:22:56 +00:00
var legacyMachineKey wgkey . Private
2020-09-28 22:28:26 +00:00
if b . prefs . Persist != nil {
legacyMachineKey = b . prefs . Persist . LegacyFrontendPrivateMachineKey
}
2021-02-04 21:12:42 +00:00
keyText , err := b . store . ReadState ( ipn . MachineKeyStateKey )
2020-09-28 22:28:26 +00:00
if err == nil {
if err := b . machinePrivKey . UnmarshalText ( keyText ) ; err != nil {
2021-02-04 21:12:42 +00:00
return fmt . Errorf ( "invalid key in %s key of %v: %w" , ipn . MachineKeyStateKey , b . store , err )
2020-09-28 22:28:26 +00:00
}
if b . machinePrivKey . IsZero ( ) {
2021-02-04 21:12:42 +00:00
return fmt . Errorf ( "invalid zero key stored in %v key of %v" , ipn . MachineKeyStateKey , b . store )
2020-09-28 22:28:26 +00:00
}
if ! legacyMachineKey . IsZero ( ) && ! bytes . Equal ( legacyMachineKey [ : ] , b . machinePrivKey [ : ] ) {
b . logf ( "frontend-provided legacy machine key ignored; used value from server state" )
}
return nil
}
2021-02-04 21:12:42 +00:00
if err != ipn . ErrStateNotExist {
return fmt . Errorf ( "error reading %v key of %v: %w" , ipn . MachineKeyStateKey , b . store , err )
2020-09-28 22:28:26 +00:00
}
// If we didn't find one already on disk and the prefs already
// have a legacy machine key, use that. Otherwise generate a
// new one.
if ! legacyMachineKey . IsZero ( ) {
2020-11-04 17:35:58 +00:00
if b . stateKey == "" {
b . logf ( "using frontend-provided legacy machine key" )
} else {
b . logf ( "using legacy machine key from state key %q" , b . stateKey )
}
2020-09-28 22:28:26 +00:00
b . machinePrivKey = legacyMachineKey
} else {
b . logf ( "generating new machine key" )
var err error
2020-12-30 01:22:56 +00:00
b . machinePrivKey , err = wgkey . NewPrivate ( )
2020-09-28 22:28:26 +00:00
if err != nil {
return fmt . Errorf ( "initializing new machine key: %w" , err )
}
}
keyText , _ = b . machinePrivKey . MarshalText ( )
2021-02-04 21:12:42 +00:00
if err := b . store . WriteState ( ipn . MachineKeyStateKey , keyText ) ; err != nil {
2020-09-28 22:28:26 +00:00
b . logf ( "error writing machine key to store: %v" , err )
return err
}
b . logf ( "machine key written to store" )
return nil
}
2020-11-02 17:52:59 +00:00
// writeServerModeStartState stores the ServerModeStartKey value based on the current
// user and prefs. If userID is blank or prefs is blank, no work is done.
//
// b.mu may either be held or not.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) writeServerModeStartState ( userID string , prefs * ipn . Prefs ) {
2020-11-02 17:52:59 +00:00
if userID == "" || prefs == nil {
return
}
if prefs . ForceDaemon {
2021-02-04 21:12:42 +00:00
stateKey := ipn . StateKey ( "user-" + userID )
if err := b . store . WriteState ( ipn . ServerModeStartKey , [ ] byte ( stateKey ) ) ; err != nil {
2020-11-02 17:52:59 +00:00
b . logf ( "WriteState error: %v" , err )
}
// It's important we do this here too, even if it looks
// redundant with the one in the 'if stateKey != ""'
// check block above. That one won't fire in the case
// where the Windows client started up in client mode.
// This happens when we transition into server mode:
if err := b . store . WriteState ( stateKey , prefs . ToBytes ( ) ) ; err != nil {
b . logf ( "WriteState error: %v" , err )
}
} else {
2021-02-04 21:12:42 +00:00
if err := b . store . WriteState ( ipn . ServerModeStartKey , nil ) ; err != nil {
2020-11-02 17:52:59 +00:00
b . logf ( "WriteState error: %v" , err )
}
}
}
2020-05-19 02:32:20 +00:00
// loadStateLocked sets b.prefs and b.stateKey based on a complex
// combination of key, prefs, and legacyPath. b.mu must be held when
// calling.
2021-04-07 16:33:14 +00:00
func ( b * LocalBackend ) loadStateLocked ( key ipn . StateKey , prefs * ipn . Prefs ) ( err error ) {
2020-02-14 00:38:36 +00:00
if prefs == nil && key == "" {
panic ( "state key and prefs are both unset" )
}
2020-11-04 17:35:58 +00:00
// Optimistically set stateKey (for initMachineKeyLocked's
// logging), but revert it if we return an error so a later SetPrefs
// call can't pick it up if it's bogus.
b . stateKey = key
defer func ( ) {
if err != nil {
b . stateKey = ""
}
} ( )
2020-02-14 00:38:36 +00:00
if key == "" {
2020-09-28 22:28:26 +00:00
// Frontend owns the state, we just need to obey it.
//
// If the frontend (e.g. on Windows) supplied the
// optional/legacy machine key then it's used as the
// value instead of making up a new one.
2020-11-04 18:24:33 +00:00
b . logf ( "using frontend prefs: %s" , prefs . Pretty ( ) )
2020-02-27 20:20:29 +00:00
b . prefs = prefs . Clone ( )
2020-11-02 17:52:59 +00:00
b . writeServerModeStartState ( b . userID , b . prefs )
2020-02-14 00:38:36 +00:00
return nil
}
if prefs != nil {
// Backend owns the state, but frontend is trying to migrate
// state into the backend.
2020-11-04 18:24:33 +00:00
b . logf ( "importing frontend prefs into backend store; frontend prefs: %s" , prefs . Pretty ( ) )
2020-02-03 18:35:52 +00:00
if err := b . store . WriteState ( key , prefs . ToBytes ( ) ) ; err != nil {
return fmt . Errorf ( "store.WriteState: %v" , err )
}
}
2020-11-04 18:24:33 +00:00
b . logf ( "using backend prefs" )
2020-02-14 00:38:36 +00:00
bs , err := b . store . ReadState ( key )
2021-03-31 23:03:23 +00:00
switch {
case errors . Is ( err , ipn . ErrStateNotExist ) :
2021-04-07 16:33:14 +00:00
b . prefs = ipn . NewPrefs ( )
b . prefs . WantRunning = false
b . logf ( "created empty state for %q: %s" , key , b . prefs . Pretty ( ) )
2021-03-31 23:03:23 +00:00
return nil
case err != nil :
2020-02-14 00:38:36 +00:00
return fmt . Errorf ( "store.ReadState(%q): %v" , key , err )
}
2021-02-04 21:12:42 +00:00
b . prefs , err = ipn . PrefsFromBytes ( bs , false )
2020-02-14 00:38:36 +00:00
if err != nil {
return fmt . Errorf ( "PrefsFromBytes: %v" , err )
}
2020-11-04 18:24:33 +00:00
b . logf ( "backend prefs for %q: %s" , key , b . prefs . Pretty ( ) )
2020-02-03 18:35:52 +00:00
return nil
}
2020-05-19 02:32:20 +00:00
// State returns the backend state machine's current state.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) State ( ) ipn . State {
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
return b . state
}
2020-11-02 17:52:59 +00:00
func ( b * LocalBackend ) InServerMode ( ) bool {
ipn, ipnserver, cmd/tailscale: add "server mode" support on Windows
This partially (but not yet fully) migrates Windows to tailscaled's
StateStore storage system.
This adds a new bool Pref, ForceDaemon, defined as:
// ForceDaemon specifies whether a platform that normally
// operates in "client mode" (that is, requires an active user
// logged in with the GUI app running) should keep running after the
// GUI ends and/or the user logs out.
//
// The only current applicable platform is Windows. This
// forced Windows to go into "server mode" where Tailscale is
// running even with no users logged in. This might also be
// used for macOS in the future. This setting has no effect
// for Linux/etc, which always operate in daemon mode.
Then, when ForceDaemon becomes true, we now write use the StateStore
to track which user started it in server mode, and store their prefs
under that key.
The ipnserver validates the connections/identities and informs that
LocalBackend which userid is currently in charge.
The GUI can then enable/disable server mode at runtime, without using
the CLI.
But the "tailscale up" CLI was also fixed, so Windows users can use
authkeys or ACL tags, etc.
Updates #275
2020-10-12 21:28:21 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2020-11-02 17:52:59 +00:00
return b . inServerMode
ipn, ipnserver, cmd/tailscale: add "server mode" support on Windows
This partially (but not yet fully) migrates Windows to tailscaled's
StateStore storage system.
This adds a new bool Pref, ForceDaemon, defined as:
// ForceDaemon specifies whether a platform that normally
// operates in "client mode" (that is, requires an active user
// logged in with the GUI app running) should keep running after the
// GUI ends and/or the user logs out.
//
// The only current applicable platform is Windows. This
// forced Windows to go into "server mode" where Tailscale is
// running even with no users logged in. This might also be
// used for macOS in the future. This setting has no effect
// for Linux/etc, which always operate in daemon mode.
Then, when ForceDaemon becomes true, we now write use the StateStore
to track which user started it in server mode, and store their prefs
under that key.
The ipnserver validates the connections/identities and informs that
LocalBackend which userid is currently in charge.
The GUI can then enable/disable server mode at runtime, without using
the CLI.
But the "tailscale up" CLI was also fixed, so Windows users can use
authkeys or ACL tags, etc.
Updates #275
2020-10-12 21:28:21 +00:00
}
2020-05-19 02:32:20 +00:00
// getEngineStatus returns a copy of b.engineStatus.
2020-03-27 20:26:35 +00:00
//
2020-05-19 02:32:20 +00:00
// TODO(bradfitz): remove this and use Status() throughout.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) getEngineStatus ( ) ipn . EngineStatus {
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
return b . engineStatus
}
2020-07-13 20:13:11 +00:00
// Login implements Backend.
2021-03-19 17:21:33 +00:00
func ( b * LocalBackend ) Login ( token * tailcfg . Oauth2Token ) {
2020-07-13 20:13:11 +00:00
b . mu . Lock ( )
b . assertClientLocked ( )
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-07-13 20:13:11 +00:00
b . mu . Unlock ( )
2021-04-08 04:12:16 +00:00
cc . Login ( token , controlclient . LoginInteractive )
2020-07-13 20:13:11 +00:00
}
2020-05-19 02:32:20 +00:00
// StartLoginInteractive implements Backend. It requests a new
// interactive login from controlclient, unless such a flow is already
// in progress, in which case StartLoginInteractive attempts to pick
// up the in-progress flow where it left off.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) StartLoginInteractive ( ) {
b . mu . Lock ( )
2020-02-25 20:30:28 +00:00
b . assertClientLocked ( )
2020-10-27 20:57:10 +00:00
b . interact = true
2020-02-05 22:16:58 +00:00
url := b . authURL
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
2020-04-11 15:35:34 +00:00
b . logf ( "StartLoginInteractive: url=%v" , url != "" )
2020-02-05 22:16:58 +00:00
if url != "" {
b . popBrowserAuthNow ( )
} else {
2021-04-08 04:12:16 +00:00
cc . Login ( nil , controlclient . LoginInteractive )
2020-02-05 22:16:58 +00:00
}
}
2020-05-19 02:32:20 +00:00
// FakeExpireAfter implements Backend.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) FakeExpireAfter ( x time . Duration ) {
2020-04-11 15:35:34 +00:00
b . logf ( "FakeExpireAfter: %v" , x )
2020-06-15 23:04:12 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
if b . netMap == nil {
return
}
// This function is called very rarely,
// so we prefer to fully copy the netmap over introducing in-place modification here.
mapCopy := * b . netMap
e := mapCopy . Expiry
if e . IsZero ( ) || time . Until ( e ) > x {
mapCopy . Expiry = time . Now ( ) . Add ( x )
2020-02-05 22:16:58 +00:00
}
2020-10-27 19:51:48 +00:00
b . setNetMapLocked ( & mapCopy )
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { NetMap : b . netMap } )
2020-02-05 22:16:58 +00:00
}
2021-03-23 22:16:15 +00:00
func ( b * LocalBackend ) Ping ( ipStr string , useTSMP bool ) {
2020-08-09 21:49:42 +00:00
ip , err := netaddr . ParseIP ( ipStr )
if err != nil {
b . logf ( "ignoring Ping request to invalid IP %q" , ipStr )
return
}
2021-03-23 22:16:15 +00:00
b . e . Ping ( ip , useTSMP , func ( pr * ipnstate . PingResult ) {
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { PingResult : pr } )
2020-08-09 21:49:42 +00:00
} )
}
2020-10-29 22:26:10 +00:00
// parseWgStatusLocked returns an EngineStatus based on s.
//
// b.mu must be held; mostly because the caller is about to anyway, and doing so
// gives us slightly better guarantees about the two peers stats lines not
// being intermixed if there are concurrent calls to our caller.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) parseWgStatusLocked ( s * wgengine . Status ) ( ret ipn . EngineStatus ) {
2020-10-29 22:26:10 +00:00
var peerStats , peerKeys strings . Builder
2020-02-05 22:16:58 +00:00
2020-05-29 16:53:04 +00:00
ret . LiveDERPs = s . DERPs
2021-02-04 21:12:42 +00:00
ret . LivePeers = map [ tailcfg . NodeKey ] ipnstate . PeerStatusLite { }
2020-02-05 22:16:58 +00:00
for _ , p := range s . Peers {
2020-05-15 20:13:44 +00:00
if ! p . LastHandshake . IsZero ( ) {
2020-10-29 22:26:10 +00:00
fmt . Fprintf ( & peerStats , "%d/%d " , p . RxBytes , p . TxBytes )
fmt . Fprintf ( & peerKeys , "%s " , p . NodeKey . ShortString ( ) )
2020-05-19 02:32:20 +00:00
ret . NumLive ++
ret . LivePeers [ p . NodeKey ] = p
2020-05-15 20:13:44 +00:00
2020-02-05 22:16:58 +00:00
}
2020-05-19 02:32:20 +00:00
ret . RBytes += p . RxBytes
ret . WBytes += p . TxBytes
2020-05-12 16:14:37 +00:00
}
2020-10-29 22:26:10 +00:00
// [GRINDER STATS LINES] - please don't remove (used for log parsing)
if peerStats . Len ( ) > 0 {
2020-12-21 18:58:06 +00:00
b . keyLogf ( "[v1] peer keys: %s" , strings . TrimSpace ( peerKeys . String ( ) ) )
b . statsLogf ( "[v1] v%v peers: %v" , version . Long , strings . TrimSpace ( peerStats . String ( ) ) )
2020-02-05 22:16:58 +00:00
}
2020-05-19 02:32:20 +00:00
return ret
2020-02-05 22:16:58 +00:00
}
2021-01-11 22:24:32 +00:00
// shouldUploadServices reports whether this node should include services
// in Hostinfo. When the user preferences currently request "shields up"
// mode, all inbound connections are refused, so services are not reported.
// Otherwise, shouldUploadServices respects NetMap.CollectServices.
func ( b * LocalBackend ) shouldUploadServices ( ) bool {
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2021-01-11 22:24:32 +00:00
if b . prefs == nil || b . netMap == nil {
return false // default to safest setting
2020-05-19 02:32:20 +00:00
}
2021-01-11 22:24:32 +00:00
return ! b . prefs . ShieldsUp && b . netMap . CollectServices
2020-02-05 22:16:58 +00:00
}
ipn, ipnserver, cmd/tailscale: add "server mode" support on Windows
This partially (but not yet fully) migrates Windows to tailscaled's
StateStore storage system.
This adds a new bool Pref, ForceDaemon, defined as:
// ForceDaemon specifies whether a platform that normally
// operates in "client mode" (that is, requires an active user
// logged in with the GUI app running) should keep running after the
// GUI ends and/or the user logs out.
//
// The only current applicable platform is Windows. This
// forced Windows to go into "server mode" where Tailscale is
// running even with no users logged in. This might also be
// used for macOS in the future. This setting has no effect
// for Linux/etc, which always operate in daemon mode.
Then, when ForceDaemon becomes true, we now write use the StateStore
to track which user started it in server mode, and store their prefs
under that key.
The ipnserver validates the connections/identities and informs that
LocalBackend which userid is currently in charge.
The GUI can then enable/disable server mode at runtime, without using
the CLI.
But the "tailscale up" CLI was also fixed, so Windows users can use
authkeys or ACL tags, etc.
Updates #275
2020-10-12 21:28:21 +00:00
func ( b * LocalBackend ) SetCurrentUserID ( uid string ) {
b . mu . Lock ( )
b . userID = uid
b . mu . Unlock ( )
}
2021-04-11 23:10:31 +00:00
func ( b * LocalBackend ) EditPrefs ( mp * ipn . MaskedPrefs ) ( * ipn . Prefs , error ) {
2020-08-11 02:42:04 +00:00
b . mu . Lock ( )
2021-04-01 04:35:21 +00:00
p0 := b . prefs . Clone ( )
p1 := b . prefs . Clone ( )
p1 . ApplyEdits ( mp )
if p1 . Equals ( p0 ) {
b . mu . Unlock ( )
2021-04-11 23:10:31 +00:00
return p1 , nil
2020-08-11 02:42:04 +00:00
}
2021-04-11 04:56:18 +00:00
cc := b . cc
2021-04-01 04:35:21 +00:00
b . logf ( "EditPrefs: %v" , mp . Pretty ( ) )
2021-04-11 04:56:18 +00:00
b . setPrefsLockedOnEntry ( "EditPrefs" , p1 ) // does a b.mu.Unlock
if ! p0 . WantRunning && p1 . WantRunning {
b . logf ( "EditPrefs: transitioning to running; doing Login..." )
cc . Login ( nil , controlclient . LoginDefault )
}
2021-04-11 23:10:31 +00:00
return p1 , nil
2020-08-11 02:42:04 +00:00
}
2020-05-19 02:32:20 +00:00
// SetPrefs saves new user preferences and propagates them throughout
// the system. Implements Backend.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) SetPrefs ( newp * ipn . Prefs ) {
2020-10-27 19:33:37 +00:00
if newp == nil {
2020-02-20 19:07:00 +00:00
panic ( "SetPrefs got nil prefs" )
}
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
2021-04-01 04:35:21 +00:00
b . setPrefsLockedOnEntry ( "SetPrefs" , newp )
}
2020-07-29 01:47:23 +00:00
2021-04-01 04:35:21 +00:00
// setPrefsLockedOnEntry requires b.mu be held to call it, but it
// unlocks b.mu when done.
func ( b * LocalBackend ) setPrefsLockedOnEntry ( caller string , newp * ipn . Prefs ) {
2020-07-29 01:47:23 +00:00
netMap := b . netMap
stateKey := b . stateKey
2020-10-27 19:33:37 +00:00
oldp := b . prefs
newp . Persist = oldp . Persist // caller isn't allowed to override this
b . prefs = newp
2020-11-02 17:52:59 +00:00
b . inServerMode = newp . ForceDaemon
2020-07-29 01:47:23 +00:00
// We do this to avoid holding the lock while doing everything else.
2020-10-27 19:33:37 +00:00
newp = b . prefs . Clone ( )
2020-07-29 01:47:23 +00:00
2020-06-15 23:04:12 +00:00
oldHi := b . hostinfo
2020-02-27 20:20:29 +00:00
newHi := oldHi . Clone ( )
2020-12-24 20:33:55 +00:00
newHi . RoutableIPs = append ( [ ] netaddr . IPPrefix ( nil ) , b . prefs . AdvertiseRoutes ... )
2020-10-27 19:33:37 +00:00
applyPrefsToHostinfo ( newHi , newp )
2020-06-15 23:04:12 +00:00
b . hostinfo = newHi
2020-07-08 23:49:02 +00:00
hostInfoChanged := ! oldHi . Equal ( newHi )
ipn, ipnserver, cmd/tailscale: add "server mode" support on Windows
This partially (but not yet fully) migrates Windows to tailscaled's
StateStore storage system.
This adds a new bool Pref, ForceDaemon, defined as:
// ForceDaemon specifies whether a platform that normally
// operates in "client mode" (that is, requires an active user
// logged in with the GUI app running) should keep running after the
// GUI ends and/or the user logs out.
//
// The only current applicable platform is Windows. This
// forced Windows to go into "server mode" where Tailscale is
// running even with no users logged in. This might also be
// used for macOS in the future. This setting has no effect
// for Linux/etc, which always operate in daemon mode.
Then, when ForceDaemon becomes true, we now write use the StateStore
to track which user started it in server mode, and store their prefs
under that key.
The ipnserver validates the connections/identities and informs that
LocalBackend which userid is currently in charge.
The GUI can then enable/disable server mode at runtime, without using
the CLI.
But the "tailscale up" CLI was also fixed, so Windows users can use
authkeys or ACL tags, etc.
Updates #275
2020-10-12 21:28:21 +00:00
userID := b . userID
2020-07-29 01:47:23 +00:00
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
2020-07-29 01:47:23 +00:00
if stateKey != "" {
2020-10-27 19:33:37 +00:00
if err := b . store . WriteState ( stateKey , newp . ToBytes ( ) ) ; err != nil {
2021-04-01 04:35:21 +00:00
b . logf ( "failed to save new controlclient state: %v" , err )
2020-07-29 01:47:23 +00:00
}
}
2020-11-02 17:52:59 +00:00
b . writeServerModeStartState ( userID , newp )
2020-07-29 01:47:23 +00:00
2020-08-25 18:42:54 +00:00
// [GRINDER STATS LINE] - please don't remove (used for log parsing)
2021-04-01 04:35:21 +00:00
if caller == "SetPrefs" {
b . logf ( "SetPrefs: %v" , newp . Pretty ( ) )
}
2020-10-27 19:51:48 +00:00
if netMap != nil {
if login := netMap . UserProfiles [ netMap . User ] . LoginName ; login != "" {
if newp . Persist == nil {
b . logf ( "active login: %s" , login )
} else if newp . Persist . LoginName != login {
// Corp issue 461: sometimes the wrong prefs are
// logged; the frontend isn't always getting
// notified (to update its prefs/persist) on
// account switch. Log this while we figure it
// out.
b . logf ( "active login: %s ([unexpected] corp#461, not %s)" , newp . Persist . LoginName )
}
}
}
2020-04-10 15:42:34 +00:00
2020-10-27 19:33:37 +00:00
if oldp . ShieldsUp != newp . ShieldsUp || hostInfoChanged {
2020-04-29 09:23:29 +00:00
b . doSetHostinfoFilterServices ( newHi )
2020-02-18 03:33:01 +00:00
}
2020-10-27 19:33:37 +00:00
b . updateFilter ( netMap , newp )
2020-06-09 17:09:43 +00:00
2020-09-18 14:44:01 +00:00
if netMap != nil {
2020-07-29 01:47:23 +00:00
b . e . SetDERPMap ( netMap . DERPMap )
2020-06-09 17:09:43 +00:00
}
2020-04-29 06:37:35 +00:00
2020-10-27 19:33:37 +00:00
if oldp . WantRunning != newp . WantRunning {
2020-02-05 22:16:58 +00:00
b . stateMachine ( )
} else {
b . authReconfig ( )
}
2021-02-04 21:12:42 +00:00
b . send ( ipn . Notify { Prefs : newp } )
2020-02-05 22:16:58 +00:00
}
2021-03-29 22:17:05 +00:00
func ( b * LocalBackend ) getPeerAPIPortForTSMPPing ( ip netaddr . IP ) ( port uint16 , ok bool ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
for _ , pln := range b . peerAPIListeners {
2021-04-07 18:32:53 +00:00
if pln . ip == ip {
2021-04-02 05:04:46 +00:00
return uint16 ( pln . port ) , true
2021-03-29 22:17:05 +00:00
}
}
return 0 , false
}
2021-03-29 19:49:13 +00:00
func ( b * LocalBackend ) peerAPIServicesLocked ( ) ( ret [ ] tailcfg . Service ) {
for _ , pln := range b . peerAPIListeners {
proto := tailcfg . ServiceProto ( "peerapi4" )
if pln . ip . Is6 ( ) {
proto = "peerapi6"
}
ret = append ( ret , tailcfg . Service {
Proto : proto ,
2021-04-02 05:04:46 +00:00
Port : uint16 ( pln . port ) ,
2021-03-29 19:49:13 +00:00
} )
}
return ret
}
2020-05-19 02:32:20 +00:00
// doSetHostinfoFilterServices calls SetHostinfo on the controlclient,
// possibly after mangling the given hostinfo.
//
// TODO(danderson): we shouldn't be mangling hostinfo here after
// painstakingly constructing it in twelvety other places.
2020-04-29 09:23:29 +00:00
func ( b * LocalBackend ) doSetHostinfoFilterServices ( hi * tailcfg . Hostinfo ) {
2021-04-09 19:10:52 +00:00
if hi == nil {
b . logf ( "[unexpected] doSetHostinfoFilterServices with nil hostinfo" )
return
}
2020-04-29 09:23:29 +00:00
b . mu . Lock ( )
2021-04-08 04:12:16 +00:00
cc := b . cc
2021-03-29 19:49:13 +00:00
if cc == nil {
// Control client isn't up yet.
b . mu . Unlock ( )
return
}
peerAPIServices := b . peerAPIServicesLocked ( )
2020-04-29 09:23:29 +00:00
b . mu . Unlock ( )
2021-03-29 19:49:13 +00:00
// Make a shallow copy of hostinfo so we can mutate
// at the Service field.
hi2 := * hi // shallow copy
if ! b . shouldUploadServices ( ) {
hi2 . Services = [ ] tailcfg . Service { }
2020-04-29 09:23:29 +00:00
}
2021-03-29 19:49:13 +00:00
// Don't mutate hi.Service's underlying array. Append to
// the slice with no free capacity.
c := len ( hi2 . Services )
hi2 . Services = append ( hi2 . Services [ : c : c ] , peerAPIServices ... )
cc . SetHostinfo ( & hi2 )
2020-04-29 09:23:29 +00:00
}
2020-05-19 02:32:20 +00:00
// NetMap returns the latest cached network map received from
// controlclient, or nil if no network map was received yet.
2021-02-05 23:44:46 +00:00
func ( b * LocalBackend ) NetMap ( ) * netmap . NetworkMap {
2020-06-24 20:55:56 +00:00
b . mu . Lock ( )
defer b . mu . Unlock ( )
2020-06-15 23:04:12 +00:00
return b . netMap
2020-02-05 22:16:58 +00:00
}
2020-05-19 02:32:20 +00:00
// blockEngineUpdate sets b.blocked to block, while holding b.mu. Its
// indirect effect is to turn b.authReconfig() into a no-op if block
// is true.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) blockEngineUpdates ( block bool ) {
2020-04-11 15:35:34 +00:00
b . logf ( "blockEngineUpdates(%v)" , block )
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
b . blocked = block
b . mu . Unlock ( )
}
2020-05-19 02:32:20 +00:00
// authReconfig pushes a new configuration into wgengine, if engine
// updates are not currently blocked, based on the cached netmap and
// user prefs.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) authReconfig ( ) {
b . mu . Lock ( )
blocked := b . blocked
uc := b . prefs
2020-06-15 23:04:12 +00:00
nm := b . netMap
2020-10-05 22:12:35 +00:00
hasPAC := b . prevIfState . HasPAC ( )
2020-11-10 18:31:07 +00:00
disableSubnetsIfPAC := nm != nil && nm . Debug != nil && nm . Debug . DisableSubnetsIfPAC . EqualBool ( true )
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
if blocked {
2020-04-11 15:35:34 +00:00
b . logf ( "authReconfig: blocked, skipping." )
2020-02-05 22:16:58 +00:00
return
}
if nm == nil {
2020-04-11 15:35:34 +00:00
b . logf ( "authReconfig: netmap not yet valid. Skipping." )
2020-02-05 22:16:58 +00:00
return
}
if ! uc . WantRunning {
2020-04-11 15:35:34 +00:00
b . logf ( "authReconfig: skipping because !WantRunning." )
2020-02-05 22:16:58 +00:00
return
}
2021-02-05 23:44:46 +00:00
var flags netmap . WGConfigFlags
2020-02-05 22:16:58 +00:00
if uc . RouteAll {
2021-02-05 23:44:46 +00:00
flags |= netmap . AllowSubnetRoutes
2020-02-05 22:16:58 +00:00
}
if uc . AllowSingleHosts {
2021-02-05 23:44:46 +00:00
flags |= netmap . AllowSingleHosts
2020-02-05 22:16:58 +00:00
}
2020-11-10 18:31:07 +00:00
if hasPAC && disableSubnetsIfPAC {
2021-02-05 23:44:46 +00:00
if flags & netmap . AllowSubnetRoutes != 0 {
2020-10-05 22:12:35 +00:00
b . logf ( "authReconfig: have PAC; disabling subnet routes" )
2021-02-05 23:44:46 +00:00
flags &^= netmap . AllowSubnetRoutes
2020-10-05 22:12:35 +00:00
}
}
2020-02-05 22:16:58 +00:00
2021-02-25 04:05:23 +00:00
cfg , err := nmcfg . WGCfg ( nm , b . logf , flags , uc . ExitNodeID )
2020-04-10 14:52:30 +00:00
if err != nil {
2020-05-21 19:51:22 +00:00
b . logf ( "wgcfg: %v" , err )
return
2020-04-10 14:52:30 +00:00
}
2020-02-05 22:16:58 +00:00
2020-07-31 20:27:09 +00:00
rcfg := routerConfig ( cfg , uc )
2021-04-02 07:34:32 +00:00
var dcfg dns . Config
// If CorpDNS is false, dcfg remains the zero value.
2020-07-31 20:27:09 +00:00
if uc . CorpDNS {
2021-04-08 08:35:14 +00:00
for _ , resolver := range nm . DNS . Resolvers {
res , err := parseResolver ( resolver )
if err != nil {
b . logf ( err . Error ( ) )
continue
}
dcfg . DefaultResolvers = append ( dcfg . DefaultResolvers , res )
}
if len ( nm . DNS . Routes ) > 0 {
2021-04-09 22:24:47 +00:00
dcfg . Routes = map [ dnsname . FQDN ] [ ] netaddr . IPPort { }
2020-07-31 20:27:09 +00:00
}
2021-04-08 08:35:14 +00:00
for suffix , resolvers := range nm . DNS . Routes {
2021-04-09 22:24:47 +00:00
fqdn , err := dnsname . ToFQDN ( suffix )
if err != nil {
b . logf ( "[unexpected] non-FQDN route suffix %q" , suffix )
2021-04-08 08:35:14 +00:00
}
for _ , resolver := range resolvers {
res , err := parseResolver ( resolver )
if err != nil {
b . logf ( err . Error ( ) )
continue
}
2021-04-09 22:24:47 +00:00
dcfg . Routes [ fqdn ] = append ( dcfg . Routes [ fqdn ] , res )
}
}
for _ , dom := range nm . DNS . Domains {
fqdn , err := dnsname . ToFQDN ( dom )
if err != nil {
b . logf ( "[unexpected] non-FQDN search domain %q" , dom )
2021-04-08 08:35:14 +00:00
}
2021-04-09 22:24:47 +00:00
dcfg . SearchDomains = append ( dcfg . SearchDomains , fqdn )
2021-04-02 07:34:32 +00:00
}
dcfg . AuthoritativeSuffixes = magicDNSRootDomains ( nm )
set := func ( name string , addrs [ ] netaddr . IPPrefix ) {
if len ( addrs ) == 0 || name == "" {
return
}
2021-04-09 22:24:47 +00:00
fqdn , err := dnsname . ToFQDN ( name )
if err != nil {
return // TODO: propagate error?
}
2021-04-02 07:34:32 +00:00
var ips [ ] netaddr . IP
for _ , addr := range addrs {
ips = append ( ips , addr . IP )
}
2021-04-09 22:24:47 +00:00
dcfg . Hosts [ fqdn ] = ips
2021-04-01 08:33:58 +00:00
}
2021-04-08 23:23:33 +00:00
if nm . DNS . Proxied { // actually means "enable MagicDNS"
2021-04-09 22:24:47 +00:00
dcfg . Hosts = map [ dnsname . FQDN ] [ ] netaddr . IP { }
2021-04-02 21:22:46 +00:00
set ( nm . Name , nm . Addresses )
for _ , peer := range nm . Peers {
set ( peer . Name , peer . Addresses )
}
2021-04-01 08:33:58 +00:00
}
}
2021-04-02 07:34:32 +00:00
err = b . e . Reconfig ( cfg , rcfg , & dcfg )
2020-04-10 15:42:34 +00:00
if err == wgengine . ErrNoChanges {
return
2020-02-05 22:16:58 +00:00
}
2020-12-21 18:58:06 +00:00
b . logf ( "[v1] authReconfig: ra=%v dns=%v 0x%02x: %v" , uc . RouteAll , uc . CorpDNS , flags , err )
2021-03-25 22:38:40 +00:00
b . initPeerAPIListener ( )
}
2021-04-08 08:35:14 +00:00
func parseResolver ( cfg tailcfg . DNSResolver ) ( netaddr . IPPort , error ) {
ip , err := netaddr . ParseIP ( cfg . Addr )
if err != nil {
return netaddr . IPPort { } , fmt . Errorf ( "[unexpected] non-IP resolver %q" , cfg . Addr )
}
return netaddr . IPPort {
IP : ip ,
Port : 53 ,
} , nil
}
2021-03-31 21:08:32 +00:00
// tailscaleVarRoot returns the root directory of Tailscale's writable
// storage area. (e.g. "/var/lib/tailscale")
func tailscaleVarRoot ( ) string {
if runtime . GOOS == "ios" {
dir , _ := paths . IOSSharedDir . Load ( ) . ( string )
return dir
}
stateFile := paths . DefaultTailscaledStateFile ( )
if stateFile == "" {
return ""
}
return filepath . Dir ( stateFile )
}
2021-04-12 21:05:44 +00:00
func ( b * LocalBackend ) fileRootLocked ( uid tailcfg . UserID ) string {
if v := b . directFileRoot ; v != "" {
return v
}
varRoot := tailscaleVarRoot ( )
if varRoot == "" {
b . logf ( "peerapi disabled; no state directory" )
return ""
}
baseDir := fmt . Sprintf ( "%s-uid-%d" ,
strings . ReplaceAll ( b . activeLogin , "@" , "-" ) ,
uid )
dir := filepath . Join ( varRoot , "files" , baseDir )
if err := os . MkdirAll ( dir , 0700 ) ; err != nil {
b . logf ( "peerapi disabled; error making directory: %v" , err )
return ""
}
return dir
}
2021-03-25 22:38:40 +00:00
func ( b * LocalBackend ) initPeerAPIListener ( ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
2021-04-09 18:13:42 +00:00
if len ( b . netMap . Addresses ) == len ( b . peerAPIListeners ) {
allSame := true
for i , pln := range b . peerAPIListeners {
if pln . ip != b . netMap . Addresses [ i ] . IP {
allSame = false
break
}
}
if allSame {
// Nothing to do.
return
}
}
2021-03-30 18:19:42 +00:00
b . peerAPIServer = nil
2021-03-25 22:38:40 +00:00
for _ , pln := range b . peerAPIListeners {
2021-03-30 16:54:52 +00:00
pln . Close ( )
2021-03-25 22:38:40 +00:00
}
b . peerAPIListeners = nil
2021-03-29 17:42:33 +00:00
selfNode := b . netMap . SelfNode
if len ( b . netMap . Addresses ) == 0 || selfNode == nil {
return
}
2021-04-12 21:05:44 +00:00
fileRoot := b . fileRootLocked ( selfNode . User )
if fileRoot == "" {
2021-03-27 04:24:02 +00:00
return
}
2021-03-26 20:44:55 +00:00
var tunName string
if ge , ok := b . e . ( wgengine . InternalsGetter ) ; ok {
2021-03-29 22:17:05 +00:00
if tunWrap , _ , ok := ge . GetInternals ( ) ; ok {
tunName , _ = tunWrap . Name ( )
}
2021-03-26 20:44:55 +00:00
}
2021-03-29 17:42:33 +00:00
ps := & peerAPIServer {
2021-04-12 21:05:44 +00:00
b : b ,
rootDir : fileRoot ,
tunName : tunName ,
selfNode : selfNode ,
directFileMode : b . directFileRoot != "" ,
2021-03-29 17:42:33 +00:00
}
2021-03-30 18:19:42 +00:00
b . peerAPIServer = ps
2021-03-29 17:42:33 +00:00
2021-03-30 16:54:52 +00:00
isNetstack := wgengine . IsNetstack ( b . e )
for i , a := range b . netMap . Addresses {
var ln net . Listener
var err error
skipListen := i > 0 && isNetstack
if ! skipListen {
ln , err = ps . listen ( a . IP , b . prevIfState )
if err != nil {
b . logf ( "[unexpected] peerapi listen(%q) error: %v" , a . IP , err )
continue
}
2021-03-25 22:38:40 +00:00
}
pln := & peerAPIListener {
2021-03-29 17:42:33 +00:00
ps : ps ,
2021-03-29 19:49:13 +00:00
ip : a . IP ,
2021-03-30 16:54:52 +00:00
ln : ln , // nil for 2nd+ on netstack
2021-03-29 17:42:33 +00:00
lb : b ,
2021-03-25 22:38:40 +00:00
}
2021-03-30 16:54:52 +00:00
if skipListen {
2021-04-02 05:04:46 +00:00
pln . port = b . peerAPIListeners [ 0 ] . port
2021-03-30 16:54:52 +00:00
} else {
2021-04-02 05:04:46 +00:00
pln . port = ln . Addr ( ) . ( * net . TCPAddr ) . Port
2021-03-30 16:54:52 +00:00
}
2021-04-02 05:04:46 +00:00
pln . urlStr = "http://" + net . JoinHostPort ( a . IP . String ( ) , strconv . Itoa ( pln . port ) )
2021-03-30 16:54:52 +00:00
b . logf ( "peerapi: serving on %s" , pln . urlStr )
2021-03-25 22:38:40 +00:00
go pln . serve ( )
b . peerAPIListeners = append ( b . peerAPIListeners , pln )
}
2021-04-09 19:10:52 +00:00
2021-04-09 20:23:34 +00:00
go b . doSetHostinfoFilterServices ( b . hostinfo . Clone ( ) )
2020-02-05 22:16:58 +00:00
}
2021-01-05 18:37:15 +00:00
// magicDNSRootDomains returns the subset of nm.DNS.Domains that are the search domains for MagicDNS.
2021-04-09 22:24:47 +00:00
func magicDNSRootDomains ( nm * netmap . NetworkMap ) [ ] dnsname . FQDN {
2021-01-10 20:03:01 +00:00
if v := nm . MagicDNSSuffix ( ) ; v != "" {
2021-04-09 22:24:47 +00:00
fqdn , err := dnsname . ToFQDN ( v )
if err != nil {
// TODO: propagate error
return nil
}
return [ ] dnsname . FQDN { fqdn }
2020-07-31 20:27:09 +00:00
}
2021-01-10 20:03:01 +00:00
return nil
2020-07-31 20:27:09 +00:00
}
2021-01-21 01:24:16 +00:00
var (
ipv4Default = netaddr . MustParseIPPrefix ( "0.0.0.0/0" )
ipv6Default = netaddr . MustParseIPPrefix ( "::/0" )
)
2021-03-04 20:04:31 +00:00
// peerRoutes returns the routerConfig.Routes to access peers.
// If there are over cgnatThreshold CGNAT routes, one big CGNAT route
// is used instead.
func peerRoutes ( peers [ ] wgcfg . Peer , cgnatThreshold int ) ( routes [ ] netaddr . IPPrefix ) {
tsULA := tsaddr . TailscaleULARange ( )
cgNAT := tsaddr . CGNATRange ( )
var didULA bool
var cgNATIPs [ ] netaddr . IPPrefix
for _ , peer := range peers {
for _ , aip := range peer . AllowedIPs {
aip = unmapIPPrefix ( aip )
// Only add the Tailscale IPv6 ULA once, if we see anybody using part of it.
if aip . IP . Is6 ( ) && aip . IsSingleIP ( ) && tsULA . Contains ( aip . IP ) {
if ! didULA {
didULA = true
routes = append ( routes , tsULA )
}
continue
}
if aip . IsSingleIP ( ) && cgNAT . Contains ( aip . IP ) {
cgNATIPs = append ( cgNATIPs , aip )
} else {
routes = append ( routes , aip )
}
}
}
if len ( cgNATIPs ) > cgnatThreshold {
// Probably the hello server. Just append one big route.
routes = append ( routes , cgNAT )
} else {
routes = append ( routes , cgNATIPs ... )
}
return routes
}
2020-07-31 20:27:09 +00:00
// routerConfig produces a router.Config from a wireguard config and IPN prefs.
2021-02-04 21:12:42 +00:00
func routerConfig ( cfg * wgcfg . Config , prefs * ipn . Prefs ) * router . Config {
2020-05-12 07:08:52 +00:00
rs := & router . Config {
2020-12-24 20:33:55 +00:00
LocalAddrs : unmapIPPrefixes ( cfg . Addresses ) ,
SubnetRoutes : unmapIPPrefixes ( prefs . AdvertiseRoutes ) ,
2020-05-13 22:35:22 +00:00
SNATSubnetRoutes : ! prefs . NoSNAT ,
2020-05-15 02:07:06 +00:00
NetfilterMode : prefs . NetfilterMode ,
2021-03-04 20:04:31 +00:00
Routes : peerRoutes ( cfg . Peers , 10_000 ) ,
2020-05-11 21:02:12 +00:00
}
2021-01-21 01:24:16 +00:00
// Sanity check: we expect the control server to program both a v4
// and a v6 default route, if default routing is on. Fill in
// blackhole routes appropriately if we're missing some. This is
// likely to break some functionality, but if the user expressed a
// preference for routing remotely, we want to avoid leaking
// traffic at the expense of functionality.
if prefs . ExitNodeID != "" || ! prefs . ExitNodeIP . IsZero ( ) {
var default4 , default6 bool
for _ , route := range rs . Routes {
if route == ipv4Default {
default4 = true
} else if route == ipv6Default {
default6 = true
}
if default4 && default6 {
break
}
}
if ! default4 {
rs . Routes = append ( rs . Routes , ipv4Default )
}
if ! default6 {
rs . Routes = append ( rs . Routes , ipv6Default )
}
}
2020-06-08 22:19:26 +00:00
rs . Routes = append ( rs . Routes , netaddr . IPPrefix {
2020-07-31 20:27:09 +00:00
IP : tsaddr . TailscaleServiceIP ( ) ,
2020-06-08 22:19:26 +00:00
Bits : 32 ,
} )
2020-05-11 21:02:12 +00:00
return rs
}
2021-03-04 20:04:31 +00:00
func unmapIPPrefix ( ipp netaddr . IPPrefix ) netaddr . IPPrefix {
return netaddr . IPPrefix { IP : ipp . IP . Unmap ( ) , Bits : ipp . Bits }
}
2020-12-24 20:33:55 +00:00
func unmapIPPrefixes ( ippsList ... [ ] netaddr . IPPrefix ) ( ret [ ] netaddr . IPPrefix ) {
for _ , ipps := range ippsList {
for _ , ipp := range ipps {
2021-03-04 20:04:31 +00:00
ret = append ( ret , unmapIPPrefix ( ipp ) )
2020-05-11 21:02:12 +00:00
}
}
return ret
}
2021-02-04 21:12:42 +00:00
func applyPrefsToHostinfo ( hi * tailcfg . Hostinfo , prefs * ipn . Prefs ) {
2020-07-24 21:05:04 +00:00
if h := prefs . Hostname ; h != "" {
hi . Hostname = h
}
if v := prefs . OSVersion ; v != "" {
hi . OSVersion = v
}
if m := prefs . DeviceModel ; m != "" {
hi . DeviceModel = m
}
2020-11-24 15:51:13 +00:00
hi . ShieldsUp = prefs . ShieldsUp
2020-07-24 21:05:04 +00:00
}
2020-05-19 02:32:20 +00:00
// enterState transitions the backend into newState, updating internal
// state and propagating events out as needed.
//
// TODO(danderson): while this isn't a lie, exactly, a ton of other
// places twiddle IPN internal state without going through here, so
// really this is more "one of several places in which random things
// happen".
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) enterState ( newState ipn . State ) {
2020-02-05 22:16:58 +00:00
b . mu . Lock ( )
state := b . state
2020-06-19 17:43:55 +00:00
b . state = newState
2020-02-05 22:16:58 +00:00
prefs := b . prefs
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-10-06 22:22:46 +00:00
networkUp := b . prevIfState . AnyInterfaceUp ( )
2020-11-24 23:35:04 +00:00
activeLogin := b . activeLogin
authURL := b . authURL
2020-02-05 22:16:58 +00:00
b . mu . Unlock ( )
if state == newState {
return
}
2020-04-11 15:35:34 +00:00
b . logf ( "Switching ipn state %v -> %v (WantRunning=%v)" ,
2020-02-05 22:16:58 +00:00
state , newState , prefs . WantRunning )
2021-02-25 05:29:51 +00:00
health . SetIPNState ( newState . String ( ) , prefs . WantRunning )
2021-04-12 17:36:19 +00:00
b . send ( ipn . Notify { State : & newState } )
2020-02-05 22:16:58 +00:00
2021-04-08 04:12:16 +00:00
if cc != nil {
cc . SetPaused ( newState == ipn . Stopped || ! networkUp )
2020-07-31 16:39:45 +00:00
}
2020-02-05 22:16:58 +00:00
switch newState {
2021-02-04 21:12:42 +00:00
case ipn . NeedsLogin :
2020-11-24 23:35:04 +00:00
systemd . Status ( "Needs login: %s" , authURL )
2020-02-05 22:16:58 +00:00
b . blockEngineUpdates ( true )
fallthrough
2021-02-04 21:12:42 +00:00
case ipn . Stopped :
2021-04-02 07:34:32 +00:00
err := b . e . Reconfig ( & wgcfg . Config { } , & router . Config { } , & dns . Config { } )
2020-02-05 22:16:58 +00:00
if err != nil {
2020-04-11 15:35:34 +00:00
b . logf ( "Reconfig(down): %v" , err )
2020-02-05 22:16:58 +00:00
}
2020-11-24 23:35:04 +00:00
if authURL == "" {
systemd . Status ( "Stopped; run 'tailscale up' to log in" )
}
2021-02-04 21:12:42 +00:00
case ipn . Starting , ipn . NeedsMachineAuth :
2020-02-05 22:16:58 +00:00
b . authReconfig ( )
// Needed so that UpdateEndpoints can run
b . e . RequestStatus ( )
2021-02-04 21:12:42 +00:00
case ipn . Running :
2020-11-24 23:35:04 +00:00
var addrs [ ] string
for _ , addr := range b . netMap . Addresses {
addrs = append ( addrs , addr . IP . String ( ) )
}
systemd . Status ( "Connected; %s; %s" , activeLogin , strings . Join ( addrs , " " ) )
2020-02-05 22:16:58 +00:00
default :
2020-04-11 15:35:34 +00:00
b . logf ( "[unexpected] unknown newState %#v" , newState )
2020-02-05 22:16:58 +00:00
}
}
2020-05-19 02:32:20 +00:00
// nextState returns the state the backend seems to be in, based on
// its internal state.
2021-02-04 21:12:42 +00:00
func ( b * LocalBackend ) nextState ( ) ipn . State {
2020-02-28 19:39:13 +00:00
b . mu . Lock ( )
b . assertClientLocked ( )
2020-02-29 02:34:56 +00:00
var (
2021-04-08 04:12:16 +00:00
cc = b . cc
2020-06-15 23:04:12 +00:00
netMap = b . netMap
2020-02-29 02:34:56 +00:00
state = b . state
wantRunning = b . prefs . WantRunning
)
2020-02-28 19:39:13 +00:00
b . mu . Unlock ( )
2020-05-19 02:32:20 +00:00
switch {
case netMap == nil :
2021-04-08 04:12:16 +00:00
if cc . AuthCantContinue ( ) {
2020-02-05 22:16:58 +00:00
// Auth was interrupted or waiting for URL visit,
// so it won't proceed without human help.
2021-02-04 21:12:42 +00:00
return ipn . NeedsLogin
2020-02-05 22:16:58 +00:00
} else {
// Auth or map request needs to finish
return state
}
2020-05-19 02:32:20 +00:00
case ! wantRunning :
2021-02-04 21:12:42 +00:00
return ipn . Stopped
2020-05-19 02:32:20 +00:00
case ! netMap . Expiry . IsZero ( ) && time . Until ( netMap . Expiry ) <= 0 :
2021-02-04 21:12:42 +00:00
return ipn . NeedsLogin
2020-05-19 02:32:20 +00:00
case netMap . MachineStatus != tailcfg . MachineAuthorized :
2020-02-05 22:16:58 +00:00
// TODO(crawshaw): handle tailcfg.MachineInvalid
2021-02-04 21:12:42 +00:00
return ipn . NeedsMachineAuth
case state == ipn . NeedsMachineAuth :
2020-02-05 22:16:58 +00:00
// (if we get here, we know MachineAuthorized == true)
2021-02-04 21:12:42 +00:00
return ipn . Starting
case state == ipn . Starting :
2020-05-19 02:32:20 +00:00
if st := b . getEngineStatus ( ) ; st . NumLive > 0 || st . LiveDERPs > 0 {
2021-02-04 21:12:42 +00:00
return ipn . Running
2020-02-05 22:16:58 +00:00
} else {
return state
}
2021-02-04 21:12:42 +00:00
case state == ipn . Running :
return ipn . Running
2020-05-19 02:32:20 +00:00
default :
2021-02-04 21:12:42 +00:00
return ipn . Starting
2020-02-05 22:16:58 +00:00
}
}
2020-05-19 02:32:20 +00:00
// RequestEngineStatus implements Backend.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) RequestEngineStatus ( ) {
b . e . RequestStatus ( )
}
2020-05-19 02:32:20 +00:00
// stateMachine updates the state machine state based on other things
// that have happened. It is invoked from the various callbacks that
// feed events into LocalBackend.
//
2020-02-05 22:16:58 +00:00
// TODO(apenwarr): use a channel or something to prevent re-entrancy?
// Or maybe just call the state machine from fewer places.
func ( b * LocalBackend ) stateMachine ( ) {
b . enterState ( b . nextState ( ) )
}
2020-05-19 02:32:20 +00:00
// stopEngineAndWait deconfigures the local network data plane, and
// waits for it to deliver a status update before returning.
//
// TODO(danderson): this may be racy. We could unblock upon receiving
// a status update that predates the "I've shut down" update.
2020-02-05 22:16:58 +00:00
func ( b * LocalBackend ) stopEngineAndWait ( ) {
2020-04-11 15:35:34 +00:00
b . logf ( "stopEngineAndWait..." )
2021-04-02 07:34:32 +00:00
b . e . Reconfig ( & wgcfg . Config { } , & router . Config { } , & dns . Config { } )
2020-02-05 22:16:58 +00:00
b . requestEngineStatusAndWait ( )
2020-04-11 15:35:34 +00:00
b . logf ( "stopEngineAndWait: done." )
2020-02-05 22:16:58 +00:00
}
// Requests the wgengine status, and does not return until the status
// was delivered (to the usual callback).
func ( b * LocalBackend ) requestEngineStatusAndWait ( ) {
2020-04-11 15:35:34 +00:00
b . logf ( "requestEngineStatusAndWait" )
2020-02-05 22:16:58 +00:00
b . statusLock . Lock ( )
go b . e . RequestStatus ( )
2020-04-11 15:35:34 +00:00
b . logf ( "requestEngineStatusAndWait: waiting..." )
2020-02-05 22:16:58 +00:00
b . statusChanged . Wait ( ) // temporarily releases lock while waiting
2020-04-11 15:35:34 +00:00
b . logf ( "requestEngineStatusAndWait: got status update." )
2020-02-05 22:16:58 +00:00
b . statusLock . Unlock ( )
}
2021-04-08 04:17:33 +00:00
// Logout tells the controlclient that we want to log out, and
// transitions the local engine to the logged-out state without
// waiting for controlclient to be in that state.
2020-05-19 02:32:20 +00:00
//
2020-02-05 22:16:58 +00:00
// NOTE(apenwarr): No easy way to persist logged-out status.
// Maybe that's for the better; if someone logs out accidentally,
// rebooting will fix it.
func ( b * LocalBackend ) Logout ( ) {
2021-04-08 04:06:31 +00:00
b . logout ( context . Background ( ) , false )
}
func ( b * LocalBackend ) LogoutSync ( ctx context . Context ) error {
return b . logout ( ctx , true )
}
func ( b * LocalBackend ) logout ( ctx context . Context , sync bool ) error {
2020-02-28 19:39:13 +00:00
b . mu . Lock ( )
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-10-27 19:51:48 +00:00
b . setNetMapLocked ( nil )
2020-02-28 19:39:13 +00:00
b . mu . Unlock ( )
2021-04-08 04:06:31 +00:00
b . EditPrefs ( & ipn . MaskedPrefs {
WantRunningSet : true ,
Prefs : ipn . Prefs { WantRunning : true } ,
} )
2021-04-08 04:12:16 +00:00
if cc == nil {
2020-09-11 22:10:29 +00:00
// Double Logout can happen via repeated IPN
// connections to ipnserver making it repeatedly
// transition from 1->0 total connections, which on
// Windows by default ("client mode") causes a Logout
// on the transition to zero.
// Previously this crashed when we asserted that c was non-nil
// here.
2021-04-08 04:06:31 +00:00
return errors . New ( "no controlclient" )
2020-09-11 22:10:29 +00:00
}
2021-04-08 04:06:31 +00:00
var err error
if sync {
err = cc . Logout ( ctx )
} else {
cc . StartLogout ( )
}
2020-02-05 22:16:58 +00:00
2020-02-25 20:30:28 +00:00
b . mu . Lock ( )
2020-10-27 19:51:48 +00:00
b . setNetMapLocked ( nil )
2020-02-28 19:39:13 +00:00
b . mu . Unlock ( )
b . stateMachine ( )
2021-04-08 04:06:31 +00:00
return err
2020-02-25 20:30:28 +00:00
}
2020-05-19 02:32:20 +00:00
// assertClientLocked crashes if there is no controlclient in this backend.
2020-02-25 20:30:28 +00:00
func ( b * LocalBackend ) assertClientLocked ( ) {
2021-04-08 04:12:16 +00:00
if b . cc == nil {
panic ( "LocalBackend.assertClient: b.cc == nil" )
2020-02-05 22:16:58 +00:00
}
}
2020-02-25 22:05:17 +00:00
2020-06-15 23:04:12 +00:00
// setNetInfo sets b.hostinfo.NetInfo to ni, and passes ni along to the
2020-05-19 02:32:20 +00:00
// controlclient, if one exists.
func ( b * LocalBackend ) setNetInfo ( ni * tailcfg . NetInfo ) {
2020-03-04 06:21:56 +00:00
b . mu . Lock ( )
2021-04-08 04:12:16 +00:00
cc := b . cc
2020-06-15 23:04:12 +00:00
if b . hostinfo != nil {
b . hostinfo . NetInfo = ni . Clone ( )
2020-02-25 22:05:17 +00:00
}
2020-03-04 06:21:56 +00:00
b . mu . Unlock ( )
2020-02-25 22:05:17 +00:00
2021-04-08 04:12:16 +00:00
if cc == nil {
2020-03-04 06:21:56 +00:00
return
2020-02-25 22:05:17 +00:00
}
2021-04-08 04:12:16 +00:00
cc . SetNetInfo ( ni )
2020-02-25 22:05:17 +00:00
}
2020-05-27 19:23:17 +00:00
2021-02-05 23:44:46 +00:00
func ( b * LocalBackend ) setNetMapLocked ( nm * netmap . NetworkMap ) {
2020-10-27 19:51:48 +00:00
var login string
if nm != nil {
login = nm . UserProfiles [ nm . User ] . LoginName
if login == "" {
login = "<missing-profile>"
}
}
b . netMap = nm
if login != b . activeLogin {
b . logf ( "active login: %v" , login )
b . activeLogin = login
}
2021-01-28 23:29:17 +00:00
if nm == nil {
b . nodeByAddr = nil
return
}
// Update the nodeByAddr index.
if b . nodeByAddr == nil {
b . nodeByAddr = map [ netaddr . IP ] * tailcfg . Node { }
}
// First pass, mark everything unwanted.
for k := range b . nodeByAddr {
b . nodeByAddr [ k ] = nil
}
addNode := func ( n * tailcfg . Node ) {
for _ , ipp := range n . Addresses {
if ipp . IsSingleIP ( ) {
b . nodeByAddr [ ipp . IP ] = n
}
}
}
if nm . SelfNode != nil {
addNode ( nm . SelfNode )
}
for _ , p := range nm . Peers {
addNode ( p )
}
// Third pass, actually delete the unwanted items.
for k , v := range b . nodeByAddr {
if v == nil {
delete ( b . nodeByAddr , k )
}
}
2020-10-27 19:51:48 +00:00
}
2020-05-27 19:23:17 +00:00
// TestOnlyPublicKeys returns the current machine and node public
// keys. Used in tests only to facilitate automated node authorization
// in the test harness.
func ( b * LocalBackend ) TestOnlyPublicKeys ( ) ( machineKey tailcfg . MachineKey , nodeKey tailcfg . NodeKey ) {
b . mu . Lock ( )
prefs := b . prefs
2020-09-28 22:28:26 +00:00
machinePrivKey := b . machinePrivKey
2020-05-27 19:23:17 +00:00
b . mu . Unlock ( )
2020-09-28 22:28:26 +00:00
if prefs == nil || machinePrivKey . IsZero ( ) {
2020-05-27 19:23:17 +00:00
return
}
2020-09-28 22:28:26 +00:00
mk := machinePrivKey . Public ( )
2020-05-27 19:23:17 +00:00
nk := prefs . Persist . PrivateNodeKey . Public ( )
return tailcfg . MachineKey ( mk ) , tailcfg . NodeKey ( nk )
}
2020-10-21 19:55:03 +00:00
2021-03-30 19:56:00 +00:00
func ( b * LocalBackend ) WaitingFiles ( ) ( [ ] WaitingFile , error ) {
b . mu . Lock ( )
apiSrv := b . peerAPIServer
b . mu . Unlock ( )
if apiSrv == nil {
return nil , errors . New ( "peerapi disabled" )
}
return apiSrv . WaitingFiles ( )
}
func ( b * LocalBackend ) DeleteFile ( name string ) error {
b . mu . Lock ( )
apiSrv := b . peerAPIServer
b . mu . Unlock ( )
if apiSrv == nil {
return errors . New ( "peerapi disabled" )
}
return apiSrv . DeleteFile ( name )
}
func ( b * LocalBackend ) OpenFile ( name string ) ( rc io . ReadCloser , size int64 , err error ) {
b . mu . Lock ( )
apiSrv := b . peerAPIServer
b . mu . Unlock ( )
if apiSrv == nil {
return nil , 0 , errors . New ( "peerapi disabled" )
}
return apiSrv . OpenFile ( name )
}
2021-03-31 18:55:21 +00:00
2021-04-05 04:35:52 +00:00
// FileTarget is a node to which files can be sent, and the PeerAPI
// URL base to do so via.
type FileTarget struct {
Node * tailcfg . Node
// PeerAPI is the http://ip:port URL base of the node's peer API,
// without any path (not even a single slash).
PeerAPIURL string
}
// FileTargets lists nodes that the current node can send files to.
func ( b * LocalBackend ) FileTargets ( ) ( [ ] * FileTarget , error ) {
var ret [ ] * FileTarget
b . mu . Lock ( )
defer b . mu . Unlock ( )
nm := b . netMap
if b . state != ipn . Running || nm == nil {
return nil , errors . New ( "not connected" )
}
for _ , p := range nm . Peers {
2021-04-08 21:02:07 +00:00
if p . User != nm . User {
2021-04-05 04:35:52 +00:00
continue
}
peerAPI := peerAPIBase ( b . netMap , p )
if peerAPI == "" {
continue
2021-04-08 21:54:25 +00:00
2021-04-05 04:35:52 +00:00
}
ret = append ( ret , & FileTarget {
Node : p ,
PeerAPIURL : peerAPI ,
} )
}
// TODO: sort a different way than the netmap already is?
return ret , nil
}
2021-04-08 21:54:25 +00:00
func ( b * LocalBackend ) registerIncomingFile ( inf * incomingFile , active bool ) {
b . mu . Lock ( )
defer b . mu . Unlock ( )
if b . incomingFiles == nil {
b . incomingFiles = make ( map [ * incomingFile ] bool )
}
if active {
b . incomingFiles [ inf ] = true
} else {
delete ( b . incomingFiles , inf )
}
}
2021-04-05 04:35:52 +00:00
// peerAPIBase returns the "http://ip:port" URL base to reach peer's peerAPI.
// It returns the empty string if the peer doesn't support the peerapi
// or there's no matching address family based on the netmap's own addresses.
func peerAPIBase ( nm * netmap . NetworkMap , peer * tailcfg . Node ) string {
if nm == nil || peer == nil {
return ""
}
var have4 , have6 bool
for _ , a := range nm . Addresses {
if ! a . IsSingleIP ( ) {
continue
}
switch {
case a . IP . Is4 ( ) :
have4 = true
case a . IP . Is6 ( ) :
have6 = true
}
}
var p4 , p6 uint16
for _ , s := range peer . Hostinfo . Services {
switch s . Proto {
case "peerapi4" :
p4 = s . Port
case "peerapi6" :
p6 = s . Port
}
}
var ipp netaddr . IPPort
switch {
case have4 && p4 != 0 :
ipp = netaddr . IPPort { IP : nodeIP ( peer , netaddr . IP . Is4 ) , Port : p4 }
case have6 && p6 != 0 :
ipp = netaddr . IPPort { IP : nodeIP ( peer , netaddr . IP . Is6 ) , Port : p6 }
}
if ipp . IP . IsZero ( ) {
return ""
}
return fmt . Sprintf ( "http://%v" , ipp )
}
func nodeIP ( n * tailcfg . Node , pred func ( netaddr . IP ) bool ) netaddr . IP {
for _ , a := range n . Addresses {
if a . IsSingleIP ( ) && pred ( a . IP ) {
return a . IP
}
}
return netaddr . IP { }
}
2021-03-31 18:55:21 +00:00
func isBSD ( s string ) bool {
return s == "dragonfly" || s == "freebsd" || s == "netbsd" || s == "openbsd"
}
func ( b * LocalBackend ) CheckIPForwarding ( ) error {
2021-04-01 16:35:41 +00:00
if wgengine . IsNetstackRouter ( b . e ) {
2021-03-31 18:55:21 +00:00
return nil
}
if isBSD ( runtime . GOOS ) {
//lint:ignore ST1005 output to users as is
return fmt . Errorf ( "Subnet routing and exit nodes only work with additional manual configuration on %v, and is not currently officially supported." , runtime . GOOS )
}
var keys [ ] string
if runtime . GOOS == "linux" {
keys = append ( keys , "net.ipv4.ip_forward" , "net.ipv6.conf.all.forwarding" )
} else if isBSD ( runtime . GOOS ) {
keys = append ( keys , "net.inet.ip.forwarding" )
} else {
return nil
}
for _ , key := range keys {
bs , err := exec . Command ( "sysctl" , "-n" , key ) . Output ( )
if err != nil {
//lint:ignore ST1005 output to users as is
return fmt . Errorf ( "couldn't check %s (%v).\nSubnet routes won't work without IP forwarding." , key , err )
}
on , err := strconv . ParseBool ( string ( bytes . TrimSpace ( bs ) ) )
if err != nil {
//lint:ignore ST1005 output to users as is
return fmt . Errorf ( "couldn't parse %s (%v).\nSubnet routes won't work without IP forwarding." , key , err )
}
if ! on {
//lint:ignore ST1005 output to users as is
return fmt . Errorf ( "%s is disabled. Subnet routes won't work." , key )
}
}
return nil
}
2021-04-06 20:38:47 +00:00
// peerDialControlFunc is non-nil on platforms that require a way to
// bind to dial out to other peers.
var peerDialControlFunc func ( * LocalBackend ) func ( network , address string , c syscall . RawConn ) error
// PeerDialControlFunc returns a net.Dialer.Control func (possibly nil) to use to
// dial other Tailscale peers from the current environment.
func ( b * LocalBackend ) PeerDialControlFunc ( ) func ( network , address string , c syscall . RawConn ) error {
if peerDialControlFunc != nil {
return peerDialControlFunc ( b )
}
return nil
}