Use py_resource module
[python/dscho.git] / Modules / dbmmodule.c
blob936fb45ab1dd0b9c3bf10fab6ae3c197133a7353
1 /***********************************************************
2 Copyright 1991-1995 by Stichting Mathematisch Centrum, Amsterdam,
3 The Netherlands.
5 All Rights Reserved
7 Permission to use, copy, modify, and distribute this software and its
8 documentation for any purpose and without fee is hereby granted,
9 provided that the above copyright notice appear in all copies and that
10 both that copyright notice and this permission notice appear in
11 supporting documentation, and that the names of Stichting Mathematisch
12 Centrum or CWI not be used in advertising or publicity pertaining to
13 distribution of the software without specific, written prior permission.
15 STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO
16 THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
17 FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE
18 FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
19 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
20 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
21 OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
23 ******************************************************************/
25 /* DBM module using dictionary interface */
28 #include "allobjects.h"
29 #include "modsupport.h"
31 #include <sys/types.h>
32 #include <sys/stat.h>
33 #include <fcntl.h>
34 #include <ndbm.h>
36 typedef struct {
37 OB_HEAD
38 int di_size; /* -1 means recompute */
39 DBM *di_dbm;
40 } dbmobject;
42 staticforward typeobject Dbmtype;
44 #define is_dbmobject(v) ((v)->ob_type == &Dbmtype)
46 static object *DbmError;
48 static object *
49 newdbmobject(file, flags, mode)
50 char *file;
51 int flags;
52 int mode;
54 dbmobject *dp;
56 dp = NEWOBJ(dbmobject, &Dbmtype);
57 if (dp == NULL)
58 return NULL;
59 dp->di_size = -1;
60 if ( (dp->di_dbm = dbm_open(file, flags, mode)) == 0 ) {
61 err_errno(DbmError);
62 DECREF(dp);
63 return NULL;
65 return (object *)dp;
68 /* Methods */
70 static void
71 dbm_dealloc(dp)
72 register dbmobject *dp;
74 if ( dp->di_dbm )
75 dbm_close(dp->di_dbm);
76 DEL(dp);
79 static int
80 dbm_length(dp)
81 dbmobject *dp;
83 if ( dp->di_size < 0 ) {
84 datum key;
85 int size;
87 size = 0;
88 for ( key=dbm_firstkey(dp->di_dbm); key.dptr;
89 key = dbm_nextkey(dp->di_dbm))
90 size++;
91 dp->di_size = size;
93 return dp->di_size;
96 static object *
97 dbm_subscript(dp, key)
98 dbmobject *dp;
99 register object *key;
101 object *v;
102 datum drec, krec;
104 if (!getargs(key, "s#", &krec.dptr, &krec.dsize) )
105 return NULL;
107 drec = dbm_fetch(dp->di_dbm, krec);
108 if ( drec.dptr == 0 ) {
109 err_setstr(KeyError, GETSTRINGVALUE((stringobject *)key));
110 return NULL;
112 if ( dbm_error(dp->di_dbm) ) {
113 dbm_clearerr(dp->di_dbm);
114 err_setstr(DbmError, "");
115 return NULL;
117 return newsizedstringobject(drec.dptr, drec.dsize);
120 static int
121 dbm_ass_sub(dp, v, w)
122 dbmobject *dp;
123 object *v, *w;
125 datum krec, drec;
127 if ( !getargs(v, "s#", &krec.dptr, &krec.dsize) ) {
128 err_setstr(TypeError, "dbm mappings have string indices only");
129 return -1;
131 dp->di_size = -1;
132 if (w == NULL) {
133 if ( dbm_delete(dp->di_dbm, krec) < 0 ) {
134 dbm_clearerr(dp->di_dbm);
135 err_setstr(KeyError, GETSTRINGVALUE((stringobject *)v));
136 return -1;
138 } else {
139 if ( !getargs(w, "s#", &drec.dptr, &drec.dsize) ) {
140 err_setstr(TypeError,
141 "dbm mappings have string elements only");
142 return -1;
144 if ( dbm_store(dp->di_dbm, krec, drec, DBM_REPLACE) < 0 ) {
145 dbm_clearerr(dp->di_dbm);
146 err_setstr(DbmError, "Cannot add item to database");
147 return -1;
150 if ( dbm_error(dp->di_dbm) ) {
151 dbm_clearerr(dp->di_dbm);
152 err_setstr(DbmError, "");
153 return -1;
155 return 0;
158 static mapping_methods dbm_as_mapping = {
159 (inquiry)dbm_length, /*mp_length*/
160 (binaryfunc)dbm_subscript, /*mp_subscript*/
161 (objobjargproc)dbm_ass_sub, /*mp_ass_subscript*/
164 static object *
165 dbm__close(dp, args)
166 register dbmobject *dp;
167 object *args;
169 if ( !getnoarg(args) )
170 return NULL;
171 if ( dp->di_dbm )
172 dbm_close(dp->di_dbm);
173 dp->di_dbm = NULL;
174 INCREF(None);
175 return None;
178 static object *
179 dbm_keys(dp, args)
180 register dbmobject *dp;
181 object *args;
183 register object *v, *item;
184 datum key;
185 int err;
187 if (!getnoarg(args))
188 return NULL;
189 v = newlistobject(0);
190 if (v == NULL)
191 return NULL;
192 for (key = dbm_firstkey(dp->di_dbm); key.dptr;
193 key = dbm_nextkey(dp->di_dbm)) {
194 item = newsizedstringobject(key.dptr, key.dsize);
195 if (item == NULL) {
196 DECREF(v);
197 return NULL;
199 err = addlistitem(v, item);
200 DECREF(item);
201 if (err != 0) {
202 DECREF(v);
203 return NULL;
206 return v;
209 static object *
210 dbm_has_key(dp, args)
211 register dbmobject *dp;
212 object *args;
214 datum key, val;
216 if (!getargs(args, "s#", &key.dptr, &key.dsize))
217 return NULL;
218 val = dbm_fetch(dp->di_dbm, key);
219 return newintobject(val.dptr != NULL);
222 static struct methodlist dbm_methods[] = {
223 {"close", (method)dbm__close},
224 {"keys", (method)dbm_keys},
225 {"has_key", (method)dbm_has_key},
226 {NULL, NULL} /* sentinel */
229 static object *
230 dbm_getattr(dp, name)
231 dbmobject *dp;
232 char *name;
234 return findmethod(dbm_methods, (object *)dp, name);
237 static typeobject Dbmtype = {
238 OB_HEAD_INIT(&Typetype)
240 "dbm",
241 sizeof(dbmobject),
243 (destructor)dbm_dealloc, /*tp_dealloc*/
244 0, /*tp_print*/
245 (getattrfunc)dbm_getattr, /*tp_getattr*/
246 0, /*tp_setattr*/
247 0, /*tp_compare*/
248 0, /*tp_repr*/
249 0, /*tp_as_number*/
250 0, /*tp_as_sequence*/
251 &dbm_as_mapping, /*tp_as_mapping*/
254 /* ----------------------------------------------------------------- */
256 static object *
257 dbmopen(self, args)
258 object *self;
259 object *args;
261 char *name;
262 char *flags = "r";
263 int iflags;
264 int mode = 0666;
266 if ( !newgetargs(args, "s|si", &name, &flags, &mode) )
267 return NULL;
268 if ( strcmp(flags, "r") == 0 )
269 iflags = O_RDONLY;
270 else if ( strcmp(flags, "w") == 0 )
271 iflags = O_RDWR;
272 else if ( strcmp(flags, "rw") == 0 ) /* B/W compat */
273 iflags = O_RDWR|O_CREAT;
274 else if ( strcmp(flags, "c") == 0 )
275 iflags = O_RDWR|O_CREAT;
276 else if ( strcmp(flags, "n") == 0 )
277 iflags = O_RDWR|O_CREAT|O_TRUNC;
278 else {
279 err_setstr(DbmError,
280 "Flags should be one of 'r', 'w', 'c' or 'n'");
281 return NULL;
283 return newdbmobject(name, iflags, mode);
286 static struct methodlist dbmmodule_methods[] = {
287 { "open", (method)dbmopen, 1 },
288 { 0, 0 },
291 void
292 initdbm() {
293 object *m, *d;
295 m = initmodule("dbm", dbmmodule_methods);
296 d = getmoduledict(m);
297 DbmError = newstringobject("dbm.error");
298 if ( DbmError == NULL || dictinsert(d, "error", DbmError) )
299 fatal("can't define dbm.error");