mirror of
https://github.com/samba-team/samba.git
synced 2024-12-31 17:18:04 +03:00
A tdb module based on some tips from Mike Noriega. This version supports
locking and tdb traversal using python callbacks.
(This used to be commit f3530fd618
)
This commit is contained in:
parent
27960c8f4f
commit
8a35fba95b
606
source3/python/py_tdb.c
Normal file
606
source3/python/py_tdb.c
Normal file
@ -0,0 +1,606 @@
|
||||
/*
|
||||
Python wrappers for TDB module
|
||||
|
||||
Copyright (C) Tim Potter, 2002
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#include "includes.h"
|
||||
#include "Python.h"
|
||||
|
||||
/* Tdb exception */
|
||||
|
||||
PyObject *py_tdb_error;
|
||||
|
||||
/* tdb handle object */
|
||||
|
||||
typedef struct {
|
||||
PyObject_HEAD
|
||||
TDB_CONTEXT *tdb;
|
||||
} tdb_hnd_object;
|
||||
|
||||
PyTypeObject tdb_hnd_type;
|
||||
|
||||
PyObject *new_tdb_hnd_object(TDB_CONTEXT *tdb)
|
||||
{
|
||||
tdb_hnd_object *obj;
|
||||
|
||||
obj = PyObject_New(tdb_hnd_object, &tdb_hnd_type);
|
||||
obj->tdb = tdb;
|
||||
|
||||
return (PyObject *)obj;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_close(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj;
|
||||
|
||||
if (!PyArg_ParseTuple(args, "O!", &tdb_hnd_type, &obj))
|
||||
return NULL;
|
||||
|
||||
if (tdb_close(obj->tdb) == -1) {
|
||||
obj->tdb = NULL;
|
||||
PyErr_SetString(py_tdb_error, strerror(errno));
|
||||
return NULL;
|
||||
}
|
||||
|
||||
obj->tdb = NULL;
|
||||
|
||||
Py_INCREF(Py_None);
|
||||
return Py_None;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_open(PyObject *self, PyObject *args, PyObject *kw)
|
||||
{
|
||||
static char *kwlist[] = { "name", "hash_size", "tdb_flags",
|
||||
"open_flags", "mode", NULL };
|
||||
char *name;
|
||||
int hash_size = 0, flags = TDB_DEFAULT, open_flags = -1, open_mode = 0600;
|
||||
TDB_CONTEXT *tdb;
|
||||
|
||||
if (!PyArg_ParseTupleAndKeywords(
|
||||
args, kw, "s|iiii", kwlist, &name, &hash_size, &flags,
|
||||
&open_flags, &open_mode))
|
||||
return NULL;
|
||||
|
||||
/* Default open_flags to read/write */
|
||||
|
||||
if (open_flags == -1) {
|
||||
if (access(name, W_OK) == -1)
|
||||
open_flags = O_RDONLY;
|
||||
else
|
||||
open_flags = O_RDWR;
|
||||
}
|
||||
|
||||
if (!(tdb = tdb_open(name, hash_size, flags, open_flags, open_mode))) {
|
||||
PyErr_SetString(py_tdb_error, strerror(errno));
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return new_tdb_hnd_object(tdb);
|
||||
}
|
||||
|
||||
/*
|
||||
* Allow a tdb to act as a python mapping (dictionary)
|
||||
*/
|
||||
|
||||
static int tdb_traverse_count(TDB_CONTEXT *tdb, TDB_DATA key, TDB_DATA value,
|
||||
void *state)
|
||||
{
|
||||
/* Do nothing - tdb_traverse will return the number of records
|
||||
traversed. */
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tdb_hnd_length(tdb_hnd_object *obj)
|
||||
{
|
||||
int result;
|
||||
|
||||
result = tdb_traverse(obj->tdb, tdb_traverse_count, NULL);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static PyObject *tdb_hnd_subscript(tdb_hnd_object *obj, PyObject *key)
|
||||
{
|
||||
TDB_DATA drec, krec;
|
||||
PyObject *result;
|
||||
|
||||
if (!PyArg_Parse(key, "s#", &krec.dptr, &krec.dsize))
|
||||
return NULL;
|
||||
|
||||
drec = tdb_fetch(obj->tdb, krec);
|
||||
|
||||
if (!drec.dptr) {
|
||||
PyErr_SetString(PyExc_KeyError,
|
||||
PyString_AsString(key));
|
||||
return NULL;
|
||||
}
|
||||
|
||||
result = PyString_FromStringAndSize(drec.dptr, drec.dsize);
|
||||
free(drec.dptr);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static int tdb_ass_subscript(tdb_hnd_object *obj, PyObject *key, PyObject *value)
|
||||
{
|
||||
TDB_DATA krec, drec;
|
||||
|
||||
if (!PyArg_Parse(key, "s#", &krec.dptr, &krec.dsize)) {
|
||||
PyErr_SetString(PyExc_TypeError,
|
||||
"tdb mappings have string indices only");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(
|
||||
py_tdb_error, "tdb object has been closed");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!value) {
|
||||
|
||||
/* Delete value */
|
||||
|
||||
if (tdb_delete(obj->tdb, krec) == -1) {
|
||||
PyErr_SetString(PyExc_KeyError,
|
||||
PyString_AsString(value));
|
||||
return -1;
|
||||
}
|
||||
|
||||
} else {
|
||||
|
||||
/* Set value */
|
||||
|
||||
if (!PyArg_Parse(value, "s#", &drec.dptr, &drec.dsize)) {
|
||||
PyErr_SetString(PyExc_TypeError,
|
||||
"tdb mappings have string elements only");
|
||||
return -1;
|
||||
}
|
||||
|
||||
errno = 0;
|
||||
|
||||
if (tdb_store(obj->tdb, krec, drec, 0) < 0 ) {
|
||||
if (errno != 0)
|
||||
PyErr_SetFromErrno(py_tdb_error);
|
||||
else
|
||||
PyErr_SetString(
|
||||
py_tdb_error,
|
||||
(char *)tdb_errorstr(obj->tdb));
|
||||
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static PyMappingMethods tdb_mapping = {
|
||||
(inquiry) tdb_hnd_length,
|
||||
(binaryfunc) tdb_hnd_subscript,
|
||||
(objobjargproc) tdb_ass_subscript
|
||||
};
|
||||
|
||||
/*
|
||||
* Utility methods
|
||||
*/
|
||||
|
||||
/* Return non-zero if a given key exists in the tdb */
|
||||
|
||||
PyObject *py_tdb_hnd_has_key(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
TDB_DATA key;
|
||||
|
||||
if (!PyArg_ParseTuple(args, "s#", &key.dptr, &key.dsize))
|
||||
return NULL;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(
|
||||
py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return PyInt_FromLong(tdb_exists(obj->tdb, key));
|
||||
}
|
||||
|
||||
/* Return a list of keys in the tdb */
|
||||
|
||||
static int tdb_traverse_keys(TDB_CONTEXT *tdb, TDB_DATA key, TDB_DATA value,
|
||||
void *state)
|
||||
{
|
||||
PyObject *key_list = (PyObject *)state;
|
||||
|
||||
PyList_Append(key_list,
|
||||
PyString_FromStringAndSize(key.dptr, key.dsize));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_keys(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
PyObject *key_list = PyList_New(0);
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (tdb_traverse(obj->tdb, tdb_traverse_keys, key_list) == -1) {
|
||||
PyErr_SetString(py_tdb_error, "error traversing tdb");
|
||||
Py_DECREF(key_list);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return key_list;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_first_key(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
TDB_DATA key;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
key = tdb_firstkey(obj->tdb);
|
||||
|
||||
return Py_BuildValue("s#", key.dptr, key.dsize);
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_next_key(PyObject *self, PyObject *py_oldkey)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
TDB_DATA key, oldkey;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (!PyArg_Parse(py_oldkey, "s#", &oldkey.dptr, &oldkey.dsize))
|
||||
return NULL;
|
||||
|
||||
key = tdb_nextkey(obj->tdb, oldkey);
|
||||
|
||||
return Py_BuildValue("s#", key.dptr, key.dsize);
|
||||
}
|
||||
|
||||
/*
|
||||
* Locking routines
|
||||
*/
|
||||
|
||||
PyObject *py_tdb_hnd_lock_all(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
int result;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
result = tdb_lockall(obj->tdb);
|
||||
|
||||
return PyInt_FromLong(result != -1);
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_unlock_all(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
tdb_unlockall(obj->tdb);
|
||||
|
||||
Py_INCREF(Py_None);
|
||||
return Py_None;
|
||||
}
|
||||
|
||||
/* Return an array of keys from a python object which must be a string or a
|
||||
list of strings. */
|
||||
|
||||
static BOOL make_lock_list(PyObject *py_keys, TDB_DATA **keys, int *num_keys)
|
||||
{
|
||||
/* Are we a list or a string? */
|
||||
|
||||
if (!PyList_Check(py_keys) && !PyString_Check(py_keys)) {
|
||||
PyErr_SetString(PyExc_TypeError, "arg must be list of string");
|
||||
return False;
|
||||
}
|
||||
|
||||
if (PyList_Check(py_keys)) {
|
||||
int i;
|
||||
|
||||
/* Turn python list into array of keys */
|
||||
|
||||
*num_keys = PyList_Size(py_keys);
|
||||
*keys = (TDB_DATA *)malloc(sizeof(TDB_DATA) * (*num_keys));
|
||||
|
||||
for (i = 0; i < *num_keys; i++) {
|
||||
PyObject *key = PyList_GetItem(py_keys, i);
|
||||
|
||||
if (!PyString_Check(key)) {
|
||||
PyErr_SetString(
|
||||
PyExc_TypeError,
|
||||
"list elements must be strings");
|
||||
return False;
|
||||
}
|
||||
|
||||
PyArg_Parse(key, "s#", &(*keys)[i].dptr,
|
||||
&(*keys)[i].dsize);
|
||||
}
|
||||
|
||||
} else {
|
||||
|
||||
/* Turn python string into a single key */
|
||||
|
||||
*keys = (TDB_DATA *)malloc(sizeof(TDB_DATA));
|
||||
*num_keys = 1;
|
||||
PyArg_Parse(py_keys, "s#", &(*keys)->dptr, &(*keys)->dsize);
|
||||
}
|
||||
|
||||
return True;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_lock(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
PyObject *py_keys;
|
||||
TDB_DATA *keys;
|
||||
int num_keys, result;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (!PyArg_ParseTuple(args, "O", &py_keys))
|
||||
return NULL;
|
||||
|
||||
if (!make_lock_list(py_keys, &keys, &num_keys))
|
||||
return NULL;
|
||||
|
||||
result = tdb_lockkeys(obj->tdb, num_keys, keys);
|
||||
|
||||
free(keys);
|
||||
|
||||
return PyInt_FromLong(result != -1);
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_unlock(PyObject *self, PyObject *args)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
|
||||
if (!obj->tdb) {
|
||||
PyErr_SetString(py_tdb_error, "tdb object has been closed");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (!PyArg_ParseTuple(args, ""))
|
||||
return NULL;
|
||||
|
||||
tdb_unlockkeys(obj->tdb);
|
||||
|
||||
Py_INCREF(Py_None);
|
||||
return Py_None;
|
||||
}
|
||||
|
||||
/*
|
||||
* tdb traversal
|
||||
*/
|
||||
|
||||
struct traverse_info {
|
||||
PyObject *callback;
|
||||
PyObject *state;
|
||||
};
|
||||
|
||||
static int tdb_traverse_traverse(TDB_CONTEXT *tdb, TDB_DATA key, TDB_DATA value,
|
||||
void *state)
|
||||
{
|
||||
struct traverse_info *info = state;
|
||||
PyObject *arglist, *py_result;
|
||||
int result;
|
||||
|
||||
arglist = Py_BuildValue("(s#s#O)", key.dptr, key.dsize, value.dptr,
|
||||
value.dsize, info->state);
|
||||
|
||||
py_result = PyEval_CallObject(info->callback, arglist);
|
||||
|
||||
Py_DECREF(arglist);
|
||||
|
||||
if (!PyInt_Check(py_result)) {
|
||||
result = 1; /* Hmm - non-integer object returned by callback */
|
||||
goto done;
|
||||
}
|
||||
|
||||
result = PyInt_AsLong(py_result);
|
||||
|
||||
done:
|
||||
Py_DECREF(py_result);
|
||||
return result;
|
||||
}
|
||||
|
||||
PyObject *py_tdb_hnd_traverse(PyObject *self, PyObject *args, PyObject *kw)
|
||||
{
|
||||
tdb_hnd_object *obj = (tdb_hnd_object *)self;
|
||||
static char *kwlist[] = { "traverse_fn", "state", NULL };
|
||||
PyObject *state = Py_None, *callback;
|
||||
struct traverse_info info;
|
||||
int result;
|
||||
|
||||
if (!PyArg_ParseTupleAndKeywords(
|
||||
args, kw, "O|O", kwlist, &callback, &state))
|
||||
return NULL;
|
||||
|
||||
if (!PyCallable_Check(callback)) {
|
||||
PyErr_SetString(PyExc_TypeError, "parameter must be callable");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
Py_INCREF(callback);
|
||||
Py_INCREF(state);
|
||||
|
||||
info.callback = callback;
|
||||
info.state = state;
|
||||
|
||||
result = tdb_traverse(obj->tdb, tdb_traverse_traverse, &info);
|
||||
|
||||
Py_DECREF(callback);
|
||||
Py_DECREF(state);
|
||||
|
||||
return PyInt_FromLong(result);
|
||||
}
|
||||
|
||||
/*
|
||||
* Method dispatch table for this module
|
||||
*/
|
||||
|
||||
static PyMethodDef tdb_methods[] = {
|
||||
{ "open", (PyCFunction)py_tdb_open, METH_VARARGS | METH_KEYWORDS },
|
||||
{ "close", (PyCFunction)py_tdb_close, METH_VARARGS },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
/*
|
||||
* Methods on a tdb object
|
||||
*/
|
||||
|
||||
static PyMethodDef tdb_hnd_methods[] = {
|
||||
{ "keys", (PyCFunction)py_tdb_hnd_keys, METH_VARARGS },
|
||||
{ "has_key", (PyCFunction)py_tdb_hnd_has_key, METH_VARARGS },
|
||||
{ "first_key", (PyCFunction)py_tdb_hnd_first_key, METH_VARARGS },
|
||||
{ "next_key", (PyCFunction)py_tdb_hnd_next_key, METH_VARARGS },
|
||||
{ "lock_all", (PyCFunction)py_tdb_hnd_lock_all, METH_VARARGS },
|
||||
{ "unlock_all", (PyCFunction)py_tdb_hnd_unlock_all, METH_VARARGS },
|
||||
{ "lock", (PyCFunction)py_tdb_hnd_lock, METH_VARARGS },
|
||||
{ "unlock", (PyCFunction)py_tdb_hnd_unlock, METH_VARARGS },
|
||||
{ "traverse", (PyCFunction)py_tdb_hnd_traverse, METH_VARARGS | METH_KEYWORDS },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
/* Deallocate a tdb handle object */
|
||||
|
||||
static void tdb_hnd_dealloc(PyObject* self)
|
||||
{
|
||||
tdb_hnd_object *hnd = (tdb_hnd_object *)self;
|
||||
|
||||
if (hnd->tdb) {
|
||||
tdb_close(hnd->tdb);
|
||||
hnd->tdb = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
/* Return tdb handle attributes */
|
||||
|
||||
static PyObject *tdb_hnd_getattr(PyObject *self, char *attrname)
|
||||
{
|
||||
return Py_FindMethod(tdb_hnd_methods, self, attrname);
|
||||
}
|
||||
|
||||
static char tdb_hnd_type_doc[] =
|
||||
"Python wrapper for tdb.";
|
||||
|
||||
PyTypeObject tdb_hnd_type = {
|
||||
PyObject_HEAD_INIT(NULL)
|
||||
0,
|
||||
"tdb",
|
||||
sizeof(tdb_hnd_object),
|
||||
0,
|
||||
tdb_hnd_dealloc, /* tp_dealloc*/
|
||||
0, /* tp_print*/
|
||||
tdb_hnd_getattr, /* tp_getattr*/
|
||||
0, /* tp_setattr*/
|
||||
0, /* tp_compare*/
|
||||
0, /* tp_repr*/
|
||||
0, /* tp_as_number*/
|
||||
0, /* tp_as_sequence*/
|
||||
&tdb_mapping, /* tp_as_mapping*/
|
||||
0, /* tp_hash */
|
||||
0, /* tp_call */
|
||||
0, /* tp_str */
|
||||
0, /* tp_getattro */
|
||||
0, /* tp_setattro */
|
||||
0, /* tp_as_buffer*/
|
||||
Py_TPFLAGS_DEFAULT, /* tp_flags */
|
||||
tdb_hnd_type_doc, /* tp_doc */
|
||||
};
|
||||
|
||||
/* Constants */
|
||||
|
||||
static struct const_vals {
|
||||
char *name;
|
||||
uint32 value;
|
||||
} module_const_vals[] = {
|
||||
|
||||
/* Flags for tdb_open() */
|
||||
|
||||
{ "TDB_DEFAULT", TDB_DEFAULT },
|
||||
{ "TDB_CLEAR_IF_FIRST", TDB_CLEAR_IF_FIRST },
|
||||
{ "TDB_INTERNAL", TDB_INTERNAL },
|
||||
{ "TDB_NOLOCK", TDB_NOLOCK },
|
||||
{ "TDB_NOMMAP", TDB_NOMMAP },
|
||||
{ "TDB_CONVERT", TDB_CONVERT },
|
||||
{ "TDB_BIGENDIAN", TDB_BIGENDIAN },
|
||||
|
||||
{ NULL },
|
||||
};
|
||||
|
||||
static void const_init(PyObject *dict)
|
||||
{
|
||||
struct const_vals *tmp;
|
||||
PyObject *obj;
|
||||
|
||||
for (tmp = module_const_vals; tmp->name; tmp++) {
|
||||
obj = PyInt_FromLong(tmp->value);
|
||||
PyDict_SetItemString(dict, tmp->name, obj);
|
||||
Py_DECREF(obj);
|
||||
}
|
||||
}
|
||||
|
||||
/* Module initialisation */
|
||||
|
||||
void inittdb(void)
|
||||
{
|
||||
PyObject *module, *dict;
|
||||
|
||||
/* Initialise module */
|
||||
|
||||
module = Py_InitModule("tdb", tdb_methods);
|
||||
dict = PyModule_GetDict(module);
|
||||
|
||||
py_tdb_error = PyErr_NewException("tdb.error", NULL, NULL);
|
||||
PyDict_SetItemString(dict, "error", py_tdb_error);
|
||||
|
||||
/* Initialise policy handle object */
|
||||
|
||||
tdb_hnd_type.ob_type = &PyType_Type;
|
||||
|
||||
PyDict_SetItemString(dict, "tdb.hnd",
|
||||
(PyObject *)&tdb_hnd_type);
|
||||
|
||||
/* Initialise constants */
|
||||
|
||||
const_init(dict);
|
||||
}
|
29
source3/python/py_tdb.h
Normal file
29
source3/python/py_tdb.h
Normal file
@ -0,0 +1,29 @@
|
||||
/*
|
||||
Python wrappers for DCERPC/SMB client routines.
|
||||
|
||||
Copyright (C) Tim Potter, 2002
|
||||
|
||||
This program is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program; if not, write to the Free Software
|
||||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#ifndef _PY_TDB_H
|
||||
#define _PY_TDB_H
|
||||
|
||||
#include "includes.h"
|
||||
#include "Python.h"
|
||||
|
||||
#include "python/py_common.h"
|
||||
|
||||
#endif /* _PY_TDB_H */
|
@ -141,5 +141,14 @@ setup(
|
||||
libraries = lib_list,
|
||||
library_dirs = ["/usr/kerberos/lib"],
|
||||
extra_objects = obj_list),
|
||||
|
||||
# tdb module
|
||||
|
||||
Extension(name = "tdb",
|
||||
sources = [samba_srcdir + "python/py_tdb.c"],
|
||||
libraries = lib_list,
|
||||
library_dirs = ["/usr/kerberos/lib"],
|
||||
extra_objects = obj_list),
|
||||
|
||||
],
|
||||
)
|
||||
|
Loading…
Reference in New Issue
Block a user