mirror of
https://github.com/tailscale/tailscale.git
synced 2024-11-29 13:05:46 +00:00
ssh/tailssh: evaluate tailcfg.SSHPolicy on incoming connections
Updates #3802 Fixes #3960 Change-Id: Ieda2007d462ddce6c217b958167417ae9755774e Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
parent
66f5aa6814
commit
e1e20f6d39
@ -10,12 +10,14 @@
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/creack/pty"
|
"github.com/creack/pty"
|
||||||
@ -24,6 +26,7 @@
|
|||||||
"tailscale.com/envknob"
|
"tailscale.com/envknob"
|
||||||
"tailscale.com/ipn/ipnlocal"
|
"tailscale.com/ipn/ipnlocal"
|
||||||
"tailscale.com/net/tsaddr"
|
"tailscale.com/net/tsaddr"
|
||||||
|
"tailscale.com/tailcfg"
|
||||||
"tailscale.com/types/logger"
|
"tailscale.com/types/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -66,6 +69,33 @@ type server struct {
|
|||||||
logf logger.Logf
|
logf logger.Logf
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var debugPolicyFile = envknob.String("TS_DEBUG_SSH_POLICY_FILE")
|
||||||
|
|
||||||
|
func (srv *server) sshPolicy() (_ *tailcfg.SSHPolicy, ok bool) {
|
||||||
|
lb := srv.lb
|
||||||
|
nm := lb.NetMap()
|
||||||
|
if nm == nil {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
if pol := nm.SSHPolicy; pol != nil {
|
||||||
|
return pol, true
|
||||||
|
}
|
||||||
|
if debugPolicyFile != "" {
|
||||||
|
f, err := os.ReadFile(debugPolicyFile)
|
||||||
|
if err != nil {
|
||||||
|
srv.logf("error reading debug SSH policy file: %v", err)
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
p := new(tailcfg.SSHPolicy)
|
||||||
|
if err := json.Unmarshal(f, p); err != nil {
|
||||||
|
srv.logf("invalid JSON in %v: %v", debugPolicyFile, err)
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
return p, true
|
||||||
|
}
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
func (srv *server) handleSSH(s ssh.Session) {
|
func (srv *server) handleSSH(s ssh.Session) {
|
||||||
lb := srv.lb
|
lb := srv.lb
|
||||||
logf := srv.logf
|
logf := srv.logf
|
||||||
@ -91,35 +121,54 @@ func (srv *server) handleSSH(s ssh.Session) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ptyReq, winCh, isPty := s.Pty()
|
pol, ok := srv.sshPolicy()
|
||||||
if !isPty {
|
|
||||||
fmt.Fprintf(s, "TODO scp etc")
|
|
||||||
s.Exit(1)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
srcIPP := netaddr.IPPortFrom(tanetaddr, uint16(ta.Port))
|
|
||||||
node, uprof, ok := lb.WhoIs(srcIPP)
|
|
||||||
if !ok {
|
if !ok {
|
||||||
fmt.Fprintf(s, "Hello, %v. I don't know who you are.\n", srcIPP)
|
logf("tsshd: rejecting connection; no SSH policy")
|
||||||
s.Exit(0)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
allow := envknob.String("TS_SSH_ALLOW_LOGIN")
|
|
||||||
if allow == "" || uprof.LoginName != allow {
|
|
||||||
logf("ssh: access denied for %q (only allowing %q)", uprof.LoginName, allow)
|
|
||||||
jnode, _ := json.Marshal(node)
|
|
||||||
jprof, _ := json.Marshal(uprof)
|
|
||||||
fmt.Fprintf(s, "Access denied.\n\nYou are node: %s\n\nYour profile: %s\n\nYou wanted %+v\n", jnode, jprof, ptyReq)
|
|
||||||
s.Exit(1)
|
s.Exit(1)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ptyReq, winCh, isPty := s.Pty()
|
||||||
|
srcIPP := netaddr.IPPortFrom(tanetaddr, uint16(ta.Port))
|
||||||
|
node, uprof, ok := lb.WhoIs(srcIPP)
|
||||||
|
if !ok {
|
||||||
|
fmt.Fprintf(s, "Hello, %v. I don't know who you are.\n", srcIPP)
|
||||||
|
s.Exit(1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
srcIP := srcIPP.IP()
|
||||||
|
sctx := &sshContext{
|
||||||
|
now: time.Now(),
|
||||||
|
sshUser: s.User(),
|
||||||
|
srcIP: srcIP,
|
||||||
|
node: node,
|
||||||
|
uprof: &uprof,
|
||||||
|
}
|
||||||
|
action, localUser, ok := evalSSHPolicy(pol, sctx)
|
||||||
|
if ok && action.Message != "" {
|
||||||
|
io.WriteString(s, action.Message)
|
||||||
|
}
|
||||||
|
if !ok || action.Reject {
|
||||||
|
logf("ssh: access denied for %q from %v", uprof.LoginName, srcIP)
|
||||||
|
s.Exit(1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !action.Accept || action.HoldAndDelegate != "" {
|
||||||
|
fmt.Fprintf(s, "TODO: other SSHAction outcomes")
|
||||||
|
s.Exit(1)
|
||||||
|
|
||||||
|
}
|
||||||
|
if !isPty {
|
||||||
|
fmt.Fprintf(s, "TODO scp etc\n")
|
||||||
|
s.Exit(1)
|
||||||
|
return
|
||||||
|
}
|
||||||
var cmd *exec.Cmd
|
var cmd *exec.Cmd
|
||||||
sshUser := s.User()
|
if os.Getuid() != 0 || localUser == "root" {
|
||||||
if os.Getuid() != 0 || sshUser == "root" {
|
|
||||||
cmd = exec.Command("/bin/bash")
|
cmd = exec.Command("/bin/bash")
|
||||||
} else {
|
} else {
|
||||||
cmd = exec.Command("/usr/bin/env", "su", "-", sshUser)
|
cmd = exec.Command("/usr/bin/env", "su", "-", localUser)
|
||||||
}
|
}
|
||||||
cmd.Env = append(cmd.Env, fmt.Sprintf("TERM=%s", ptyReq.Term))
|
cmd.Env = append(cmd.Env, fmt.Sprintf("TERM=%s", ptyReq.Term))
|
||||||
f, err := pty.Start(cmd)
|
f, err := pty.Start(cmd)
|
||||||
@ -128,6 +177,16 @@ func (srv *server) handleSSH(s ssh.Session) {
|
|||||||
s.Exit(1)
|
s.Exit(1)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if action.SesssionDuration != 0 {
|
||||||
|
t := time.AfterFunc(action.SesssionDuration, func() {
|
||||||
|
logf("terminating SSH session from %v after max duration", srcIP)
|
||||||
|
cmd.Process.Kill()
|
||||||
|
f.Close()
|
||||||
|
})
|
||||||
|
defer t.Stop()
|
||||||
|
}
|
||||||
|
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
go func() {
|
go func() {
|
||||||
for win := range winCh {
|
for win := range winCh {
|
||||||
@ -150,3 +209,91 @@ func setWinsize(f *os.File, w, h int) {
|
|||||||
syscall.Syscall(syscall.SYS_IOCTL, f.Fd(), uintptr(syscall.TIOCSWINSZ),
|
syscall.Syscall(syscall.SYS_IOCTL, f.Fd(), uintptr(syscall.TIOCSWINSZ),
|
||||||
uintptr(unsafe.Pointer(&struct{ h, w, x, y uint16 }{uint16(h), uint16(w), 0, 0})))
|
uintptr(unsafe.Pointer(&struct{ h, w, x, y uint16 }{uint16(h), uint16(w), 0, 0})))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type sshContext struct {
|
||||||
|
// now is the time to consider the present moment for the
|
||||||
|
// purposes of rule evaluation.
|
||||||
|
now time.Time
|
||||||
|
|
||||||
|
// sshUser is the requested local SSH username ("root", "alice", etc).
|
||||||
|
sshUser string
|
||||||
|
|
||||||
|
// srcIP is the Tailscale IP that the connection came from.
|
||||||
|
srcIP netaddr.IP
|
||||||
|
|
||||||
|
// node is srcIP's node.
|
||||||
|
node *tailcfg.Node
|
||||||
|
|
||||||
|
// uprof is node's UserProfile.
|
||||||
|
uprof *tailcfg.UserProfile
|
||||||
|
}
|
||||||
|
|
||||||
|
func evalSSHPolicy(pol *tailcfg.SSHPolicy, sctx *sshContext) (a *tailcfg.SSHAction, localUser string, ok bool) {
|
||||||
|
for _, r := range pol.Rules {
|
||||||
|
if a, localUser, err := matchRule(r, sctx); err == nil {
|
||||||
|
return a, localUser, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, "", false
|
||||||
|
}
|
||||||
|
|
||||||
|
// internal errors for testing; they don't escape to callers or logs.
|
||||||
|
var (
|
||||||
|
errNilRule = errors.New("nil rule")
|
||||||
|
errNilAction = errors.New("nil action")
|
||||||
|
errRuleExpired = errors.New("rule expired")
|
||||||
|
errPrincipalMatch = errors.New("principal didn't match")
|
||||||
|
errUserMatch = errors.New("user didn't match")
|
||||||
|
)
|
||||||
|
|
||||||
|
func matchRule(r *tailcfg.SSHRule, sctx *sshContext) (a *tailcfg.SSHAction, localUser string, err error) {
|
||||||
|
if r == nil {
|
||||||
|
return nil, "", errNilRule
|
||||||
|
}
|
||||||
|
if r.Action == nil {
|
||||||
|
return nil, "", errNilAction
|
||||||
|
}
|
||||||
|
if r.RuleExpires != nil && sctx.now.After(*r.RuleExpires) {
|
||||||
|
return nil, "", errRuleExpired
|
||||||
|
}
|
||||||
|
if !matchesPrincipal(r.Principals, sctx) {
|
||||||
|
return nil, "", errPrincipalMatch
|
||||||
|
}
|
||||||
|
if !r.Action.Reject || r.SSHUsers != nil {
|
||||||
|
localUser = mapLocalUser(r.SSHUsers, sctx.sshUser)
|
||||||
|
if localUser == "" {
|
||||||
|
return nil, "", errUserMatch
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r.Action, localUser, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func mapLocalUser(ruleSSHUsers map[string]string, reqSSHUser string) (localUser string) {
|
||||||
|
if v, ok := ruleSSHUsers[reqSSHUser]; ok {
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
return ruleSSHUsers["*"]
|
||||||
|
}
|
||||||
|
|
||||||
|
func matchesPrincipal(ps []*tailcfg.SSHPrincipal, sctx *sshContext) bool {
|
||||||
|
for _, p := range ps {
|
||||||
|
if p == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if p.Any {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if !p.Node.IsZero() && sctx.node != nil && p.Node == sctx.node.StableID {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if p.NodeIP != "" {
|
||||||
|
if ip, _ := netaddr.ParseIP(p.NodeIP); ip == sctx.srcIP {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if p.UserLogin != "" && sctx.uprof != nil && sctx.uprof.LoginName == p.UserLogin {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
157
ssh/tailssh/tailssh_test.go
Normal file
157
ssh/tailssh/tailssh_test.go
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
// Copyright (c) 2022 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.
|
||||||
|
|
||||||
|
//go:build linux
|
||||||
|
// +build linux
|
||||||
|
|
||||||
|
package tailssh
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"inet.af/netaddr"
|
||||||
|
"tailscale.com/tailcfg"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMatchRule(t *testing.T) {
|
||||||
|
someAction := new(tailcfg.SSHAction)
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
rule *tailcfg.SSHRule
|
||||||
|
ctx *sshContext
|
||||||
|
wantErr error
|
||||||
|
wantUser string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "nil-rule",
|
||||||
|
rule: nil,
|
||||||
|
wantErr: errNilRule,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "nil-action",
|
||||||
|
rule: &tailcfg.SSHRule{},
|
||||||
|
wantErr: errNilAction,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "expired",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
RuleExpires: timePtr(time.Unix(100, 0)),
|
||||||
|
},
|
||||||
|
ctx: &sshContext{now: time.Unix(200, 0)},
|
||||||
|
wantErr: errRuleExpired,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "no-principal",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
},
|
||||||
|
wantErr: errPrincipalMatch,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "no-user-match",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{Any: true}},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{sshUser: "alice"},
|
||||||
|
wantErr: errUserMatch,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok-wildcard",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{Any: true}},
|
||||||
|
SSHUsers: map[string]string{
|
||||||
|
"*": "ubuntu",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{sshUser: "alice"},
|
||||||
|
wantUser: "ubuntu",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok-wildcard-and-nil-principal",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{
|
||||||
|
nil, // don't crash on this
|
||||||
|
{Any: true},
|
||||||
|
},
|
||||||
|
SSHUsers: map[string]string{
|
||||||
|
"*": "ubuntu",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{sshUser: "alice"},
|
||||||
|
wantUser: "ubuntu",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok-exact",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{Any: true}},
|
||||||
|
SSHUsers: map[string]string{
|
||||||
|
"*": "ubuntu",
|
||||||
|
"alice": "thealice",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{sshUser: "alice"},
|
||||||
|
wantUser: "thealice",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "no-users-for-reject",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{Any: true}},
|
||||||
|
Action: &tailcfg.SSHAction{Reject: true},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{sshUser: "alice"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "match-principal-node-ip",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{NodeIP: "1.2.3.4"}},
|
||||||
|
SSHUsers: map[string]string{"*": "ubuntu"},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{srcIP: netaddr.MustParseIP("1.2.3.4")},
|
||||||
|
wantUser: "ubuntu",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "match-principal-node-id",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{Node: "some-node-ID"}},
|
||||||
|
SSHUsers: map[string]string{"*": "ubuntu"},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{node: &tailcfg.Node{StableID: "some-node-ID"}},
|
||||||
|
wantUser: "ubuntu",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "match-principal-userlogin",
|
||||||
|
rule: &tailcfg.SSHRule{
|
||||||
|
Action: someAction,
|
||||||
|
Principals: []*tailcfg.SSHPrincipal{{UserLogin: "foo@bar.com"}},
|
||||||
|
SSHUsers: map[string]string{"*": "ubuntu"},
|
||||||
|
},
|
||||||
|
ctx: &sshContext{uprof: &tailcfg.UserProfile{LoginName: "foo@bar.com"}},
|
||||||
|
wantUser: "ubuntu",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
got, gotUser, err := matchRule(tt.rule, tt.ctx)
|
||||||
|
if err != tt.wantErr {
|
||||||
|
t.Errorf("err = %v; want %v", err, tt.wantErr)
|
||||||
|
}
|
||||||
|
if gotUser != tt.wantUser {
|
||||||
|
t.Errorf("user = %q; want %q", gotUser, tt.wantUser)
|
||||||
|
}
|
||||||
|
if err == nil && got == nil {
|
||||||
|
t.Errorf("expected non-nil action on success")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func timePtr(t time.Time) *time.Time { return &t }
|
@ -51,7 +51,8 @@
|
|||||||
// 24: 2021-09-18: MapResponse.Health from control to node; node shows in "tailscale status"
|
// 24: 2021-09-18: MapResponse.Health from control to node; node shows in "tailscale status"
|
||||||
// 25: 2021-11-01: MapResponse.Debug.Exit
|
// 25: 2021-11-01: MapResponse.Debug.Exit
|
||||||
// 26: 2022-01-12: (nothing, just bumping for 1.20.0)
|
// 26: 2022-01-12: (nothing, just bumping for 1.20.0)
|
||||||
const CurrentMapRequestVersion = 26
|
// 27: 2022-02-18: start of SSHPolicy being respected
|
||||||
|
const CurrentMapRequestVersion = 27
|
||||||
|
|
||||||
type StableID string
|
type StableID string
|
||||||
|
|
||||||
@ -1545,6 +1546,9 @@ type SSHRule struct {
|
|||||||
// contain a key for either ssh-user or, as a fallback, "*" to
|
// contain a key for either ssh-user or, as a fallback, "*" to
|
||||||
// match anything. If it does, the map entry's value is the
|
// match anything. If it does, the map entry's value is the
|
||||||
// actual user that's logged in.
|
// actual user that's logged in.
|
||||||
|
// If the map value is the empty string (for either the
|
||||||
|
// requested SSH user or "*"), the rule doesn't match.
|
||||||
|
// It may be nil if the Action is reject.
|
||||||
SSHUsers map[string]string `json:"sshUsers"`
|
SSHUsers map[string]string `json:"sshUsers"`
|
||||||
|
|
||||||
// Action is the outcome to task.
|
// Action is the outcome to task.
|
||||||
@ -1553,12 +1557,15 @@ type SSHRule struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// SSHPrincipal is either a particular node or a user on any node.
|
// SSHPrincipal is either a particular node or a user on any node.
|
||||||
// At most one field should be non-zero specified.
|
// Any matching field causes a match.
|
||||||
type SSHPrincipal struct {
|
type SSHPrincipal struct {
|
||||||
Node StableNodeID `json:"node,omitempty"`
|
Node StableNodeID `json:"node,omitempty"`
|
||||||
NodeIP string `json:"nodeIP,omitempty"`
|
NodeIP string `json:"nodeIP,omitempty"`
|
||||||
UserLogin string `json:"userLogin,omitempty"` // email-ish: foo@example.com, bar@github
|
UserLogin string `json:"userLogin,omitempty"` // email-ish: foo@example.com, bar@github
|
||||||
|
|
||||||
|
// Any, if true, matches any user.
|
||||||
|
Any bool `json:"any,omitempty"`
|
||||||
|
|
||||||
// TODO(bradfitz): add StableUserID, once that exists
|
// TODO(bradfitz): add StableUserID, once that exists
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1579,9 +1586,9 @@ type SSHAction struct {
|
|||||||
// without further prompts.
|
// without further prompts.
|
||||||
Accept bool `json:"accept,omitempty"`
|
Accept bool `json:"accept,omitempty"`
|
||||||
|
|
||||||
// SesssionExpires, if non-nil, is the time at which this
|
// SesssionDuration, if non-zero, is how long the session can stay open
|
||||||
// session should forcefully terminate.
|
// before being forcefully terminated.
|
||||||
SesssionExpires *time.Time `json:"sessionExpires,omitempty"`
|
SesssionDuration time.Duration `json:"sessionDuration,omitempty"`
|
||||||
|
|
||||||
// HoldAndDelegate, if non-empty, is a URL that serves an outcome verdict.
|
// HoldAndDelegate, if non-empty, is a URL that serves an outcome verdict.
|
||||||
// The connection will be accepted and will block until the
|
// The connection will be accepted and will block until the
|
||||||
|
Loading…
Reference in New Issue
Block a user