ctdb-daemon: Do not allow database detach if AllowClientDBAttach=1
[Samba.git] / source3 / lib / conn_tdb.c
blobbf66d7d7b66c5251e6715eb3e4aa3c95bf01b94e
1 /*
2 Unix SMB/CIFS implementation.
3 Low-level connections.tdb access functions
4 Copyright (C) Volker Lendecke 2007
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include "includes.h"
21 #include "system/filesys.h"
22 #include "smbd/globals.h"
23 #include "dbwrap/dbwrap.h"
24 #include "dbwrap/dbwrap_open.h"
25 #include "dbwrap/dbwrap_rbt.h"
26 #include "messages.h"
27 #include "lib/conn_tdb.h"
28 #include "util_tdb.h"
30 struct connections_forall_state {
31 struct db_context *session_by_pid;
32 int (*fn)(const struct connections_key *key,
33 const struct connections_data *data,
34 void *private_data);
35 void *private_data;
36 int count;
39 struct connections_forall_session {
40 uid_t uid;
41 gid_t gid;
42 fstring machine;
43 fstring addr;
46 static int collect_sessions_fn(struct smbXsrv_session_global0 *global,
47 void *connections_forall_state)
49 NTSTATUS status;
50 struct connections_forall_state *state =
51 (struct connections_forall_state*)connections_forall_state;
53 uint32_t id = global->session_global_id;
54 struct connections_forall_session sess;
56 if (global->auth_session_info == NULL) {
57 sess.uid = -1;
58 sess.gid = -1;
59 } else {
60 sess.uid = global->auth_session_info->unix_token->uid;
61 sess.gid = global->auth_session_info->unix_token->gid;
63 fstrcpy(sess.machine, global->channels[0].remote_name);
64 fstrcpy(sess.addr, global->channels[0].remote_address);
66 status = dbwrap_store(state->session_by_pid,
67 make_tdb_data((void*)&id, sizeof(id)),
68 make_tdb_data((void*)&sess, sizeof(sess)),
69 TDB_INSERT);
70 if (!NT_STATUS_IS_OK(status)) {
71 DEBUG(0, ("Failed to store record: %s\n", nt_errstr(status)));
73 return 0;
76 static int traverse_tcon_fn(struct smbXsrv_tcon_global0 *global,
77 void *connections_forall_state)
79 NTSTATUS status;
80 struct connections_forall_state *state =
81 (struct connections_forall_state*)connections_forall_state;
83 struct connections_key key;
84 struct connections_data data;
86 uint32_t sess_id = global->session_global_id;
87 struct connections_forall_session sess = {
88 .uid = -1,
89 .gid = -1,
92 TDB_DATA val = tdb_null;
95 * Note: that share_name is defined as array without a pointer.
96 * that's why it's always a valid pointer here.
98 if (strlen(global->share_name) == 0) {
100 * when a smbXsrv_tcon is created it's created
101 * with emtpy share_name first in order to allocate
102 * an id, before filling in the details.
104 return 0;
107 status = dbwrap_fetch(state->session_by_pid, state,
108 make_tdb_data((void*)&sess_id, sizeof(sess_id)),
109 &val);
110 if (NT_STATUS_IS_OK(status)) {
111 memcpy((uint8_t *)&sess, val.dptr, val.dsize);
114 ZERO_STRUCT(key);
115 ZERO_STRUCT(data);
117 key.pid = data.pid = global->server_id;
118 key.cnum = data.cnum = global->tcon_global_id;
119 fstrcpy(key.name, global->share_name);
120 fstrcpy(data.servicename, global->share_name);
121 data.uid = sess.uid;
122 data.gid = sess.gid;
123 fstrcpy(data.addr, sess.addr);
124 fstrcpy(data.machine, sess.machine);
125 data.start = nt_time_to_unix(global->creation_time);
127 state->count++;
129 return state->fn(&key, &data, state->private_data);
132 int connections_forall_read(int (*fn)(const struct connections_key *key,
133 const struct connections_data *data,
134 void *private_data),
135 void *private_data)
137 TALLOC_CTX *frame = talloc_stackframe();
138 struct connections_forall_state *state =
139 talloc_zero(talloc_tos(), struct connections_forall_state);
140 NTSTATUS status;
141 int ret = -1;
143 state->session_by_pid = db_open_rbt(state);
144 state->fn = fn;
145 state->private_data = private_data;
146 status = smbXsrv_session_global_traverse(collect_sessions_fn, state);
147 if (!NT_STATUS_IS_OK(status)) {
148 DEBUG(0, ("Failed to traverse sessions: %s\n",
149 nt_errstr(status)));
150 goto done;
153 status = smbXsrv_tcon_global_traverse(traverse_tcon_fn, state);
154 if (!NT_STATUS_IS_OK(status)) {
155 DEBUG(0, ("Failed to traverse tree connects: %s\n",
156 nt_errstr(status)));
157 goto done;
159 ret = state->count;
160 done:
161 talloc_free(frame);
162 return ret;