|  | /*********************************************************** | 
|  | Written by: | 
|  | Fred Gansevles <Fred.Gansevles@cs.utwente.nl> | 
|  | B&O group, | 
|  | Faculteit der Informatica, | 
|  | Universiteit Twente, | 
|  | Enschede, | 
|  | the Netherlands. | 
|  | ******************************************************************/ | 
|  |  | 
|  | /* NIS module implementation */ | 
|  |  | 
|  | #include "Python.h" | 
|  |  | 
|  | #include <sys/time.h> | 
|  | #include <sys/types.h> | 
|  | #include <rpc/rpc.h> | 
|  | #include <rpcsvc/yp_prot.h> | 
|  | #include <rpcsvc/ypclnt.h> | 
|  |  | 
|  | #ifdef __sgi | 
|  | /* This is missing from rpcsvc/ypclnt.h */ | 
|  | extern int yp_get_default_domain(char **); | 
|  | #endif | 
|  |  | 
|  | PyDoc_STRVAR(get_default_domain__doc__, | 
|  | "get_default_domain() -> str\n\ | 
|  | Corresponds to the C library yp_get_default_domain() call, returning\n\ | 
|  | the default NIS domain.\n"); | 
|  |  | 
|  | PyDoc_STRVAR(match__doc__, | 
|  | "match(key, map, domain = defaultdomain)\n\ | 
|  | Corresponds to the C library yp_match() call, returning the value of\n\ | 
|  | key in the given map. Optionally domain can be specified but it\n\ | 
|  | defaults to the system default domain.\n"); | 
|  |  | 
|  | PyDoc_STRVAR(cat__doc__, | 
|  | "cat(map, domain = defaultdomain)\n\ | 
|  | Returns the entire map as a dictionary. Optionally domain can be\n\ | 
|  | specified but it defaults to the system default domain.\n"); | 
|  |  | 
|  | PyDoc_STRVAR(maps__doc__, | 
|  | "maps(domain = defaultdomain)\n\ | 
|  | Returns an array of all available NIS maps within a domain. If domain\n\ | 
|  | is not specified it defaults to the system default domain.\n"); | 
|  |  | 
|  | typedef struct { | 
|  | PyObject *nis_error; | 
|  | } nis_state; | 
|  |  | 
|  | static inline nis_state* | 
|  | get_nis_state(PyObject *module) | 
|  | { | 
|  | void *state = PyModule_GetState(module); | 
|  | assert(state != NULL); | 
|  | return (nis_state *)state; | 
|  | } | 
|  |  | 
|  | static int | 
|  | nis_clear(PyObject *m) | 
|  | { | 
|  | Py_CLEAR(get_nis_state(m)->nis_error); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int | 
|  | nis_traverse(PyObject *m, visitproc visit, void *arg) | 
|  | { | 
|  | Py_VISIT(get_nis_state(m)->nis_error); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void | 
|  | nis_free(void *m) | 
|  | { | 
|  | nis_clear((PyObject *) m); | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | nis_error(nis_state *state, int err) | 
|  | { | 
|  | PyErr_SetString(state->nis_error, yperr_string(err)); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static struct nis_map { | 
|  | char *alias; | 
|  | char *map; | 
|  | int  fix; | 
|  | } aliases [] = { | 
|  | {"passwd",          "passwd.byname",        0}, | 
|  | {"group",           "group.byname",         0}, | 
|  | {"networks",        "networks.byaddr",      0}, | 
|  | {"hosts",           "hosts.byname",         0}, | 
|  | {"protocols",       "protocols.bynumber",   0}, | 
|  | {"services",        "services.byname",      0}, | 
|  | {"aliases",         "mail.aliases",         1}, /* created with 'makedbm -a' */ | 
|  | {"ethers",          "ethers.byname",        0}, | 
|  | {0L,                0L,                     0} | 
|  | }; | 
|  |  | 
|  | static char * | 
|  | nis_mapname(char *map, int *pfix) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | *pfix = 0; | 
|  | for (i=0; aliases[i].alias != 0L; i++) { | 
|  | if (!strcmp (aliases[i].alias, map) || !strcmp (aliases[i].map, map)) { | 
|  | *pfix = aliases[i].fix; | 
|  | return aliases[i].map; | 
|  | } | 
|  | } | 
|  |  | 
|  | return map; | 
|  | } | 
|  |  | 
|  | #if defined(__APPLE__) || defined(__OpenBSD__) || defined(__FreeBSD__) | 
|  | typedef int (*foreachfunc)(unsigned long, char *, int, char *, int, void *); | 
|  | #else | 
|  | typedef int (*foreachfunc)(int, char *, int, char *, int, char *); | 
|  | #endif | 
|  |  | 
|  | struct ypcallback_data { | 
|  | PyObject            *dict; | 
|  | int                         fix; | 
|  | PyThreadState *state; | 
|  | }; | 
|  |  | 
|  | static int | 
|  | nis_foreach(int instatus, char *inkey, int inkeylen, char *inval, | 
|  | int invallen, struct ypcallback_data *indata) | 
|  | { | 
|  | if (instatus == YP_TRUE) { | 
|  | PyObject *key; | 
|  | PyObject *val; | 
|  | int err; | 
|  |  | 
|  | PyEval_RestoreThread(indata->state); | 
|  | if (indata->fix) { | 
|  | if (inkeylen > 0 && inkey[inkeylen-1] == '\0') | 
|  | inkeylen--; | 
|  | if (invallen > 0 && inval[invallen-1] == '\0') | 
|  | invallen--; | 
|  | } | 
|  | key = PyUnicode_DecodeFSDefaultAndSize(inkey, inkeylen); | 
|  | val = PyUnicode_DecodeFSDefaultAndSize(inval, invallen); | 
|  | if (key == NULL || val == NULL) { | 
|  | /* XXX error -- don't know how to handle */ | 
|  | PyErr_Clear(); | 
|  | Py_XDECREF(key); | 
|  | Py_XDECREF(val); | 
|  | indata->state = PyEval_SaveThread(); | 
|  | return 1; | 
|  | } | 
|  | err = PyDict_SetItem(indata->dict, key, val); | 
|  | Py_DECREF(key); | 
|  | Py_DECREF(val); | 
|  | if (err != 0) | 
|  | PyErr_Clear(); | 
|  | indata->state = PyEval_SaveThread(); | 
|  | if (err != 0) | 
|  | return 1; | 
|  | return 0; | 
|  | } | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | nis_get_default_domain(PyObject *module, PyObject *Py_UNUSED(ignored)) | 
|  | { | 
|  | char *domain; | 
|  | int err; | 
|  | PyObject *res; | 
|  | nis_state *state = get_nis_state(module); | 
|  | if ((err = yp_get_default_domain(&domain)) != 0) { | 
|  | return nis_error(state, err); | 
|  | } | 
|  |  | 
|  | res = PyUnicode_FromStringAndSize (domain, strlen(domain)); | 
|  | return res; | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | nis_match(PyObject *module, PyObject *args, PyObject *kwdict) | 
|  | { | 
|  | char *match; | 
|  | char *domain = NULL; | 
|  | Py_ssize_t keylen; | 
|  | int len; | 
|  | char *key, *map; | 
|  | int err; | 
|  | PyObject *ukey, *bkey, *res; | 
|  | int fix; | 
|  | static char *kwlist[] = {"key", "map", "domain", NULL}; | 
|  |  | 
|  | if (!PyArg_ParseTupleAndKeywords(args, kwdict, | 
|  | "Us|s:match", kwlist, | 
|  | &ukey, &map, &domain)) { | 
|  | return NULL; | 
|  | } | 
|  | if ((bkey = PyUnicode_EncodeFSDefault(ukey)) == NULL) { | 
|  | return NULL; | 
|  | } | 
|  | /* check for embedded null bytes */ | 
|  | if (PyBytes_AsStringAndSize(bkey, &key, &keylen) == -1) { | 
|  | Py_DECREF(bkey); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | nis_state *state = get_nis_state(module); | 
|  | if (!domain && ((err = yp_get_default_domain(&domain)) != 0)) { | 
|  | Py_DECREF(bkey); | 
|  | return nis_error(state, err); | 
|  | } | 
|  | map = nis_mapname (map, &fix); | 
|  | if (fix) | 
|  | keylen++; | 
|  | Py_BEGIN_ALLOW_THREADS | 
|  | err = yp_match (domain, map, key, keylen, &match, &len); | 
|  | Py_END_ALLOW_THREADS | 
|  | Py_DECREF(bkey); | 
|  | if (fix) | 
|  | len--; | 
|  | if (err != 0) { | 
|  | return nis_error(state, err); | 
|  | } | 
|  | res = PyUnicode_DecodeFSDefaultAndSize(match, len); | 
|  | free (match); | 
|  | return res; | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | nis_cat(PyObject *module, PyObject *args, PyObject *kwdict) | 
|  | { | 
|  | char *domain = NULL; | 
|  | char *map; | 
|  | struct ypall_callback cb; | 
|  | struct ypcallback_data data; | 
|  | PyObject *dict; | 
|  | int err; | 
|  | static char *kwlist[] = {"map", "domain", NULL}; | 
|  |  | 
|  | if (!PyArg_ParseTupleAndKeywords(args, kwdict, "s|s:cat", | 
|  | kwlist, &map, &domain)) { | 
|  | return NULL; | 
|  | } | 
|  | nis_state *state = get_nis_state(module); | 
|  | if (!domain && ((err = yp_get_default_domain(&domain)) != 0)) { | 
|  | return nis_error(state, err); | 
|  | } | 
|  | dict = PyDict_New (); | 
|  | if (dict == NULL) | 
|  | return NULL; | 
|  | cb.foreach = (foreachfunc)nis_foreach; | 
|  | data.dict = dict; | 
|  | map = nis_mapname (map, &data.fix); | 
|  | cb.data = (char *)&data; | 
|  | data.state = PyEval_SaveThread(); | 
|  | err = yp_all (domain, map, &cb); | 
|  | PyEval_RestoreThread(data.state); | 
|  | if (err != 0) { | 
|  | Py_DECREF(dict); | 
|  | return nis_error(state, err); | 
|  | } | 
|  | return dict; | 
|  | } | 
|  |  | 
|  | /* These should be u_long on Sun h/w but not on 64-bit h/w. | 
|  | This is not portable to machines with 16-bit ints and no prototypes */ | 
|  | #ifndef YPPROC_MAPLIST | 
|  | #define YPPROC_MAPLIST  11 | 
|  | #endif | 
|  | #ifndef YPPROG | 
|  | #define YPPROG          100004 | 
|  | #endif | 
|  | #ifndef YPVERS | 
|  | #define YPVERS          2 | 
|  | #endif | 
|  |  | 
|  | typedef char *domainname; | 
|  | typedef char *mapname; | 
|  |  | 
|  | enum nisstat { | 
|  | NIS_TRUE = 1, | 
|  | NIS_NOMORE = 2, | 
|  | NIS_FALSE = 0, | 
|  | NIS_NOMAP = -1, | 
|  | NIS_NODOM = -2, | 
|  | NIS_NOKEY = -3, | 
|  | NIS_BADOP = -4, | 
|  | NIS_BADDB = -5, | 
|  | NIS_YPERR = -6, | 
|  | NIS_BADARGS = -7, | 
|  | NIS_VERS = -8 | 
|  | }; | 
|  | typedef enum nisstat nisstat; | 
|  |  | 
|  | struct nismaplist { | 
|  | mapname map; | 
|  | struct nismaplist *next; | 
|  | }; | 
|  | typedef struct nismaplist nismaplist; | 
|  |  | 
|  | struct nisresp_maplist { | 
|  | nisstat stat; | 
|  | nismaplist *maps; | 
|  | }; | 
|  | typedef struct nisresp_maplist nisresp_maplist; | 
|  |  | 
|  | static struct timeval TIMEOUT = { 25, 0 }; | 
|  |  | 
|  | static | 
|  | bool_t | 
|  | nis_xdr_domainname(XDR *xdrs, domainname *objp) | 
|  | { | 
|  | if (!xdr_string(xdrs, objp, YPMAXDOMAIN)) { | 
|  | return (FALSE); | 
|  | } | 
|  | return (TRUE); | 
|  | } | 
|  |  | 
|  | static | 
|  | bool_t | 
|  | nis_xdr_mapname(XDR *xdrs, mapname *objp) | 
|  | { | 
|  | if (!xdr_string(xdrs, objp, YPMAXMAP)) { | 
|  | return (FALSE); | 
|  | } | 
|  | return (TRUE); | 
|  | } | 
|  |  | 
|  | static | 
|  | bool_t | 
|  | nis_xdr_ypmaplist(XDR *xdrs, nismaplist *objp) | 
|  | { | 
|  | if (!nis_xdr_mapname(xdrs, &objp->map)) { | 
|  | return (FALSE); | 
|  | } | 
|  | if (!xdr_pointer(xdrs, (char **)&objp->next, | 
|  | sizeof(nismaplist), (xdrproc_t)nis_xdr_ypmaplist)) | 
|  | { | 
|  | return (FALSE); | 
|  | } | 
|  | return (TRUE); | 
|  | } | 
|  |  | 
|  | static | 
|  | bool_t | 
|  | nis_xdr_ypstat(XDR *xdrs, nisstat *objp) | 
|  | { | 
|  | if (!xdr_enum(xdrs, (enum_t *)objp)) { | 
|  | return (FALSE); | 
|  | } | 
|  | return (TRUE); | 
|  | } | 
|  |  | 
|  |  | 
|  | static | 
|  | bool_t | 
|  | nis_xdr_ypresp_maplist(XDR *xdrs, nisresp_maplist *objp) | 
|  | { | 
|  | if (!nis_xdr_ypstat(xdrs, &objp->stat)) { | 
|  | return (FALSE); | 
|  | } | 
|  | if (!xdr_pointer(xdrs, (char **)&objp->maps, | 
|  | sizeof(nismaplist), (xdrproc_t)nis_xdr_ypmaplist)) | 
|  | { | 
|  | return (FALSE); | 
|  | } | 
|  | return (TRUE); | 
|  | } | 
|  |  | 
|  |  | 
|  | static | 
|  | nisresp_maplist * | 
|  | nisproc_maplist_2(domainname *argp, CLIENT *clnt) | 
|  | { | 
|  | static nisresp_maplist res; | 
|  |  | 
|  | memset(&res, 0, sizeof(res)); | 
|  | if (clnt_call(clnt, YPPROC_MAPLIST, | 
|  | (xdrproc_t)nis_xdr_domainname, (caddr_t)argp, | 
|  | (xdrproc_t)nis_xdr_ypresp_maplist, (caddr_t)&res, | 
|  | TIMEOUT) != RPC_SUCCESS) | 
|  | { | 
|  | return (NULL); | 
|  | } | 
|  | return (&res); | 
|  | } | 
|  |  | 
|  | static | 
|  | nismaplist * | 
|  | nis_maplist(nis_state *state, char *dom) | 
|  | { | 
|  | nisresp_maplist *list; | 
|  | CLIENT *cl; | 
|  | char *server = NULL; | 
|  | int mapi = 0; | 
|  |  | 
|  | while (!server && aliases[mapi].map != 0L) { | 
|  | yp_master (dom, aliases[mapi].map, &server); | 
|  | mapi++; | 
|  | } | 
|  | if (!server) { | 
|  | PyErr_SetString(state->nis_error, "No NIS master found for any map"); | 
|  | return NULL; | 
|  | } | 
|  | cl = clnt_create(server, YPPROG, YPVERS, "tcp"); | 
|  | if (cl == NULL) { | 
|  | PyErr_SetString(state->nis_error, clnt_spcreateerror(server)); | 
|  | goto finally; | 
|  | } | 
|  | list = nisproc_maplist_2 (&dom, cl); | 
|  | clnt_destroy(cl); | 
|  | if (list == NULL) | 
|  | goto finally; | 
|  | if (list->stat != NIS_TRUE) | 
|  | goto finally; | 
|  |  | 
|  | free(server); | 
|  | return list->maps; | 
|  |  | 
|  | finally: | 
|  | free(server); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static PyObject * | 
|  | nis_maps (PyObject *module, PyObject *args, PyObject *kwdict) | 
|  | { | 
|  | char *domain = NULL; | 
|  | nismaplist *maps; | 
|  | PyObject *list; | 
|  | int err; | 
|  | static char *kwlist[] = {"domain", NULL}; | 
|  |  | 
|  | if (!PyArg_ParseTupleAndKeywords(args, kwdict, | 
|  | "|s:maps", kwlist, &domain)) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | nis_state *state = get_nis_state(module); | 
|  | if (!domain && ((err = yp_get_default_domain (&domain)) != 0)) { | 
|  | nis_error(state, err); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | if ((maps = nis_maplist(state, domain)) == NULL) { | 
|  | return NULL; | 
|  | } | 
|  | if ((list = PyList_New(0)) == NULL) { | 
|  | return NULL; | 
|  | } | 
|  | for (; maps; maps = maps->next) { | 
|  | PyObject *str = PyUnicode_FromString(maps->map); | 
|  | if (!str || PyList_Append(list, str) < 0) | 
|  | { | 
|  | Py_XDECREF(str); | 
|  | Py_DECREF(list); | 
|  | list = NULL; | 
|  | break; | 
|  | } | 
|  | Py_DECREF(str); | 
|  | } | 
|  | /* XXX Shouldn't we free the list of maps now? */ | 
|  | return list; | 
|  | } | 
|  |  | 
|  | static PyMethodDef nis_methods[] = { | 
|  | {"match",                   (PyCFunction)(void(*)(void))nis_match, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | match__doc__}, | 
|  | {"cat",                     (PyCFunction)(void(*)(void))nis_cat, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | cat__doc__}, | 
|  | {"maps",                    (PyCFunction)(void(*)(void))nis_maps, | 
|  | METH_VARARGS | METH_KEYWORDS, | 
|  | maps__doc__}, | 
|  | {"get_default_domain",      nis_get_default_domain, | 
|  | METH_NOARGS, | 
|  | get_default_domain__doc__}, | 
|  | {NULL,                      NULL}            /* Sentinel */ | 
|  | }; | 
|  |  | 
|  | static int | 
|  | nis_exec(PyObject *module) | 
|  | { | 
|  | nis_state* state = get_nis_state(module); | 
|  | state->nis_error = PyErr_NewException("nis.error", NULL, NULL); | 
|  | if (state->nis_error == NULL) { | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | Py_INCREF(state->nis_error); | 
|  | if (PyModule_AddObject(module, "error", state->nis_error) < 0) { | 
|  | Py_DECREF(state->nis_error); | 
|  | return -1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static PyModuleDef_Slot nis_slots[] = { | 
|  | {Py_mod_exec, nis_exec}, | 
|  | {0, NULL} | 
|  | }; | 
|  |  | 
|  | PyDoc_STRVAR(nis__doc__, | 
|  | "This module contains functions for accessing NIS maps.\n"); | 
|  |  | 
|  | static struct PyModuleDef nismodule = { | 
|  | PyModuleDef_HEAD_INIT, | 
|  | .m_name = "nis", | 
|  | .m_doc = nis__doc__, | 
|  | .m_size = sizeof(nis_state), | 
|  | .m_methods = nis_methods, | 
|  | .m_traverse = nis_traverse, | 
|  | .m_clear = nis_clear, | 
|  | .m_free = nis_free, | 
|  | .m_slots = nis_slots, | 
|  | }; | 
|  |  | 
|  | PyMODINIT_FUNC | 
|  | PyInit_nis(void) | 
|  | { | 
|  | return PyModuleDef_Init(&nismodule); | 
|  | } |