// 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 cgo,!osusergo package user /* #include <unistd.h> #include <sys/types.h> #include <stdlib.h> static int mygetgrouplist(const char* user, gid_t group, gid_t* groups, int* ngroups) { int* buf = malloc(*ngroups * sizeof(int)); int rv = getgrouplist(user, (int) group, buf, ngroups); int i; if (rv == 0) { for (i = 0; i < *ngroups; i++) { groups[i] = (gid_t) buf[i]; } } free(buf); return rv; } */ import "C" import ( "fmt" "unsafe" ) func getGroupList(name *C.char, userGID C.gid_t, gids *C.gid_t, n *C.int) C.int { return C.mygetgrouplist(name, userGID, gids, n) } // groupRetry retries getGroupList with an increasingly large size for n. The // result is stored in gids. func groupRetry(username string, name []byte, userGID C.gid_t, gids *[]C.gid_t, n *C.int) error { *n = C.int(256 * 2) for { *gids = make([]C.gid_t, *n) rv := getGroupList((*C.char)(unsafe.Pointer(&name[0])), userGID, &(*gids)[0], n) if rv >= 0 { // n is set correctly break } if *n > maxGroups { return fmt.Errorf("user: %q is a member of more than %d groups", username, maxGroups) } *n = *n * C.int(2) } return nil }