]> arthur.barton.de Git - netatalk.git/blob - libatalk/cnid/db3/cnid_db3_delete.c
big merge for db frontend and unicode.
[netatalk.git] / libatalk / cnid / db3 / cnid_db3_delete.c
1 /*
2  * $Id: cnid_db3_delete.c,v 1.1.4.1 2003-09-09 16:42:21 didg Exp $
3  *
4  * Copyright (c) 1999. Adrian Sun (asun@zoology.washington.edu)
5  * All Rights Reserved. See COPYRIGHT.
6  *
7  * cnid_delete: delete a CNID from the database 
8  */
9
10 #ifdef HAVE_CONFIG_H
11 #include "config.h"
12 #endif /* HAVE_CONFIG_H */
13
14 #ifdef CNID_BACKEND_DB3
15
16 #include <stdio.h>
17 #include <string.h>
18 #include <errno.h>
19 #include <atalk/logger.h>
20
21 #include <db.h>
22 #include <netatalk/endian.h>
23 #include <atalk/adouble.h>
24 #include "cnid_db3.h"
25
26 #include "cnid_db3_private.h"
27
28 int cnid_db3_delete(struct _cnid_db *cdb, const cnid_t id) {
29     CNID_private *db;
30     DBT key, data;
31     DB_TXN *tid;
32     int rc, found = 0;
33
34     if (!cdb || !(db = cdb->_private) || !id || (db->flags & CNIDFLAG_DB_RO)) {
35         return -1;
36     }
37
38     memset(&key, 0, sizeof(key));
39     memset(&data, 0, sizeof(data));
40
41     /* Get from ain CNID database. */
42     key.data = (cnid_t *)&id;
43     key.size = sizeof(id);
44     while (!found) {
45         rc = db->db_cnid->get(db->db_cnid, NULL, &key, &data, 0);
46         switch (rc) {
47         case 0:
48             found = 1;
49             break;
50         case DB_LOCK_DEADLOCK:
51             break;
52         case DB_NOTFOUND:
53 #ifdef DEBUG
54             LOG(log_info, logtype_default, "cnid_delete: CNID %u not in database",
55                 ntohl(id));
56 #endif
57             return 0;
58         default:
59             LOG(log_error, logtype_default, "cnid_delete: Unable to delete entry: %s",
60                 db_strerror(rc));
61             return rc;
62         }
63     }
64
65 retry:
66     if ((rc = db3_txn_begin(db->dbenv, NULL, &tid, 0)) != 0) {
67         LOG(log_error, logtype_default, "cnid_delete: Failed to begin transaction: %s",
68             db_strerror(rc));
69         return rc;
70     }
71
72     /* Now delete from the main CNID database. */
73     key.data = (cnid_t *)&id;
74     key.size = sizeof(id);
75     if ((rc = db->db_cnid->del(db->db_cnid, tid, &key, 0))) {
76         int ret;
77         if ((ret = db3_txn_abort(tid)) != 0) {
78             LOG(log_error, logtype_default, "cnid_delete: txn_abort: %s", db_strerror(ret));
79             return ret;
80         }
81         switch (rc) {
82         case DB_LOCK_DEADLOCK:
83             goto retry;
84         default:
85             goto abort_err;
86         }
87     }
88
89     /* Now delete from dev/ino database. */
90     key.data = data.data;
91     key.size = CNID_DEVINO_LEN;
92     if ((rc = db->db_devino->del(db->db_devino, tid, &key, 0))) {
93         switch (rc) {
94         case DB_LOCK_DEADLOCK:
95             if ((rc = db3_txn_abort(tid)) != 0) {
96                 LOG(log_error, logtype_default, "cnid_delete: txn_abort: %s",
97                     db_strerror(rc));
98                 return rc;
99             }
100             goto retry;
101         case DB_NOTFOUND:
102             /* Quietly fall through if the entry isn't found. */
103             break;
104         default:
105             if ((rc = db3_txn_abort(tid)) != 0) {
106                 LOG(log_error, logtype_default, "cnid_delete: txn_abort: %s",
107                     db_strerror(rc));
108                 return rc;
109             }
110             goto abort_err;
111         }
112     }
113
114     /* Get data from the did/name database.
115      * TODO Also handle did/macname, did/shortname, and did/longname. */
116     key.data = (char *)data.data + CNID_DEVINO_LEN;
117     key.size = data.size - CNID_DEVINO_LEN;
118     if ((rc = db->db_didname->del(db->db_didname, tid, &key, 0))) {
119         switch (rc) {
120         case DB_LOCK_DEADLOCK:
121             if ((rc = db3_txn_abort(tid)) != 0) {
122                 LOG(log_error, logtype_default, "cnid_delete: txn_abort: %s",
123                     db_strerror(rc));
124                 return rc;
125             }
126             goto retry;
127         case DB_NOTFOUND:
128             break;
129         default:
130             if ((rc = db3_txn_abort(tid)) != 0) {
131                 LOG(log_error, logtype_default, "cnid_delete: txn_abort: %s",
132                     db_strerror(rc));
133                 return rc;
134             }
135             goto abort_err;
136         }
137     }
138
139 #ifdef DEBUG
140     LOG(log_info, logtype_default, "cnid_delete: Deleting CNID %u", ntohl(id));
141 #endif
142     if ((rc = db3_txn_commit(tid, 0)) != 0) {
143         LOG(log_error, logtype_default, "cnid_delete: Failed to commit transaction: %s",
144             db_strerror(rc));
145         return rc;
146     }
147     return 0;
148
149 abort_err:
150     LOG(log_error, logtype_default, "cnid_delete: Unable to delete CNID %u: %s",
151         ntohl(id), db_strerror(rc));
152     return rc;
153 }
154
155 #endif /* CNID_BACKEND_DB3 */