添加服务到名称服务开关

Dav*_*ond 10 c gnu nss passwd libc

所以我正在尝试向NSS(名称服务交换机)添加服务.请注意GNU指南,了解如何在此处执行此操作.我一直在关注那个指南.我需要实现一个与passwd数据库一起使用的服务.

我遇到的问题是我的模块没有被调用某些功能.让我在这里重现一些代码......

enum nss_status
_nss_myservice_setpwent (void) {
 printf( "@ %s\n", __FUNCTION__ ) ;
 return NSS_STATUS_SUCCESS ;
} ;

enum nss_status
_nss_myservice_endpwent (void) {
 printf( "@ %s\n", __FUNCTION__ ) ;
 return NSS_STATUS_SUCCESS ;
} ;

enum nss_status
_nss_myservice_getpwent_r (struct passwd *result, char *buffer,
   size_t buflen, int *errnop) {

 static int i = 0 ;

 if( i++ == 0 ) {
   printf( "@ %s\n", __FUNCTION__ ) ;
   return init_result( result, buffer, buflen, errnop ) ;
 } else {
   i = 0 ;
   return NSS_STATUS_NOTFOUND ;
 }
} ;

enum nss_status
_nss_myservice_getpwbynam (const char *nam, struct passwd *result, char *buffer,
   size_t buflen, int *errnop) {
 printf( "@ %s with name %s\n", __FUNCTION__, nam ) ;
 return init_result( result, buffer, buflen, errnop ) ;
} ;

enum nss_status
_nss_myservice_getpwbynam_r (const char *nam, struct passwd *result, char *buffer,
   size_t buflen, int *errnop) {
 printf( "@ %s with name_r %s\n", __FUNCTION__, nam ) ;
 return init_result( result, buffer, buflen, errnop ) ;
} ;
Run Code Online (Sandbox Code Playgroud)

Init_result是一个内联函数,无论PARAMS是什么,只需用虚拟用户填充结果.

现在我的/etc/nsswitch.conf设置如下:

passwd:         myservice compat
Run Code Online (Sandbox Code Playgroud)

为了完整性,这里是我的Makefile.

all:
       gcc -fPIC -shared -o libnss_myservice.so.2 -Wl,-soname,libnss_myservice.so.2 myservice.c
install:
       sudo install -m 0644 libnss_myservice.so.2 /lib
       sudo /sbin/ldconfig -n /lib /usr/lib
clean:
       /bin/rf -rf libnss_myservice.so.2
Run Code Online (Sandbox Code Playgroud)

现在安装这个nss模块后,我在命令行上运行getent,这是我的输出:

username@host:~/nss$ getent passwd
@ _nss_myservice_setpwent
@ _nss_myservice_getpwent_r
myuser:mypass:1:1:realname::
root:x:0:0:root:/root:/bin/bash
...
@ _nss_myservice_endpwent
Run Code Online (Sandbox Code Playgroud)

所以你可以看到它正如我所期望的那样工作.进行迭代调用,返回用户,然后调用compat服务,返回/ etc/passwd中的所有用户.

问题是,当我进行此调用时,"getent passwd myuser",我得到的返回值为2,"数据库中找不到密钥".这告诉我我的_nss_myservice_getpwbynam_r函数没有被调用.有什么想法吗?如果有帮助,我可以提供完整的代码.

Adr*_*iuk 7

你需要调用函数_nss_myservice_getpwnam_r而不是_nss_myservice_getpwbynam_r.

看完 ftp://ftp.acer-euro.com/gpl/Utility/glibc/glibc-2.2.5.tar/include/pwd.h后:

#define DECLARE_NSS_PROTOTYPES(service)                 \
extern enum nss_status _nss_ ## service ## _setpwent (int);     \
extern enum nss_status _nss_ ## service ## _endpwent (void);        \
extern enum nss_status _nss_ ## service ## _getpwnam_r          \        <<< this line
                       (const char *name, struct passwd *pwd,       \
            char *buffer, size_t buflen, int *errnop);  \
extern enum nss_status _nss_ ## service ## _getpwuid_r          \
                       (uid_t uid, struct passwd *pwd,          \
            char *buffer, size_t buflen, int *errnop);  \
extern enum nss_status _nss_ ## service ##_getpwent_r           \
                       (struct passwd *result, char *buffer,        \
            size_t buflen, int *errnop);
Run Code Online (Sandbox Code Playgroud)

  • 谢谢,那行得通。这让我想知道为什么GNU文档是这样写的。(枚举nss_status _nss_DATABASE_getdbbyXX_r(PARAMS,结构*结果,char *缓冲区,size_t buflen,int * errnop))我是否缺少某些内容?如果没有,我会给他们发个便条... (2认同)