Skip to content

Commit

Permalink
addressing PR feedback
Browse files Browse the repository at this point in the history
  • Loading branch information
eriktate committed Aug 21, 2024
1 parent 9aa975b commit 977d91d
Show file tree
Hide file tree
Showing 3 changed files with 30 additions and 26 deletions.
2 changes: 1 addition & 1 deletion api/types/constants.go
Original file line number Diff line number Diff line change
Expand Up @@ -1289,7 +1289,7 @@ const (
// TeleportManagedGroup is a default group that users of the
// teleport automated user provisioning system get added to
// when provisioned in KEEP mode. This prevents already
// existing users from being tampered with ordeleted.
// existing users from being tampered with or deleted.
TeleportManagedGroup = "teleport-managed"
)

Expand Down
28 changes: 12 additions & 16 deletions lib/srv/usermgmt.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import (
"maps"
"os/user"
"regexp"
"slices"
"strings"
"syscall"
"time"
Expand Down Expand Up @@ -225,8 +226,8 @@ func (u *HostSudoersManagement) RemoveSudoers(name string) error {
return nil
}

// UnmanagedUserErr is returned when attempting to modify or interact with a user that is not managed by Teleport.
var UnmanagedUserErr = errors.New("user not managed by teleport")
// unmanagedUserErr is returned when attempting to modify or interact with a user that is not managed by Teleport.
var unmanagedUserErr = errors.New("user not managed by teleport")

func (u *HostUserManagement) updateUser(name string, ui services.HostUsersInfo) error {

Expand All @@ -250,23 +251,18 @@ func (u *HostUserManagement) updateUser(name string, ui services.HostUsersInfo)
currentGroups[group.Name] = struct{}{}
}

// allow for direct assignming of teleport-managed group even if the user is not currently managed
assigningManagedGroup := false
for _, group := range ui.Groups {
if group == types.TeleportManagedGroup {
assigningManagedGroup = true
break
}
}
// allow for explicit assignment of teleport-managed group in order to facilitate migrating KEEP users that existed before we added
// the teleport-managed group
migrateKeepUser := slices.Contains(ui.Groups, types.TeleportManagedGroup)

if assigningManagedGroup && ui.Mode == types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP {
return trace.Errorf("can not explicitly assign %q group in 'INSECURE_DROP' mode: %w", types.TeleportManagedGroup, UnmanagedUserErr)
if migrateKeepUser && ui.Mode == types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP {
return trace.Errorf("can not explicitly assign %q group in 'INSECURE_DROP' mode: %w", types.TeleportManagedGroup, unmanagedUserErr)
}

_, hasSystemGroup := currentGroups[types.TeleportServiceGroup]
_, hasManagedGroup := currentGroups[types.TeleportManagedGroup]
if !hasSystemGroup && !hasManagedGroup && !assigningManagedGroup {
return trace.Errorf("%q %w", name, UnmanagedUserErr)
if !hasSystemGroup && !hasManagedGroup && !migrateKeepUser {
return trace.Errorf("%q %w", name, unmanagedUserErr)
}

if ui.Mode == types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP {
Expand All @@ -286,7 +282,7 @@ func (u *HostUserManagement) updateUser(name string, ui services.HostUsersInfo)
}

// no need to duplicate the group if it's already there
if !assigningManagedGroup {
if !migrateKeepUser {
ui.Groups = append(ui.Groups, types.TeleportManagedGroup)
}
}
Expand Down Expand Up @@ -367,7 +363,7 @@ func (u *HostUserManagement) createUser(name string, ui services.HostUsersInfo)
}

func (u *HostUserManagement) ensureGroupsExist(groups ...string) error {
errs := make([]error, 0, len(groups))
var errs []error

for _, group := range groups {
if err := u.createGroupIfNotExist(group); err != nil {
Expand Down
26 changes: 17 additions & 9 deletions lib/srv/usermgmt_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -245,7 +245,7 @@ func TestUserMgmt_CreateTemporaryUser(t *testing.T) {

// an existing, unmanaged user should not be changed
closer, err = users.UpsertUser("simon", userinfo)
require.ErrorIs(t, err, UnmanagedUserErr)
require.ErrorIs(t, err, unmanagedUserErr)
require.Equal(t, nil, closer)
}

Expand Down Expand Up @@ -291,12 +291,12 @@ func TestUserMgmtSudoers_CreateTemporaryUser(t *testing.T) {
_, err := users.UpsertUser("testuser", services.HostUsersInfo{
Mode: types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP,
})
require.ErrorIs(t, err, UnmanagedUserErr)
require.ErrorIs(t, err, unmanagedUserErr)
backend.CreateGroup(types.TeleportServiceGroup, "")
_, err = users.UpsertUser("testuser", services.HostUsersInfo{
Mode: types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP,
})
require.ErrorIs(t, err, UnmanagedUserErr)
require.ErrorIs(t, err, unmanagedUserErr)
})
}

Expand Down Expand Up @@ -428,6 +428,7 @@ func Test_UpdateUserGroups_Keep(t *testing.T) {
assert.Equal(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportManagedGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportServiceGroup)

// Update user with new groups.
userinfo.Groups = slices.Clone(allGroups[2:])
Expand All @@ -437,13 +438,15 @@ func Test_UpdateUserGroups_Keep(t *testing.T) {
assert.Equal(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportManagedGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportServiceGroup)

// Upsert again with same groups should not call SetUserGroups.
closer, err = users.UpsertUser("alice", userinfo)
assert.NoError(t, err)
assert.Equal(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportManagedGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportServiceGroup)

// Updates with INSECURE_DROP mode should convert the managed user
userinfo.Mode = types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP
Expand All @@ -453,6 +456,7 @@ func Test_UpdateUserGroups_Keep(t *testing.T) {
assert.NotEqual(t, nil, closer)
assert.Equal(t, 2, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportServiceGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportManagedGroup)
}

func Test_UpdateUserGroups_Drop(t *testing.T) {
Expand Down Expand Up @@ -482,6 +486,7 @@ func Test_UpdateUserGroups_Drop(t *testing.T) {
assert.NotEqual(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportServiceGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportManagedGroup)

// Update user with new groups.
userinfo.Groups = slices.Clone(allGroups[2:])
Expand All @@ -491,13 +496,15 @@ func Test_UpdateUserGroups_Drop(t *testing.T) {
assert.NotEqual(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportServiceGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportManagedGroup)

// Upsert again with same groups should not call SetUserGroups.
closer, err = users.UpsertUser("alice", userinfo)
assert.NoError(t, err)
assert.NotEqual(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportServiceGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportManagedGroup)

// Updates with KEEP mode should convert the ephemeral user
userinfo.Mode = types.CreateHostUserMode_HOST_USER_MODE_KEEP
Expand All @@ -508,6 +515,7 @@ func Test_UpdateUserGroups_Drop(t *testing.T) {
assert.Equal(t, 2, backend.setUserGroupsCalls)
assert.Equal(t, 1, backend.createHomeDirectoryCalls)
assert.ElementsMatch(t, append(userinfo.Groups, types.TeleportManagedGroup), backend.users["alice"])
assert.NotContains(t, backend.users["alice"], types.TeleportServiceGroup)
}

func Test_DontManageExistingUser(t *testing.T) {
Expand Down Expand Up @@ -536,15 +544,15 @@ func Test_DontManageExistingUser(t *testing.T) {

// Update user in DROP mode
closer, err := users.UpsertUser("alice", userinfo)
assert.Error(t, err, UnmanagedUserErr)
assert.Error(t, err, unmanagedUserErr)
assert.Equal(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, allGroups, backend.users["alice"])

// Update user in KEEP mode
userinfo.Mode = types.CreateHostUserMode_HOST_USER_MODE_KEEP
closer, err = users.UpsertUser("alice", userinfo)
assert.ErrorIs(t, err, UnmanagedUserErr)
assert.ErrorIs(t, err, unmanagedUserErr)
assert.Equal(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, allGroups, backend.users["alice"])
Expand Down Expand Up @@ -575,7 +583,7 @@ func Test_DontUpdateUnmanagedUsers(t *testing.T) {

// Try to update existing, unmanaged user in KEEP mode
closer, err := users.UpsertUser("alice", userinfo)
assert.ErrorIs(t, err, UnmanagedUserErr)
assert.ErrorIs(t, err, unmanagedUserErr)
assert.Equal(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, allGroups[2:], backend.users["alice"])
Expand All @@ -587,7 +595,7 @@ func Test_DontUpdateUnmanagedUsers(t *testing.T) {

// Try to update existing, unmanaged user in DROP mode
closer, err = users.UpsertUser("alice", userinfo)
assert.ErrorIs(t, err, UnmanagedUserErr)
assert.ErrorIs(t, err, unmanagedUserErr)
assert.Equal(t, nil, closer)
assert.Zero(t, backend.setUserGroupsCalls)
assert.ElementsMatch(t, allGroups[2:], backend.users["alice"])
Expand Down Expand Up @@ -621,7 +629,7 @@ func Test_AllowExplicitlyManageExistingUsers(t *testing.T) {

// Take ownership of existing user when in KEEP mode
closer, err := users.UpsertUser("alice-keep", userinfo)
assert.NoError(t, err, UnmanagedUserErr)
assert.NoError(t, err)
assert.Equal(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
// slice off the end because teleport-system should be explicitly excluded
Expand All @@ -631,7 +639,7 @@ func Test_AllowExplicitlyManageExistingUsers(t *testing.T) {
// Don't take ownership of existing user when in DROP mode
userinfo.Mode = types.CreateHostUserMode_HOST_USER_MODE_INSECURE_DROP
closer, err = users.UpsertUser("alice-drop", userinfo)
assert.ErrorIs(t, err, UnmanagedUserErr)
assert.ErrorIs(t, err, unmanagedUserErr)
assert.Contains(t, err.Error(), "can not explicitly assign")
assert.Equal(t, nil, closer)
assert.Equal(t, 1, backend.setUserGroupsCalls)
Expand Down

0 comments on commit 977d91d

Please sign in to comment.