Line data Source code
1 : /*
2 : notification control module
3 :
4 : Copyright (C) Stefan Metzmacher 2015
5 :
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.
10 :
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.
15 :
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/>.
18 : */
19 :
20 :
21 : #include "includes.h"
22 : #include "ldb/include/ldb.h"
23 : #include "ldb/include/ldb_errors.h"
24 : #include "ldb/include/ldb_module.h"
25 : #include "dsdb/samdb/samdb.h"
26 : #include "dsdb/samdb/ldb_modules/util.h"
27 :
28 : struct dsdb_notification_cookie {
29 : uint64_t known_usn;
30 : };
31 :
32 1543 : static int dsdb_notification_verify_tree(struct ldb_parse_tree *tree)
33 : {
34 : unsigned int i;
35 : int ret;
36 1543 : unsigned int num_ok = 0;
37 : /*
38 : * these attributes are present on every object
39 : * and windows accepts them.
40 : *
41 : * While [MS-ADTS] says only '(objectClass=*)'
42 : * would be allowed.
43 : */
44 : static const char * const attrs_ok[] = {
45 : "objectClass",
46 : "objectGUID",
47 : "distinguishedName",
48 : "name",
49 : NULL,
50 : };
51 :
52 1543 : switch (tree->operation) {
53 4 : case LDB_OP_AND:
54 8 : for (i = 0; i < tree->u.list.num_elements; i++) {
55 : /*
56 : * all elements need to be valid
57 : */
58 8 : ret = dsdb_notification_verify_tree(tree->u.list.elements[i]);
59 8 : if (ret != LDB_SUCCESS) {
60 4 : return ret;
61 : }
62 4 : num_ok++;
63 : }
64 0 : break;
65 5 : case LDB_OP_OR:
66 5 : for (i = 0; i < tree->u.list.num_elements; i++) {
67 : /*
68 : * at least one element needs to be valid
69 : */
70 5 : ret = dsdb_notification_verify_tree(tree->u.list.elements[i]);
71 5 : if (ret == LDB_SUCCESS) {
72 5 : num_ok++;
73 5 : break;
74 : }
75 : }
76 5 : break;
77 24 : case LDB_OP_NOT:
78 : case LDB_OP_EQUALITY:
79 : case LDB_OP_GREATER:
80 : case LDB_OP_LESS:
81 : case LDB_OP_APPROX:
82 : case LDB_OP_SUBSTRING:
83 : case LDB_OP_EXTENDED:
84 24 : break;
85 :
86 1510 : case LDB_OP_PRESENT:
87 1510 : ret = ldb_attr_in_list(attrs_ok, tree->u.present.attr);
88 1510 : if (ret == 1) {
89 33 : num_ok++;
90 : }
91 1510 : break;
92 : }
93 :
94 1539 : if (num_ok != 0) {
95 38 : return LDB_SUCCESS;
96 : }
97 :
98 1501 : return LDB_ERR_UNWILLING_TO_PERFORM;
99 : }
100 :
101 1530 : static int dsdb_notification_filter_search(struct ldb_module *module,
102 : struct ldb_request *req,
103 : struct ldb_control *control)
104 : {
105 1530 : struct ldb_context *ldb = ldb_module_get_ctx(module);
106 1530 : char *filter_usn = NULL;
107 1530 : struct ldb_parse_tree *down_tree = NULL;
108 1530 : struct ldb_request *down_req = NULL;
109 1530 : struct dsdb_notification_cookie *cookie = NULL;
110 : int ret;
111 :
112 1530 : if (req->op.search.tree == NULL) {
113 0 : return dsdb_module_werror(module, LDB_ERR_OTHER,
114 : WERR_DS_NOTIFY_FILTER_TOO_COMPLEX,
115 : "Search filter missing.");
116 : }
117 :
118 1530 : ret = dsdb_notification_verify_tree(req->op.search.tree);
119 1530 : if (ret != LDB_SUCCESS) {
120 1501 : return dsdb_module_werror(module, ret,
121 : WERR_DS_NOTIFY_FILTER_TOO_COMPLEX,
122 : "Search filter too complex.");
123 : }
124 :
125 : /*
126 : * For now we use a very simple design:
127 : *
128 : * - We don't do fully async ldb_requests,
129 : * the caller needs to retry periodically!
130 : * - The only useful caller is the LDAP server, which is a long
131 : * running task that can do periodic retries.
132 : * - We use a cookie in order to transfer state between the
133 : * retries.
134 : * - We just search the available new objects each time we're
135 : * called.
136 : *
137 : * As the only valid search filter is '(objectClass=*)' or
138 : * something similar that matches every object, we simply
139 : * replace it with (uSNChanged >= ) filter.
140 : * We could improve this later if required...
141 : */
142 :
143 : /*
144 : * The ldap_control_handler() decode_flag_request for
145 : * LDB_CONTROL_NOTIFICATION_OID. This makes sure
146 : * notification_control->data is NULL when comming from
147 : * the client.
148 : */
149 29 : if (control->data == NULL) {
150 27 : cookie = talloc_zero(control, struct dsdb_notification_cookie);
151 27 : if (cookie == NULL) {
152 0 : return ldb_module_oom(module);
153 : }
154 27 : control->data = (uint8_t *)cookie;
155 :
156 : /* mark the control as done */
157 27 : control->critical = 0;
158 : }
159 :
160 29 : cookie = talloc_get_type_abort(control->data,
161 : struct dsdb_notification_cookie);
162 :
163 29 : if (cookie->known_usn != 0) {
164 2 : filter_usn = talloc_asprintf(req, "%llu",
165 2 : (unsigned long long)(cookie->known_usn)+1);
166 2 : if (filter_usn == NULL) {
167 0 : return ldb_module_oom(module);
168 : }
169 : }
170 :
171 29 : ret = ldb_sequence_number(ldb, LDB_SEQ_HIGHEST_SEQ,
172 : &cookie->known_usn);
173 29 : if (ret != LDB_SUCCESS) {
174 0 : return ret;
175 : }
176 :
177 29 : if (filter_usn == NULL) {
178 : /*
179 : * It's the first time, let the caller comeback later
180 : * as we won't find any new objects.
181 : */
182 27 : return ldb_module_done(req, NULL, NULL, LDB_SUCCESS);
183 : }
184 :
185 2 : down_tree = talloc_zero(req, struct ldb_parse_tree);
186 2 : if (down_tree == NULL) {
187 0 : return ldb_module_oom(module);
188 : }
189 2 : down_tree->operation = LDB_OP_GREATER;
190 2 : down_tree->u.equality.attr = "uSNChanged";
191 2 : down_tree->u.equality.value = data_blob_string_const(filter_usn);
192 2 : (void)talloc_move(down_req, &filter_usn);
193 :
194 2 : ret = ldb_build_search_req_ex(&down_req, ldb, req,
195 : req->op.search.base,
196 : req->op.search.scope,
197 : down_tree,
198 : req->op.search.attrs,
199 : req->controls,
200 : req, dsdb_next_callback,
201 : req);
202 2 : LDB_REQ_SET_LOCATION(down_req);
203 2 : if (ret != LDB_SUCCESS) {
204 0 : return ret;
205 : }
206 :
207 : /* perform the search */
208 2 : return ldb_next_request(module, down_req);
209 : }
210 :
211 11997611 : static int dsdb_notification_search(struct ldb_module *module, struct ldb_request *req)
212 : {
213 11997611 : struct ldb_control *control = NULL;
214 :
215 11997611 : if (ldb_dn_is_special(req->op.search.base)) {
216 737905 : return ldb_next_request(module, req);
217 : }
218 :
219 : /*
220 : * check if there's an extended dn control
221 : */
222 11259706 : control = ldb_request_get_control(req, LDB_CONTROL_NOTIFICATION_OID);
223 11259706 : if (control == NULL) {
224 : /* not found go on */
225 11258176 : return ldb_next_request(module, req);
226 : }
227 :
228 1530 : return dsdb_notification_filter_search(module, req, control);
229 : }
230 :
231 108013 : static int dsdb_notification_init(struct ldb_module *module)
232 : {
233 : int ret;
234 :
235 108013 : ret = ldb_mod_register_control(module, LDB_CONTROL_NOTIFICATION_OID);
236 108013 : if (ret != LDB_SUCCESS) {
237 0 : struct ldb_context *ldb = ldb_module_get_ctx(module);
238 :
239 0 : ldb_debug(ldb, LDB_DEBUG_ERROR,
240 : "notification: Unable to register control with rootdse!\n");
241 0 : return ldb_module_operr(module);
242 : }
243 :
244 108013 : return ldb_next_init(module);
245 : }
246 :
247 : static const struct ldb_module_ops ldb_dsdb_notification_module_ops = {
248 : .name = "dsdb_notification",
249 : .search = dsdb_notification_search,
250 : .init_context = dsdb_notification_init,
251 : };
252 :
253 : /*
254 : initialise the module
255 : */
256 4310 : _PUBLIC_ int ldb_dsdb_notification_module_init(const char *version)
257 : {
258 : int ret;
259 4310 : LDB_MODULE_CHECK_VERSION(version);
260 4310 : ret = ldb_register_module(&ldb_dsdb_notification_module_ops);
261 4310 : return ret;
262 : }
|