27d6b51071
Patch by Svante Signell. Reviewed-on: https://go-review.googlesource.com/c/160822 From-SVN: r268459
60 lines
1.5 KiB
Go
60 lines
1.5 KiB
Go
// Copyright 2016 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// +build dragonfly darwin freebsd hurd !android,linux netbsd openbsd
|
|
// +build cgo,!osusergo
|
|
|
|
package user
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
"syscall"
|
|
)
|
|
|
|
/*
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
*/
|
|
|
|
const maxGroups = 2048
|
|
|
|
func listGroups(u *User) ([]string, error) {
|
|
ug, err := strconv.Atoi(u.Gid)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("user: list groups for %s: invalid gid %q", u.Username, u.Gid)
|
|
}
|
|
userGID := syscall.Gid_t(ug)
|
|
nameC, err := syscall.BytePtrFromString(u.Username)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("user: invalid user name %q: %v", strconv.Quote(u.Username), err)
|
|
}
|
|
|
|
n := int32(256)
|
|
gidsC := make([]syscall.Gid_t, n)
|
|
syscall.Entersyscall()
|
|
rv := libc_getgrouplist(nameC, userGID, &gidsC[0], &n)
|
|
syscall.Exitsyscall()
|
|
if rv == -1 {
|
|
// More than initial buffer, but now n contains the correct size.
|
|
const maxGroups = 2048
|
|
if n > maxGroups {
|
|
return nil, fmt.Errorf("user: list groups for %s: member of more than %d groups", u.Username, maxGroups)
|
|
}
|
|
gidsC = make([]syscall.Gid_t, n)
|
|
syscall.Entersyscall()
|
|
rv := libc_getgrouplist(nameC, userGID, &gidsC[0], &n)
|
|
syscall.Exitsyscall()
|
|
if rv == -1 {
|
|
return nil, fmt.Errorf("user: list groups for %s failed (changed groups?)", u.Username)
|
|
}
|
|
}
|
|
gidsC = gidsC[:n]
|
|
gids := make([]string, 0, n)
|
|
for _, g := range gidsC[:n] {
|
|
gids = append(gids, strconv.Itoa(int(g)))
|
|
}
|
|
return gids, nil
|
|
}
|