improve treatment of multi-line replies, ignore empty lines
[python/dscho.git] / Modules / dbmmodule.c
blob867461fc654c340abe957296a31daedb5ad9f3d2
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 0;
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 0;
112 if ( dbm_error(dp->di_dbm) ) {
113 dbm_clearerr(dp->di_dbm);
114 err_setstr(DbmError, "");
115 return 0;
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_keys(dp, args)
166 register dbmobject *dp;
167 object *args;
169 register object *v, *item;
170 datum key;
172 if (dp == NULL || !is_dbmobject(dp)) {
173 err_badcall();
174 return NULL;
176 if (!getnoarg(args))
177 return NULL;
178 v = newlistobject(0);
179 if (v == NULL)
180 return NULL;
181 for (key = dbm_firstkey(dp->di_dbm); key.dptr;
182 key = dbm_nextkey(dp->di_dbm) ) {
183 item = newsizedstringobject(key.dptr, key.dsize);
184 if ( item == 0 )
185 return NULL;
186 addlistitem(v, item);
188 return v;
192 static object *
193 dbm_has_key(dp, args)
194 register dbmobject *dp;
195 object *args;
197 datum key, val;
199 if (!getargs(args, "s#", &key.dptr, &key.dsize))
200 return NULL;
201 val = dbm_fetch(dp->di_dbm, key);
202 return newintobject(val.dptr != NULL);
205 static struct methodlist dbm_methods[] = {
206 {"keys", (method)dbm_keys},
207 {"has_key", (method)dbm_has_key},
208 {NULL, NULL} /* sentinel */
211 static object *
212 dbm_getattr(dp, name)
213 dbmobject *dp;
214 char *name;
216 return findmethod(dbm_methods, (object *)dp, name);
219 static typeobject Dbmtype = {
220 OB_HEAD_INIT(&Typetype)
222 "Dbm_dictionary",
223 sizeof(dbmobject),
225 (destructor)dbm_dealloc, /*tp_dealloc*/
226 0, /*tp_print*/
227 (getattrfunc)dbm_getattr, /*tp_getattr*/
228 0, /*tp_setattr*/
229 0, /*tp_compare*/
230 0, /*tp_repr*/
231 0, /*tp_as_number*/
232 0, /*tp_as_sequence*/
233 &dbm_as_mapping, /*tp_as_mapping*/
236 /* ----------------------------------------------------------------- */
238 static object *
239 dbmopen(self, args)
240 object *self;
241 object *args;
243 char *name, *flags;
244 int iflags, mode;
246 if ( !getargs(args, "(ssi)", &name, &flags, &mode) )
247 return 0;
248 if ( strcmp(flags, "r") == 0 )
249 iflags = O_RDONLY;
250 else if ( strcmp(flags, "w") == 0 )
251 iflags = O_WRONLY|O_CREAT;
252 else if ( strcmp(flags, "rw") == 0 )
253 iflags = O_RDWR|O_CREAT;
254 else {
255 err_setstr(DbmError,
256 "Flags should be one of 'r', 'w' or 'rw'");
257 return 0;
259 return newdbmobject(name, iflags, mode);
262 static struct methodlist dbmmodule_methods[] = {
263 { "open", (method)dbmopen },
264 { 0, 0 },
267 void
268 initdbm() {
269 object *m, *d;
271 m = initmodule("dbm", dbmmodule_methods);
272 d = getmoduledict(m);
273 DbmError = newstringobject("dbm.error");
274 if ( DbmError == NULL || dictinsert(d, "error", DbmError) )
275 fatal("can't define dbm.error");