我正在尝试进行网络日志记录,并使用 getpwnam()功能检查用户名是否存在。但是对于有效的用户名 getpwnam 会返回错误:没有这样的文件或目录。所以我尝试了 getpwnam_r(),但它也因同样的错误而失败了。我在嵌入式arm linux上运行,我使用 / etc / passwd 进行密码存储(我没有 / etc / shadow )。我的测试计划是:
#include <pwd.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
int
main(int argc, char *argv[])
{
struct passwd pwd;
struct passwd *result;
char *buf;
size_t bufsize;
int s;
if (argc != 2) {
fprintf(stderr, "Usage: %s username\n", argv[0]);
exit(EXIT_FAILURE);
}
bufsize = sysconf(_SC_GETPW_R_SIZE_MAX);
if (bufsize == -1) /* Value was indeterminate */
bufsize = 16384; /* Should be more than enough */
buf = malloc(bufsize);
if (buf == NULL) {
perror("malloc");
exit(EXIT_FAILURE);
}
s = getpwnam_r(argv[1], &pwd, buf, bufsize, &result);
if (result == NULL) {
if (s == 0)
printf("Not found\n");
else {
errno = s;
perror("getpwnam_r");
}
exit(EXIT_FAILURE);
}
printf("Name: %s; UID: %ld\n", pwd.pw_gecos, (long) pwd.pw_uid);
exit(EXIT_SUCCESS);
}
密码文件只能由root编写:
/ # ls -l /etc/passwd
-rw-r--r-- 1 root root 207 Jan 1 00:29 /etc/passwd
/ #
我还尝试使用root权限运行我的程序(测试),但是当我给它一个现有用户名时它也失败了。
/ # /tmp/test admin
getpwnam_r: No such file or directory
/ #
1)那么,我忘记了什么,或者应该做些什么呢?
2)我是否需要使用 / etc / shadow 文件存储系统用户的密码?
更新
我的passwd文件是:
~ # cat /etc/passwd
root:b6MVch7fPLasN:0:0:root:/home/root:/bin/ash
admin:8Mt/Jtxcyg8AY:1000:1000:admin:/tmp:/tmp/cli
user:5v4HoPrA9NtUo:1001:1000:user:/tmp:/tmp/cli
~ #
提前致谢!巴克尔
答案 0 :(得分:1)
1)密码数据库中使用的搜索服务或方法( / etc / passwd )在 /etc/nsswitch.conf 。要使用此服务, getpwnam 函数会调用lib目录中的共享库:/lib/libnss_SERVICE.so.x,其中SERVICE是搜索方法。在我的情况下, compat 是默认方法,因为没有 /etc/nsswitch.conf 。所以,我需要将libnss_compat.so.2添加到/ lib。
strace 是有用的东西!
非常感谢 osqx 和 alk !