Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation. Xattr manipulation bindings.
3 : Copyright (C) Matthieu Patou <mat@matws.net> 2009-2010
4 : Base on work of pyglue.c by Jelmer Vernooij <jelmer@samba.org> 2007 and
5 : Matthias Dieter Wallnöfer 2009
6 :
7 : This program is free software; you can redistribute it and/or modify
8 : it under the terms of the GNU General Public License as published by
9 : the Free Software Foundation; either version 3 of the License, or
10 : (at your option) any later version.
11 :
12 : This program is distributed in the hope that it will be useful,
13 : but WITHOUT ANY WARRANTY; without even the implied warranty of
14 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 : GNU General Public License for more details.
16 :
17 : You should have received a copy of the GNU General Public License
18 : along with this program. If not, see <http://www.gnu.org/licenses/>.
19 : */
20 :
21 : #include <Python.h>
22 : #include "python/py3compat.h"
23 : #include "includes.h"
24 : #include "system/filesys.h"
25 : #include <tdb.h>
26 : #include "lib/tdb_wrap/tdb_wrap.h"
27 : #include "librpc/ndr/libndr.h"
28 : #include "ntvfs/posix/posix_eadb.h"
29 : #include "libcli/util/pyerrors.h"
30 : #include "param/pyparam.h"
31 : #include "lib/dbwrap/dbwrap.h"
32 : #include "lib/dbwrap/dbwrap_open.h"
33 : #include "lib/dbwrap/dbwrap_tdb.h"
34 : #include "source3/lib/xattr_tdb.h"
35 :
36 0 : static PyObject *py_is_xattr_supported(PyObject *self,
37 : PyObject *Py_UNUSED(ignored))
38 : {
39 0 : Py_RETURN_TRUE;
40 : }
41 :
42 4 : static PyObject *py_wrap_setxattr(PyObject *self, PyObject *args)
43 : {
44 : char *filename, *attribute, *tdbname;
45 : DATA_BLOB blob;
46 : Py_ssize_t blobsize;
47 : int ret;
48 : TALLOC_CTX *mem_ctx;
49 : struct loadparm_context *lp_ctx;
50 4 : struct db_context *eadb = NULL;
51 : struct file_id id;
52 : struct stat sbuf;
53 :
54 4 : if (!PyArg_ParseTuple(args, "sss"PYARG_BYTES_LEN, &tdbname, &filename, &attribute,
55 : &blob.data, &blobsize))
56 0 : return NULL;
57 :
58 4 : blob.length = blobsize;
59 4 : mem_ctx = talloc_new(NULL);
60 :
61 4 : lp_ctx = py_default_loadparm_context(mem_ctx);
62 4 : eadb = db_open_tdb(mem_ctx, tdbname, 50000,
63 : lpcfg_tdb_flags(lp_ctx, TDB_DEFAULT),
64 : O_RDWR|O_CREAT, 0600, DBWRAP_LOCK_ORDER_2,
65 : DBWRAP_FLAG_NONE);
66 :
67 4 : if (eadb == NULL) {
68 0 : PyErr_SetFromErrno(PyExc_IOError);
69 0 : talloc_free(mem_ctx);
70 0 : return NULL;
71 : }
72 :
73 4 : ret = stat(filename, &sbuf);
74 4 : if (ret < 0) {
75 0 : PyErr_SetFromErrno(PyExc_IOError);
76 0 : talloc_free(mem_ctx);
77 0 : return NULL;
78 : }
79 :
80 4 : ZERO_STRUCT(id);
81 4 : id.devid = sbuf.st_dev;
82 4 : id.inode = sbuf.st_ino;
83 :
84 4 : ret = xattr_tdb_setattr(eadb, &id, attribute, blob.data, blob.length, 0);
85 4 : if (ret < 0) {
86 0 : PyErr_SetFromErrno(PyExc_TypeError);
87 0 : talloc_free(mem_ctx);
88 0 : return NULL;
89 : }
90 4 : talloc_free(mem_ctx);
91 4 : Py_RETURN_NONE;
92 : }
93 :
94 7 : static PyObject *py_wrap_getxattr(PyObject *self, PyObject *args)
95 : {
96 : char *filename, *attribute, *tdbname;
97 : TALLOC_CTX *mem_ctx;
98 : struct loadparm_context *lp_ctx;
99 : DATA_BLOB blob;
100 : PyObject *ret_obj;
101 : int ret;
102 : ssize_t xattr_size;
103 7 : struct db_context *eadb = NULL;
104 : struct file_id id;
105 : struct stat sbuf;
106 :
107 7 : if (!PyArg_ParseTuple(args, "sss", &tdbname, &filename, &attribute))
108 0 : return NULL;
109 :
110 7 : mem_ctx = talloc_new(NULL);
111 :
112 7 : lp_ctx = py_default_loadparm_context(mem_ctx);
113 7 : eadb = db_open_tdb(mem_ctx, tdbname, 50000,
114 : lpcfg_tdb_flags(lp_ctx, TDB_DEFAULT),
115 : O_RDWR|O_CREAT, 0600, DBWRAP_LOCK_ORDER_2,
116 : DBWRAP_FLAG_NONE);
117 :
118 7 : if (eadb == NULL) {
119 0 : PyErr_SetFromErrno(PyExc_IOError);
120 0 : talloc_free(mem_ctx);
121 0 : return NULL;
122 : }
123 :
124 7 : ret = stat(filename, &sbuf);
125 7 : if (ret < 0) {
126 0 : PyErr_SetFromErrno(PyExc_IOError);
127 0 : talloc_free(mem_ctx);
128 0 : return NULL;
129 : }
130 :
131 7 : ZERO_STRUCT(id);
132 7 : id.devid = sbuf.st_dev;
133 7 : id.inode = sbuf.st_ino;
134 :
135 7 : xattr_size = xattr_tdb_getattr(eadb, mem_ctx, &id, attribute, &blob);
136 7 : if (xattr_size < 0) {
137 0 : PyErr_SetFromErrno(PyExc_TypeError);
138 0 : talloc_free(mem_ctx);
139 0 : return NULL;
140 : }
141 7 : ret_obj = Py_BuildValue(PYARG_BYTES_LEN, blob.data, xattr_size);
142 7 : talloc_free(mem_ctx);
143 7 : return ret_obj;
144 : }
145 :
146 : static PyMethodDef py_xattr_methods[] = {
147 : { "wrap_getxattr", (PyCFunction)py_wrap_getxattr, METH_VARARGS,
148 : "wrap_getxattr(filename,attribute) -> blob\n"
149 : "Retrieve given attribute on the given file." },
150 : { "wrap_setxattr", (PyCFunction)py_wrap_setxattr, METH_VARARGS,
151 : "wrap_setxattr(filename,attribute,value)\n"
152 : "Set the given attribute to the given value on the given file." },
153 : { "is_xattr_supported", (PyCFunction)py_is_xattr_supported, METH_NOARGS,
154 : "Return true if xattr are supported on this system\n"},
155 : {0}
156 : };
157 :
158 : static struct PyModuleDef moduledef = {
159 : PyModuleDef_HEAD_INIT,
160 : .m_name = "xattr_tdb",
161 : .m_doc = "Python bindings for xattr manipulation.",
162 : .m_size = -1,
163 : .m_methods = py_xattr_methods,
164 : };
165 :
166 1184 : MODULE_INIT_FUNC(xattr_tdb)
167 : {
168 : PyObject *m;
169 :
170 1184 : m = PyModule_Create(&moduledef);
171 :
172 1184 : if (m == NULL)
173 0 : return NULL;
174 :
175 1184 : return m;
176 : }
177 :
|