mirror of
https://github.com/samba-team/samba.git
synced 2025-01-13 13:18:06 +03:00
8420a36dc7
Signed-off-by: Andrew Tridgell <tridge@samba.org>
430 lines
8.4 KiB
C
430 lines
8.4 KiB
C
/*
|
|
LDB nsswitch module
|
|
|
|
Copyright (C) Simo Sorce 2006
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 3 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "ldb-nss.h"
|
|
|
|
extern struct _ldb_nss_context *_ldb_nss_ctx;
|
|
|
|
const char *_ldb_nss_gr_attrs[] = {
|
|
"cn",
|
|
"userPassword",
|
|
"gidNumber",
|
|
NULL
|
|
};
|
|
|
|
const char *_ldb_nss_mem_attrs[] = {
|
|
"uid",
|
|
NULL
|
|
};
|
|
|
|
#define _NSS_LDB_ENOMEM(amem) \
|
|
do { \
|
|
if ( ! amem) { \
|
|
errno = ENOMEM; \
|
|
talloc_free(memctx); \
|
|
return NSS_STATUS_UNAVAIL; \
|
|
} \
|
|
} while(0)
|
|
|
|
/* This setgrent, getgrent, endgrent is not very efficient */
|
|
|
|
NSS_STATUS _nss_ldb_setgrent(void)
|
|
{
|
|
int ret;
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
_ldb_nss_ctx->gr_cur = 0;
|
|
if (_ldb_nss_ctx->gr_res != NULL) {
|
|
talloc_free(_ldb_nss_ctx->gr_res);
|
|
_ldb_nss_ctx->gr_res = NULL;
|
|
}
|
|
|
|
ret = ldb_search(_ldb_nss_ctx->ldb,
|
|
_ldb_nss_ctx->ldb,
|
|
&_ldb_nss_ctx->gr_res,
|
|
_ldb_nss_ctx->base,
|
|
LDB_SCOPE_SUBTREE,
|
|
_ldb_nss_gr_attrs,
|
|
_LDB_NSS_GRENT_FILTER);
|
|
if (ret != LDB_SUCCESS) {
|
|
return NSS_STATUS_UNAVAIL;
|
|
}
|
|
|
|
return NSS_STATUS_SUCCESS;
|
|
}
|
|
|
|
NSS_STATUS _nss_ldb_endgrent(void)
|
|
{
|
|
int ret;
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
_ldb_nss_ctx->gr_cur = 0;
|
|
if (_ldb_nss_ctx->gr_res) {
|
|
talloc_free(_ldb_nss_ctx->gr_res);
|
|
_ldb_nss_ctx->gr_res = NULL;
|
|
}
|
|
|
|
return NSS_STATUS_SUCCESS;
|
|
}
|
|
|
|
NSS_STATUS _nss_ldb_getgrent_r(struct group *result_buf, char *buffer, size_t buflen, int *errnop)
|
|
{
|
|
int ret;
|
|
struct ldb_result *res;
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
*errnop = 0;
|
|
|
|
if (_ldb_nss_ctx->gr_cur >= _ldb_nss_ctx->gr_res->count) {
|
|
/* already returned all entries */
|
|
return NSS_STATUS_NOTFOUND;
|
|
}
|
|
|
|
res = talloc_zero(_ldb_nss_ctx->gr_res, struct ldb_result);
|
|
if ( ! res) {
|
|
errno = *errnop = ENOMEM;
|
|
_ldb_nss_ctx->gr_cur++; /* skip this entry */
|
|
return NSS_STATUS_UNAVAIL;
|
|
}
|
|
|
|
ret = _ldb_nss_group_request(&res,
|
|
_ldb_nss_ctx->gr_res->msgs[_ldb_nss_ctx->gr_cur]->dn,
|
|
_ldb_nss_mem_attrs,
|
|
"member");
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
*errnop = errno;
|
|
talloc_free(res);
|
|
_ldb_nss_ctx->gr_cur++; /* skip this entry */
|
|
return ret;
|
|
}
|
|
|
|
ret = _ldb_nss_fill_group(result_buf,
|
|
buffer,
|
|
buflen,
|
|
errnop,
|
|
_ldb_nss_ctx->gr_res->msgs[_ldb_nss_ctx->gr_cur],
|
|
res);
|
|
|
|
talloc_free(res);
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
if (ret != NSS_STATUS_TRYAGAIN) {
|
|
_ldb_nss_ctx->gr_cur++; /* skip this entry */
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/* this entry is ok, increment counter to nex entry */
|
|
_ldb_nss_ctx->gr_cur++;
|
|
|
|
return NSS_STATUS_SUCCESS;
|
|
}
|
|
|
|
NSS_STATUS _nss_ldb_getgrnam_r(const char *name, struct group *result_buf, char *buffer, size_t buflen, int *errnop)
|
|
{
|
|
int ret;
|
|
char *filter;
|
|
TALLOC_CTX *ctx;
|
|
struct ldb_result *gr_res;
|
|
struct ldb_result *mem_res;
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
ctx = talloc_new(_ldb_nss_ctx->ldb);
|
|
if ( ! ctx) {
|
|
*errnop = errno = ENOMEM;
|
|
return NSS_STATUS_UNAVAIL;
|
|
}
|
|
|
|
/* build the filter for this uid */
|
|
filter = talloc_asprintf(ctx, _LDB_NSS_GRNAM_FILTER, name);
|
|
if (filter == NULL) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOMEM;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
/* search the entry */
|
|
ret = ldb_search(_ldb_nss_ctx->ldb,
|
|
_ldb_nss_ctx->ldb,
|
|
&gr_res,
|
|
_ldb_nss_ctx->base,
|
|
LDB_SCOPE_SUBTREE,
|
|
_ldb_nss_gr_attrs,
|
|
filter);
|
|
if (ret != LDB_SUCCESS) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
talloc_steal(ctx, gr_res);
|
|
|
|
/* if none found return */
|
|
if (gr_res->count == 0) {
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_NOTFOUND;
|
|
goto done;
|
|
}
|
|
|
|
if (gr_res->count != 1) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
mem_res = talloc_zero(ctx, struct ldb_result);
|
|
if ( ! mem_res) {
|
|
errno = *errnop = ENOMEM;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_group_request(&mem_res,
|
|
gr_res->msgs[0]->dn,
|
|
_ldb_nss_mem_attrs,
|
|
"member");
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
*errnop = errno;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_fill_group(result_buf,
|
|
buffer,
|
|
buflen,
|
|
errnop,
|
|
gr_res->msgs[0],
|
|
mem_res);
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
goto done;
|
|
}
|
|
|
|
ret = NSS_STATUS_SUCCESS;
|
|
done:
|
|
talloc_free(ctx);
|
|
return ret;
|
|
}
|
|
|
|
NSS_STATUS _nss_ldb_getgrgid_r(gid_t gid, struct group *result_buf, char *buffer, size_t buflen, int *errnop)
|
|
{
|
|
int ret;
|
|
char *filter;
|
|
TALLOC_CTX *ctx;
|
|
struct ldb_result *gr_res;
|
|
struct ldb_result *mem_res;
|
|
|
|
if (gid == 0) { /* we don't serve root gid by policy */
|
|
*errnop = errno = ENOENT;
|
|
return NSS_STATUS_NOTFOUND;
|
|
}
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
ctx = talloc_new(_ldb_nss_ctx->ldb);
|
|
if ( ! ctx) {
|
|
*errnop = errno = ENOMEM;
|
|
return NSS_STATUS_UNAVAIL;
|
|
}
|
|
|
|
/* build the filter for this uid */
|
|
filter = talloc_asprintf(ctx, _LDB_NSS_GRGID_FILTER, gid);
|
|
if (filter == NULL) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOMEM;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
/* search the entry */
|
|
ret = ldb_search(_ldb_nss_ctx->ldb,
|
|
_ldb_nss_ctx->ldb,
|
|
&gr_res,
|
|
_ldb_nss_ctx->base,
|
|
LDB_SCOPE_SUBTREE,
|
|
_ldb_nss_gr_attrs,
|
|
filter);
|
|
if (ret != LDB_SUCCESS) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
talloc_steal(ctx, gr_res);
|
|
|
|
/* if none found return */
|
|
if (gr_res->count == 0) {
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_NOTFOUND;
|
|
goto done;
|
|
}
|
|
|
|
if (gr_res->count != 1) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
mem_res = talloc_zero(ctx, struct ldb_result);
|
|
if ( ! mem_res) {
|
|
errno = *errnop = ENOMEM;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_group_request(&mem_res,
|
|
gr_res->msgs[0]->dn,
|
|
_ldb_nss_mem_attrs,
|
|
"member");
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
*errnop = errno;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_fill_group(result_buf,
|
|
buffer,
|
|
buflen,
|
|
errnop,
|
|
gr_res->msgs[0],
|
|
mem_res);
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
goto done;
|
|
}
|
|
|
|
ret = NSS_STATUS_SUCCESS;
|
|
done:
|
|
talloc_free(ctx);
|
|
return ret;
|
|
}
|
|
|
|
NSS_STATUS _nss_ldb_initgroups_dyn(const char *user, gid_t group, long int *start, long int *size, gid_t **groups, long int limit, int *errnop)
|
|
{
|
|
int ret;
|
|
char *filter;
|
|
const char * attrs[] = { "uidNumber", "gidNumber", NULL };
|
|
struct ldb_result *uid_res;
|
|
struct ldb_result *mem_res;
|
|
|
|
ret = _ldb_nss_init();
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
return ret;
|
|
}
|
|
|
|
mem_res = talloc_zero(_ldb_nss_ctx, struct ldb_result);
|
|
if ( ! mem_res) {
|
|
errno = *errnop = ENOMEM;
|
|
return NSS_STATUS_UNAVAIL;
|
|
}
|
|
|
|
/* build the filter for this name */
|
|
filter = talloc_asprintf(mem_res, _LDB_NSS_PWNAM_FILTER, user);
|
|
if (filter == NULL) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
/* search the entry */
|
|
ret = ldb_search(_ldb_nss_ctx->ldb,
|
|
_ldb_nss_ctx->ldb,
|
|
&uid_res,
|
|
_ldb_nss_ctx->base,
|
|
LDB_SCOPE_SUBTREE,
|
|
attrs,
|
|
filter);
|
|
if (ret != LDB_SUCCESS) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
talloc_steal(mem_res, uid_res);
|
|
|
|
/* if none found return */
|
|
if (uid_res->count == 0) {
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_NOTFOUND;
|
|
goto done;
|
|
}
|
|
|
|
if (uid_res->count != 1) {
|
|
/* this is a fatal error */
|
|
*errnop = errno = ENOENT;
|
|
ret = NSS_STATUS_UNAVAIL;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_group_request(&mem_res,
|
|
uid_res->msgs[0]->dn,
|
|
attrs,
|
|
"memberOf");
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
*errnop = errno;
|
|
goto done;
|
|
}
|
|
|
|
ret = _ldb_nss_fill_initgr(group,
|
|
limit,
|
|
start,
|
|
size,
|
|
groups,
|
|
errnop,
|
|
mem_res);
|
|
|
|
if (ret != NSS_STATUS_SUCCESS) {
|
|
goto done;
|
|
}
|
|
|
|
ret = NSS_STATUS_SUCCESS;
|
|
|
|
done:
|
|
talloc_free(mem_res);
|
|
return ret;
|
|
}
|