OSDN Git Service

7060530154a2d3589207b4dfa3772e9e6262d8a5
[pf3gnuchains/gcc-fork.git] / libgo / go / os / user / lookup_unix.go
1 // Copyright 2011 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
4
5 package user
6
7 import (
8         "fmt"
9         "os"
10         "strings"
11         "syscall"
12         "unsafe"
13 )
14
15 /*
16 #include <unistd.h>
17 #include <sys/types.h>
18 #include <pwd.h>
19 #include <stdlib.h>
20
21 static int mygetpwuid_r(int uid, struct passwd *pwd,
22         char *buf, size_t buflen, struct passwd **result) {
23  return getpwuid_r(uid, pwd, buf, buflen, result);
24 }
25 */
26
27 func libc_getpwnam_r(name *byte, pwd *syscall.Passwd, buf *byte, buflen syscall.Size_t, result **syscall.Passwd) int __asm__ ("getpwnam_r")
28 func libc_getpwuid_r(uid syscall.Uid_t, pwd *syscall.Passwd, buf *byte, buflen syscall.Size_t, result **syscall.Passwd) int __asm__ ("getpwuid_r")
29
30 // Lookup looks up a user by username. If the user cannot be found,
31 // the returned error is of type UnknownUserError.
32 func Lookup(username string) (*User, os.Error) {
33         return lookup(-1, username, true)
34 }
35
36 // LookupId looks up a user by userid. If the user cannot be found,
37 // the returned error is of type UnknownUserIdError.
38 func LookupId(uid int) (*User, os.Error) {
39         return lookup(uid, "", false)
40 }
41
42 func lookup(uid int, username string, lookupByName bool) (*User, os.Error) {
43         var pwd syscall.Passwd
44         var result *syscall.Passwd
45
46         // FIXME: Should let buf grow if necessary.
47         const bufSize = 1024
48         buf := make([]byte, bufSize)
49         if lookupByName {
50                 rv := libc_getpwnam_r(syscall.StringBytePtr(username),
51                         &pwd,
52                         &buf[0],
53                         bufSize,
54                         &result)
55                 if rv != 0 {
56                         return nil, fmt.Errorf("user: lookup username %s: %s", username, os.Errno(syscall.GetErrno()))
57                 }
58                 if result == nil {
59                         return nil, UnknownUserError(username)
60                 }
61         } else {
62                 rv := libc_getpwuid_r(syscall.Uid_t(uid),
63                         &pwd,
64                         &buf[0],
65                         bufSize,
66                         &result)
67                 if rv != 0 {
68                         return nil, fmt.Errorf("user: lookup userid %d: %s", uid, os.Errno(syscall.GetErrno()))
69                 }
70                 if result == nil {
71                         return nil, UnknownUserIdError(uid)
72                 }
73         }
74         u := &User{
75                 Uid:      int(pwd.Pw_uid),
76                 Gid:      int(pwd.Pw_gid),
77                 Username: syscall.BytePtrToString((*byte)(unsafe.Pointer(pwd.Pw_name))),
78                 Name:     syscall.BytePtrToString((*byte)(unsafe.Pointer(pwd.Pw_gecos))),
79                 HomeDir:  syscall.BytePtrToString((*byte)(unsafe.Pointer(pwd.Pw_dir))),
80         }
81         // The pw_gecos field isn't quite standardized.  Some docs
82         // say: "It is expected to be a comma separated list of
83         // personal data where the first item is the full name of the
84         // user."
85         if i := strings.Index(u.Name, ","); i >= 0 {
86                 u.Name = u.Name[:i]
87         }
88         return u, nil
89 }