Skip to content

Commit c22a746

Browse files
authored
Rewrite chuser() for simplicity and correctness (#1203)
- Use unambiguous variable names (w/o package name conflict). - Fail on invalid input such as the empty string or `:`. - Do not change group without user, i.e. fail on `:group`. - Parse input using mnemonic APIs. - Do not juggle between integer types. - Unset supplementary groups. - Use set[ug]id(2) to follow the idiom of OpenBSD base programs. (cannot use setres[ug]id(2) as macOS does not have them.) Includes/Supersedes #1202. Fixes #927. I only tested on OpenBSD (so far), but other systems should just work.
1 parent 67ec5a9 commit c22a746

File tree

2 files changed

+111
-67
lines changed

2 files changed

+111
-67
lines changed

cmd/yggdrasil/chuser_unix.go

Lines changed: 31 additions & 67 deletions
Original file line numberDiff line numberDiff line change
@@ -4,89 +4,53 @@
44
package main
55

66
import (
7-
"errors"
87
"fmt"
9-
"math"
10-
osuser "os/user"
8+
"os/user"
119
"strconv"
1210
"strings"
13-
"syscall"
11+
12+
"golang.org/x/sys/unix"
1413
)
1514

16-
func chuser(user string) error {
17-
group := ""
18-
if i := strings.IndexByte(user, ':'); i >= 0 {
19-
user, group = user[:i], user[i+1:]
20-
}
15+
func chuser(input string) error {
16+
givenUser, givenGroup, _ := strings.Cut(input, ":")
2117

22-
u := (*osuser.User)(nil)
23-
g := (*osuser.Group)(nil)
18+
var (
19+
err error
20+
usr *user.User
21+
grp *user.Group
22+
uid, gid int
23+
)
2424

25-
if user != "" {
26-
if _, err := strconv.ParseUint(user, 10, 32); err == nil {
27-
u, err = osuser.LookupId(user)
28-
if err != nil {
29-
return fmt.Errorf("failed to lookup user by id %q: %v", user, err)
30-
}
31-
} else {
32-
u, err = osuser.Lookup(user)
33-
if err != nil {
34-
return fmt.Errorf("failed to lookup user by name %q: %v", user, err)
35-
}
25+
if usr, err = user.LookupId(givenUser); err != nil {
26+
if usr, err = user.Lookup(givenUser); err != nil {
27+
return err
3628
}
3729
}
38-
if group != "" {
39-
if _, err := strconv.ParseUint(group, 10, 32); err == nil {
40-
g, err = osuser.LookupGroupId(group)
41-
if err != nil {
42-
return fmt.Errorf("failed to lookup group by id %q: %v", user, err)
43-
}
44-
} else {
45-
g, err = osuser.LookupGroup(group)
46-
if err != nil {
47-
return fmt.Errorf("failed to lookup group by name %q: %v", user, err)
48-
}
49-
}
30+
if uid, err = strconv.Atoi(usr.Uid); err != nil {
31+
return err
5032
}
5133

52-
if g != nil {
53-
gid, _ := strconv.ParseUint(g.Gid, 10, 32)
54-
var err error
55-
if gid < math.MaxInt {
56-
if err := syscall.Setgroups([]int{int(gid)}); err != nil {
57-
return fmt.Errorf("failed to setgroups %d: %v", gid, err)
34+
if givenGroup != "" {
35+
if grp, err = user.LookupGroupId(givenGroup); err != nil {
36+
if grp, err = user.LookupGroup(givenGroup); err != nil {
37+
return err
5838
}
59-
err = syscall.Setgid(int(gid))
60-
} else {
61-
err = errors.New("gid too big")
6239
}
6340

64-
if err != nil {
65-
return fmt.Errorf("failed to setgid %d: %v", gid, err)
66-
}
67-
} else if u != nil {
68-
gid, _ := strconv.ParseUint(u.Gid, 10, 32)
69-
if err := syscall.Setgroups([]int{int(uint32(gid))}); err != nil {
70-
return fmt.Errorf("failed to setgroups %d: %v", gid, err)
71-
}
72-
err := syscall.Setgid(int(uint32(gid)))
73-
if err != nil {
74-
return fmt.Errorf("failed to setgid %d: %v", gid, err)
75-
}
41+
gid, _ = strconv.Atoi(grp.Gid)
42+
} else {
43+
gid, _ = strconv.Atoi(usr.Gid)
7644
}
7745

78-
if u != nil {
79-
uid, _ := strconv.ParseUint(u.Uid, 10, 32)
80-
var err error
81-
if uid < math.MaxInt {
82-
err = syscall.Setuid(int(uid))
83-
} else {
84-
err = errors.New("uid too big")
85-
}
86-
87-
if err != nil {
88-
return fmt.Errorf("failed to setuid %d: %v", uid, err)
89-
}
46+
if err := unix.Setgroups([]int{gid}); err != nil {
47+
return fmt.Errorf("setgroups: %d: %v", gid, err)
48+
}
49+
if err := unix.Setgid(gid); err != nil {
50+
return fmt.Errorf("setgid: %d: %v", gid, err)
51+
}
52+
if err := unix.Setuid(uid); err != nil {
53+
return fmt.Errorf("setuid: %d: %v", uid, err)
9054
}
9155

9256
return nil

cmd/yggdrasil/chuser_unix_test.go

Lines changed: 80 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
//go:build aix || darwin || dragonfly || freebsd || linux || netbsd || openbsd || solaris
2+
// +build aix darwin dragonfly freebsd linux netbsd openbsd solaris
3+
4+
package main
5+
6+
import (
7+
"testing"
8+
"os/user"
9+
)
10+
11+
// Usernames must not contain a number sign.
12+
func TestEmptyString (t *testing.T) {
13+
if chuser("") == nil {
14+
t.Fatal("the empty string is not a valid user")
15+
}
16+
}
17+
18+
// Either omit delimiter and group, or omit both.
19+
func TestEmptyGroup (t *testing.T) {
20+
if chuser("0:") == nil {
21+
t.Fatal("the empty group is not allowed")
22+
}
23+
}
24+
25+
// Either user only or user and group.
26+
func TestGroupOnly (t *testing.T) {
27+
if chuser(":0") == nil {
28+
t.Fatal("group only is not allowed")
29+
}
30+
}
31+
32+
// Usenames must not contain the number sign.
33+
func TestInvalidUsername (t *testing.T) {
34+
const username = "#user"
35+
if chuser(username) == nil {
36+
t.Fatalf("'%s' is not a valid username", username)
37+
}
38+
}
39+
40+
// User IDs must be non-negative.
41+
func TestInvalidUserid (t *testing.T) {
42+
if chuser("-1") == nil {
43+
t.Fatal("User ID cannot be negative")
44+
}
45+
}
46+
47+
// Change to the current user by ID.
48+
func TestCurrentUserid (t *testing.T) {
49+
usr, err := user.Current()
50+
if err != nil {
51+
t.Fatal(err)
52+
}
53+
54+
if usr.Uid != "0" {
55+
t.Skip("setgroups(2): Only the superuser may set new groups.")
56+
}
57+
58+
if err = chuser(usr.Uid); err != nil {
59+
t.Fatal(err)
60+
}
61+
}
62+
63+
// Change to a common user by name.
64+
func TestCommonUsername (t *testing.T) {
65+
usr, err := user.Current()
66+
if err != nil {
67+
t.Fatal(err)
68+
}
69+
70+
if usr.Uid != "0" {
71+
t.Skip("setgroups(2): Only the superuser may set new groups.")
72+
}
73+
74+
if err := chuser("nobody"); err != nil {
75+
if _, ok := err.(user.UnknownUserError); ok {
76+
t.Skip(err)
77+
}
78+
t.Fatal(err)
79+
}
80+
}

0 commit comments

Comments
 (0)