Line data Source code
1 : /*
2 : Unix SMB/CIFS Implementation.
3 : DSDB replication service helper function for outgoing traffic
4 :
5 : Copyright (C) Stefan Metzmacher 2007
6 :
7 : This program is free software; you can redistribute it and/or modify
8 : it under the terms of the GNU General Public License as published by
9 : the Free Software Foundation; either version 3 of the License, or
10 : (at your option) any later version.
11 :
12 : This program is distributed in the hope that it will be useful,
13 : but WITHOUT ANY WARRANTY; without even the implied warranty of
14 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 : GNU General Public License for more details.
16 :
17 : You should have received a copy of the GNU General Public License
18 : along with this program. If not, see <http://www.gnu.org/licenses/>.
19 :
20 : */
21 :
22 : #include "includes.h"
23 : #include "dsdb/samdb/samdb.h"
24 : #include "auth/auth.h"
25 : #include "samba/service.h"
26 : #include "lib/events/events.h"
27 : #include "dsdb/repl/drepl_service.h"
28 : #include <ldb_errors.h>
29 : #include "../lib/util/dlinklist.h"
30 : #include "librpc/gen_ndr/ndr_misc.h"
31 : #include "librpc/gen_ndr/ndr_drsuapi.h"
32 : #include "librpc/gen_ndr/ndr_drsblobs.h"
33 : #include "libcli/composite/composite.h"
34 : #include "auth/gensec/gensec.h"
35 : #include "param/param.h"
36 : #include "../lib/util/tevent_ntstatus.h"
37 : #include "libcli/security/security.h"
38 :
39 : #undef DBGC_CLASS
40 : #define DBGC_CLASS DBGC_DRS_REPL
41 :
42 : struct dreplsrv_out_drsuapi_state {
43 : struct tevent_context *ev;
44 :
45 : struct dreplsrv_out_connection *conn;
46 :
47 : struct dreplsrv_drsuapi_connection *drsuapi;
48 :
49 : struct drsuapi_DsBindInfoCtr bind_info_ctr;
50 : struct drsuapi_DsBind bind_r;
51 : };
52 :
53 : static void dreplsrv_out_drsuapi_connect_done(struct composite_context *creq);
54 :
55 9457 : struct tevent_req *dreplsrv_out_drsuapi_send(TALLOC_CTX *mem_ctx,
56 : struct tevent_context *ev,
57 : struct dreplsrv_out_connection *conn)
58 : {
59 0 : struct tevent_req *req;
60 0 : struct dreplsrv_out_drsuapi_state *state;
61 0 : struct composite_context *creq;
62 :
63 9457 : req = tevent_req_create(mem_ctx, &state,
64 : struct dreplsrv_out_drsuapi_state);
65 9457 : if (req == NULL) {
66 0 : return NULL;
67 : }
68 :
69 9457 : state->ev = ev;
70 9457 : state->conn = conn;
71 9457 : state->drsuapi = conn->drsuapi;
72 :
73 9457 : if (state->drsuapi != NULL) {
74 4945 : struct dcerpc_binding_handle *b =
75 4945 : state->drsuapi->pipe->binding_handle;
76 4945 : bool is_connected = dcerpc_binding_handle_is_connected(b);
77 :
78 4945 : if (is_connected) {
79 4945 : tevent_req_done(req);
80 4945 : return tevent_req_post(req, ev);
81 : }
82 :
83 0 : TALLOC_FREE(conn->drsuapi);
84 : }
85 :
86 4512 : state->drsuapi = talloc_zero(state, struct dreplsrv_drsuapi_connection);
87 4512 : if (tevent_req_nomem(state->drsuapi, req)) {
88 0 : return tevent_req_post(req, ev);
89 : }
90 :
91 4512 : creq = dcerpc_pipe_connect_b_send(state, conn->binding, &ndr_table_drsuapi,
92 4512 : conn->service->system_session_info->credentials,
93 4512 : ev, conn->service->task->lp_ctx);
94 4512 : if (tevent_req_nomem(creq, req)) {
95 0 : return tevent_req_post(req, ev);
96 : }
97 4512 : composite_continue(NULL, creq, dreplsrv_out_drsuapi_connect_done, req);
98 :
99 4512 : return req;
100 : }
101 :
102 : static void dreplsrv_out_drsuapi_bind_done(struct tevent_req *subreq);
103 :
104 4511 : static void dreplsrv_out_drsuapi_connect_done(struct composite_context *creq)
105 : {
106 4511 : struct tevent_req *req = talloc_get_type(creq->async.private_data,
107 : struct tevent_req);
108 4511 : struct dreplsrv_out_drsuapi_state *state = tevent_req_data(req,
109 : struct dreplsrv_out_drsuapi_state);
110 0 : NTSTATUS status;
111 0 : struct tevent_req *subreq;
112 :
113 4511 : status = dcerpc_pipe_connect_b_recv(creq,
114 4511 : state->drsuapi,
115 4511 : &state->drsuapi->pipe);
116 4511 : if (tevent_req_nterror(req, status)) {
117 4494 : return;
118 : }
119 :
120 17 : state->drsuapi->drsuapi_handle = state->drsuapi->pipe->binding_handle;
121 :
122 17 : status = gensec_session_key(state->drsuapi->pipe->conn->security_state.generic_state,
123 17 : state->drsuapi,
124 17 : &state->drsuapi->gensec_skey);
125 17 : if (tevent_req_nterror(req, status)) {
126 0 : return;
127 : }
128 :
129 17 : state->bind_info_ctr.length = 28;
130 17 : state->bind_info_ctr.info.info28 = state->conn->service->bind_info28;
131 :
132 17 : state->bind_r.in.bind_guid = &state->conn->service->ntds_guid;
133 17 : state->bind_r.in.bind_info = &state->bind_info_ctr;
134 17 : state->bind_r.out.bind_handle = &state->drsuapi->bind_handle;
135 :
136 17 : subreq = dcerpc_drsuapi_DsBind_r_send(state,
137 : state->ev,
138 17 : state->drsuapi->drsuapi_handle,
139 : &state->bind_r);
140 17 : if (tevent_req_nomem(subreq, req)) {
141 0 : return;
142 : }
143 17 : tevent_req_set_callback(subreq, dreplsrv_out_drsuapi_bind_done, req);
144 : }
145 :
146 17 : static void dreplsrv_out_drsuapi_bind_done(struct tevent_req *subreq)
147 : {
148 17 : struct tevent_req *req = tevent_req_callback_data(subreq,
149 : struct tevent_req);
150 17 : struct dreplsrv_out_drsuapi_state *state = tevent_req_data(req,
151 : struct dreplsrv_out_drsuapi_state);
152 0 : NTSTATUS status;
153 :
154 17 : status = dcerpc_drsuapi_DsBind_r_recv(subreq, state);
155 17 : TALLOC_FREE(subreq);
156 17 : if (tevent_req_nterror(req, status)) {
157 0 : return;
158 : }
159 :
160 17 : if (!W_ERROR_IS_OK(state->bind_r.out.result)) {
161 0 : status = werror_to_ntstatus(state->bind_r.out.result);
162 0 : tevent_req_nterror(req, status);
163 0 : return;
164 : }
165 :
166 17 : ZERO_STRUCT(state->drsuapi->remote_info28);
167 17 : if (state->bind_r.out.bind_info) {
168 0 : struct drsuapi_DsBindInfo28 *info28;
169 17 : info28 = &state->drsuapi->remote_info28;
170 :
171 17 : switch (state->bind_r.out.bind_info->length) {
172 0 : case 24: {
173 0 : struct drsuapi_DsBindInfo24 *info24;
174 0 : info24 = &state->bind_r.out.bind_info->info.info24;
175 :
176 0 : info28->supported_extensions = info24->supported_extensions;
177 0 : info28->site_guid = info24->site_guid;
178 0 : info28->pid = info24->pid;
179 0 : info28->repl_epoch = 0;
180 0 : break;
181 : }
182 17 : case 28: {
183 17 : *info28 = state->bind_r.out.bind_info->info.info28;
184 17 : break;
185 : }
186 0 : case 32: {
187 0 : struct drsuapi_DsBindInfo32 *info32;
188 0 : info32 = &state->bind_r.out.bind_info->info.info32;
189 :
190 0 : info28->supported_extensions = info32->supported_extensions;
191 0 : info28->site_guid = info32->site_guid;
192 0 : info28->pid = info32->pid;
193 0 : info28->repl_epoch = info32->repl_epoch;
194 0 : break;
195 : }
196 0 : case 48: {
197 0 : struct drsuapi_DsBindInfo48 *info48;
198 0 : info48 = &state->bind_r.out.bind_info->info.info48;
199 :
200 0 : info28->supported_extensions = info48->supported_extensions;
201 0 : info28->site_guid = info48->site_guid;
202 0 : info28->pid = info48->pid;
203 0 : info28->repl_epoch = info48->repl_epoch;
204 0 : break;
205 : }
206 0 : case 52: {
207 0 : struct drsuapi_DsBindInfo52 *info52;
208 0 : info52 = &state->bind_r.out.bind_info->info.info52;
209 :
210 0 : info28->supported_extensions = info52->supported_extensions;
211 0 : info28->site_guid = info52->site_guid;
212 0 : info28->pid = info52->pid;
213 0 : info28->repl_epoch = info52->repl_epoch;
214 0 : break;
215 : }
216 0 : default:
217 0 : DEBUG(1, ("Warning: invalid info length in bind info: %d\n",
218 : state->bind_r.out.bind_info->length));
219 0 : break;
220 : }
221 : }
222 :
223 17 : tevent_req_done(req);
224 : }
225 :
226 9456 : NTSTATUS dreplsrv_out_drsuapi_recv(struct tevent_req *req)
227 : {
228 9456 : struct dreplsrv_out_drsuapi_state *state = tevent_req_data(req,
229 : struct dreplsrv_out_drsuapi_state);
230 0 : NTSTATUS status;
231 :
232 9456 : if (tevent_req_is_nterror(req, &status)) {
233 4494 : tevent_req_received(req);
234 4494 : return status;
235 : }
236 :
237 4962 : state->conn->drsuapi = talloc_move(state->conn, &state->drsuapi);
238 :
239 4962 : tevent_req_received(req);
240 4962 : return NT_STATUS_OK;
241 : }
242 :
243 : struct dreplsrv_op_pull_source_schema_cycle {
244 : struct repsFromTo1 repsFrom1;
245 : size_t object_count;
246 : struct drsuapi_DsReplicaObjectListItemEx *first_object;
247 : struct drsuapi_DsReplicaObjectListItemEx *last_object;
248 : uint32_t linked_attributes_count;
249 : struct drsuapi_DsReplicaLinkedAttribute *linked_attributes;
250 : };
251 :
252 : struct dreplsrv_op_pull_source_state {
253 : struct tevent_context *ev;
254 : struct dreplsrv_out_operation *op;
255 : void *ndr_struct_ptr;
256 : /*
257 : * Used when we have to re-try with a different NC, eg for
258 : * EXOP retry or to get a current schema first
259 : */
260 : struct dreplsrv_partition_source_dsa *source_dsa_retry;
261 : enum drsuapi_DsExtendedOperation extended_op_retry;
262 : bool retry_started;
263 : struct dreplsrv_op_pull_source_schema_cycle *schema_cycle;
264 : };
265 :
266 : static void dreplsrv_op_pull_source_connect_done(struct tevent_req *subreq);
267 :
268 3846 : struct tevent_req *dreplsrv_op_pull_source_send(TALLOC_CTX *mem_ctx,
269 : struct tevent_context *ev,
270 : struct dreplsrv_out_operation *op)
271 : {
272 0 : struct tevent_req *req;
273 0 : struct dreplsrv_op_pull_source_state *state;
274 0 : struct tevent_req *subreq;
275 :
276 3846 : req = tevent_req_create(mem_ctx, &state,
277 : struct dreplsrv_op_pull_source_state);
278 3846 : if (req == NULL) {
279 0 : return NULL;
280 : }
281 3846 : state->ev = ev;
282 3846 : state->op = op;
283 :
284 3846 : subreq = dreplsrv_out_drsuapi_send(state, ev, op->source_dsa->conn);
285 3846 : if (tevent_req_nomem(subreq, req)) {
286 0 : return tevent_req_post(req, ev);
287 : }
288 3846 : tevent_req_set_callback(subreq, dreplsrv_op_pull_source_connect_done, req);
289 :
290 3846 : return req;
291 : }
292 :
293 3835 : static bool dreplsrv_op_pull_source_detect_schema_cycle(struct tevent_req *req)
294 : {
295 0 : struct dreplsrv_op_pull_source_state *state =
296 3835 : tevent_req_data(req,
297 : struct dreplsrv_op_pull_source_state);
298 3835 : bool is_schema = false;
299 :
300 3835 : if (state->op->extended_op == DRSUAPI_EXOP_NONE) {
301 1962 : struct dreplsrv_out_operation *op = state->op;
302 1962 : struct dreplsrv_service *service = op->service;
303 1962 : struct ldb_dn *schema_dn = ldb_get_schema_basedn(service->samdb);
304 1962 : struct dreplsrv_partition *partition = op->source_dsa->partition;
305 :
306 1962 : is_schema = ldb_dn_compare(partition->dn, schema_dn) == 0;
307 : }
308 :
309 3835 : if (is_schema) {
310 0 : struct dreplsrv_op_pull_source_schema_cycle *sc;
311 :
312 121 : sc = talloc_zero(state,
313 : struct dreplsrv_op_pull_source_schema_cycle);
314 121 : if (tevent_req_nomem(sc, req)) {
315 0 : return false;
316 : }
317 121 : sc->repsFrom1 = *state->op->source_dsa->repsFrom1;
318 :
319 121 : state->schema_cycle = sc;
320 : }
321 :
322 3835 : return true;
323 : }
324 :
325 : static void dreplsrv_op_pull_source_get_changes_trigger(struct tevent_req *req);
326 :
327 3846 : static void dreplsrv_op_pull_source_connect_done(struct tevent_req *subreq)
328 : {
329 3846 : struct tevent_req *req = tevent_req_callback_data(subreq,
330 : struct tevent_req);
331 0 : NTSTATUS status;
332 0 : bool ok;
333 :
334 3846 : status = dreplsrv_out_drsuapi_recv(subreq);
335 3846 : TALLOC_FREE(subreq);
336 3846 : if (tevent_req_nterror(req, status)) {
337 20 : return;
338 : }
339 :
340 3826 : ok = dreplsrv_op_pull_source_detect_schema_cycle(req);
341 3826 : if (!ok) {
342 0 : return;
343 : }
344 :
345 3826 : dreplsrv_op_pull_source_get_changes_trigger(req);
346 : }
347 :
348 : static void dreplsrv_op_pull_source_get_changes_done(struct tevent_req *subreq);
349 :
350 : /*
351 : get a RODC partial attribute set for a replication call
352 : */
353 2391 : static NTSTATUS dreplsrv_get_rodc_partial_attribute_set(struct dreplsrv_service *service,
354 : TALLOC_CTX *mem_ctx,
355 : struct drsuapi_DsPartialAttributeSet **_pas,
356 : struct drsuapi_DsReplicaOIDMapping_Ctr **pfm,
357 : bool for_schema)
358 : {
359 0 : struct drsuapi_DsPartialAttributeSet *pas;
360 0 : struct dsdb_schema *schema;
361 0 : uint32_t i;
362 :
363 2391 : pas = talloc_zero(mem_ctx, struct drsuapi_DsPartialAttributeSet);
364 2391 : NT_STATUS_HAVE_NO_MEMORY(pas);
365 :
366 2391 : schema = dsdb_get_schema(service->samdb, NULL);
367 :
368 2391 : pas->version = 1;
369 2391 : pas->attids = talloc_array(pas, enum drsuapi_DsAttributeId, schema->num_attributes);
370 2391 : if (pas->attids == NULL) {
371 0 : TALLOC_FREE(pas);
372 0 : return NT_STATUS_NO_MEMORY;
373 : }
374 :
375 3147133 : for (i=0; i<schema->num_attributes; i++) {
376 0 : struct dsdb_attribute *a;
377 3144742 : a = schema->attributes_by_attributeID_id[i];
378 3144742 : if (a->systemFlags & (DS_FLAG_ATTR_NOT_REPLICATED | DS_FLAG_ATTR_IS_CONSTRUCTED)) {
379 289311 : continue;
380 : }
381 2855431 : if (a->searchFlags & SEARCH_FLAG_RODC_ATTRIBUTE) {
382 14346 : continue;
383 : }
384 2841085 : pas->attids[pas->num_attids] = dsdb_attribute_get_attid(a, for_schema);
385 2841085 : pas->num_attids++;
386 : }
387 :
388 2391 : pas->attids = talloc_realloc(pas, pas->attids, enum drsuapi_DsAttributeId, pas->num_attids);
389 2391 : if (pas->attids == NULL) {
390 0 : TALLOC_FREE(pas);
391 0 : return NT_STATUS_NO_MEMORY;
392 : }
393 :
394 2391 : *_pas = pas;
395 :
396 2391 : if (pfm != NULL) {
397 2391 : dsdb_get_oid_mappings_drsuapi(schema, true, mem_ctx, pfm);
398 : }
399 :
400 2391 : return NT_STATUS_OK;
401 : }
402 :
403 :
404 : /*
405 : get a GC partial attribute set for a replication call
406 : */
407 0 : static NTSTATUS dreplsrv_get_gc_partial_attribute_set(struct dreplsrv_service *service,
408 : TALLOC_CTX *mem_ctx,
409 : struct drsuapi_DsPartialAttributeSet **_pas,
410 : struct drsuapi_DsReplicaOIDMapping_Ctr **pfm)
411 : {
412 0 : struct drsuapi_DsPartialAttributeSet *pas;
413 0 : struct dsdb_schema *schema;
414 0 : uint32_t i;
415 :
416 0 : pas = talloc_zero(mem_ctx, struct drsuapi_DsPartialAttributeSet);
417 0 : NT_STATUS_HAVE_NO_MEMORY(pas);
418 :
419 0 : schema = dsdb_get_schema(service->samdb, NULL);
420 :
421 0 : pas->version = 1;
422 0 : pas->attids = talloc_array(pas, enum drsuapi_DsAttributeId, schema->num_attributes);
423 0 : if (pas->attids == NULL) {
424 0 : TALLOC_FREE(pas);
425 0 : return NT_STATUS_NO_MEMORY;
426 : }
427 :
428 0 : for (i=0; i<schema->num_attributes; i++) {
429 0 : struct dsdb_attribute *a;
430 0 : a = schema->attributes_by_attributeID_id[i];
431 0 : if (a->isMemberOfPartialAttributeSet) {
432 0 : pas->attids[pas->num_attids] = dsdb_attribute_get_attid(a, false);
433 0 : pas->num_attids++;
434 : }
435 : }
436 :
437 0 : pas->attids = talloc_realloc(pas, pas->attids, enum drsuapi_DsAttributeId, pas->num_attids);
438 0 : if (pas->attids == NULL) {
439 0 : TALLOC_FREE(pas);
440 0 : return NT_STATUS_NO_MEMORY;
441 : }
442 :
443 0 : *_pas = pas;
444 :
445 0 : if (pfm != NULL) {
446 0 : dsdb_get_oid_mappings_drsuapi(schema, true, mem_ctx, pfm);
447 : }
448 :
449 0 : return NT_STATUS_OK;
450 : }
451 :
452 : /*
453 : convert from one udv format to the other
454 : */
455 2746 : static WERROR udv_convert(TALLOC_CTX *mem_ctx,
456 : const struct replUpToDateVectorCtr2 *udv,
457 : struct drsuapi_DsReplicaCursorCtrEx *udv_ex)
458 : {
459 0 : uint32_t i;
460 :
461 2746 : udv_ex->version = 2;
462 2746 : udv_ex->reserved1 = 0;
463 2746 : udv_ex->reserved2 = 0;
464 2746 : udv_ex->count = udv->count;
465 2746 : udv_ex->cursors = talloc_array(mem_ctx, struct drsuapi_DsReplicaCursor, udv->count);
466 2746 : W_ERROR_HAVE_NO_MEMORY(udv_ex->cursors);
467 :
468 10027 : for (i=0; i<udv->count; i++) {
469 7281 : udv_ex->cursors[i].source_dsa_invocation_id = udv->cursors[i].source_dsa_invocation_id;
470 7281 : udv_ex->cursors[i].highest_usn = udv->cursors[i].highest_usn;
471 : }
472 :
473 2746 : return WERR_OK;
474 : }
475 :
476 :
477 4278 : static void dreplsrv_op_pull_source_get_changes_trigger(struct tevent_req *req)
478 : {
479 4278 : struct dreplsrv_op_pull_source_state *state = tevent_req_data(req,
480 : struct dreplsrv_op_pull_source_state);
481 4278 : const struct repsFromTo1 *rf1 = state->op->source_dsa->repsFrom1;
482 4278 : struct dreplsrv_service *service = state->op->service;
483 4278 : struct dreplsrv_partition *partition = state->op->source_dsa->partition;
484 4278 : struct dreplsrv_drsuapi_connection *drsuapi = state->op->source_dsa->conn->drsuapi;
485 0 : struct drsuapi_DsGetNCChanges *r;
486 0 : struct drsuapi_DsReplicaCursorCtrEx *uptodateness_vector;
487 0 : struct tevent_req *subreq;
488 4278 : struct drsuapi_DsPartialAttributeSet *pas = NULL;
489 0 : NTSTATUS status;
490 0 : uint32_t replica_flags;
491 0 : struct drsuapi_DsReplicaHighWaterMark highwatermark;
492 4278 : struct drsuapi_DsReplicaOIDMapping_Ctr *mappings = NULL;
493 4278 : bool is_schema = false;
494 :
495 4278 : if (state->schema_cycle != NULL) {
496 127 : is_schema = true;
497 127 : rf1 = &state->schema_cycle->repsFrom1;
498 : }
499 :
500 4278 : r = talloc(state, struct drsuapi_DsGetNCChanges);
501 4278 : if (tevent_req_nomem(r, req)) {
502 0 : return;
503 : }
504 :
505 4278 : r->out.level_out = talloc(r, uint32_t);
506 4278 : if (tevent_req_nomem(r->out.level_out, req)) {
507 0 : return;
508 : }
509 4278 : r->in.req = talloc(r, union drsuapi_DsGetNCChangesRequest);
510 4278 : if (tevent_req_nomem(r->in.req, req)) {
511 0 : return;
512 : }
513 4278 : r->out.ctr = talloc(r, union drsuapi_DsGetNCChangesCtr);
514 4278 : if (tevent_req_nomem(r->out.ctr, req)) {
515 0 : return;
516 : }
517 :
518 4278 : if (partition->uptodatevector.count != 0 &&
519 4278 : partition->uptodatevector_ex.count == 0) {
520 0 : WERROR werr;
521 2746 : werr = udv_convert(partition, &partition->uptodatevector, &partition->uptodatevector_ex);
522 2746 : if (!W_ERROR_IS_OK(werr)) {
523 0 : DEBUG(0,(__location__ ": Failed to convert UDV for %s : %s\n",
524 : ldb_dn_get_linearized(partition->dn), win_errstr(werr)));
525 0 : tevent_req_nterror(req, werror_to_ntstatus(werr));
526 0 : return;
527 : }
528 : }
529 :
530 4278 : if (partition->uptodatevector_ex.count == 0) {
531 0 : uptodateness_vector = NULL;
532 : } else {
533 4278 : uptodateness_vector = &partition->uptodatevector_ex;
534 : }
535 :
536 4278 : replica_flags = rf1->replica_flags;
537 4278 : highwatermark = rf1->highwatermark;
538 :
539 4278 : if (state->op->options & DRSUAPI_DRS_GET_ANC) {
540 35 : replica_flags |= DRSUAPI_DRS_GET_ANC;
541 : }
542 :
543 4278 : if (state->op->options & DRSUAPI_DRS_SYNC_FORCED) {
544 1105 : replica_flags |= DRSUAPI_DRS_SYNC_FORCED;
545 : }
546 :
547 4278 : if (partition->partial_replica) {
548 0 : status = dreplsrv_get_gc_partial_attribute_set(service, r,
549 : &pas,
550 : &mappings);
551 0 : if (!NT_STATUS_IS_OK(status)) {
552 0 : DEBUG(0,(__location__ ": Failed to construct GC partial attribute set : %s\n", nt_errstr(status)));
553 0 : tevent_req_nterror(req, status);
554 0 : return;
555 : }
556 0 : replica_flags &= ~DRSUAPI_DRS_WRIT_REP;
557 4278 : } else if (partition->rodc_replica || state->op->extended_op == DRSUAPI_EXOP_REPL_SECRET) {
558 2391 : status = dreplsrv_get_rodc_partial_attribute_set(service, r,
559 : &pas,
560 : &mappings,
561 : is_schema);
562 2391 : if (!NT_STATUS_IS_OK(status)) {
563 0 : DEBUG(0,(__location__ ": Failed to construct RODC partial attribute set : %s\n", nt_errstr(status)));
564 0 : tevent_req_nterror(req, status);
565 0 : return;
566 : }
567 2391 : replica_flags &= ~DRSUAPI_DRS_WRIT_REP;
568 2391 : if (state->op->extended_op == DRSUAPI_EXOP_REPL_SECRET) {
569 1849 : replica_flags &= ~DRSUAPI_DRS_SPECIAL_SECRET_PROCESSING;
570 : } else {
571 542 : replica_flags |= DRSUAPI_DRS_SPECIAL_SECRET_PROCESSING;
572 : }
573 :
574 : /*
575 : * As per MS-DRSR:
576 : *
577 : * 4.1.10.4
578 : * Client Behavior When Sending the IDL_DRSGetNCChanges Request
579 : *
580 : * 4.1.10.4.1
581 : * ReplicateNCRequestMsg
582 : */
583 2391 : replica_flags |= DRSUAPI_DRS_GET_ALL_GROUP_MEMBERSHIP;
584 : } else {
585 1887 : replica_flags |= DRSUAPI_DRS_GET_ALL_GROUP_MEMBERSHIP;
586 : }
587 :
588 4278 : if (state->op->extended_op != DRSUAPI_EXOP_NONE) {
589 : /*
590 : * If it's an exop never set the ADD_REF even if it's in
591 : * repsFrom flags.
592 : */
593 1873 : replica_flags &= ~DRSUAPI_DRS_ADD_REF;
594 : }
595 :
596 : /* is this a full resync of all objects? */
597 4278 : if (state->op->options & DRSUAPI_DRS_FULL_SYNC_NOW) {
598 10 : ZERO_STRUCT(highwatermark);
599 : /* clear the FULL_SYNC_NOW option for subsequent
600 : stages of the replication cycle */
601 10 : state->op->options &= ~DRSUAPI_DRS_FULL_SYNC_NOW;
602 10 : state->op->options |= DRSUAPI_DRS_FULL_SYNC_IN_PROGRESS;
603 10 : replica_flags |= DRSUAPI_DRS_NEVER_SYNCED;
604 : }
605 4278 : if (state->op->options & DRSUAPI_DRS_FULL_SYNC_IN_PROGRESS) {
606 161 : uptodateness_vector = NULL;
607 : }
608 :
609 4278 : r->in.bind_handle = &drsuapi->bind_handle;
610 :
611 4278 : if (drsuapi->remote_info28.supported_extensions & DRSUAPI_SUPPORTED_EXTENSION_GETCHGREQ_V10) {
612 4278 : r->in.level = 10;
613 4278 : r->in.req->req10.destination_dsa_guid = service->ntds_guid;
614 4278 : r->in.req->req10.source_dsa_invocation_id= rf1->source_dsa_invocation_id;
615 4278 : r->in.req->req10.naming_context = &partition->nc;
616 4278 : r->in.req->req10.highwatermark = highwatermark;
617 4278 : r->in.req->req10.uptodateness_vector = uptodateness_vector;
618 4278 : r->in.req->req10.replica_flags = replica_flags;
619 4278 : r->in.req->req10.max_object_count = 133;
620 4278 : r->in.req->req10.max_ndr_size = 1336811;
621 4278 : r->in.req->req10.extended_op = state->op->extended_op;
622 4278 : r->in.req->req10.fsmo_info = state->op->fsmo_info;
623 4278 : r->in.req->req10.partial_attribute_set = pas;
624 4278 : r->in.req->req10.partial_attribute_set_ex= NULL;
625 4278 : r->in.req->req10.mapping_ctr.num_mappings= mappings == NULL ? 0 : mappings->num_mappings;
626 4278 : r->in.req->req10.mapping_ctr.mappings = mappings == NULL ? NULL : mappings->mappings;
627 :
628 : /* the only difference to v8 is the more_flags */
629 4278 : r->in.req->req10.more_flags = state->op->more_flags;
630 :
631 0 : } else if (drsuapi->remote_info28.supported_extensions & DRSUAPI_SUPPORTED_EXTENSION_GETCHGREQ_V8) {
632 0 : r->in.level = 8;
633 0 : r->in.req->req8.destination_dsa_guid = service->ntds_guid;
634 0 : r->in.req->req8.source_dsa_invocation_id= rf1->source_dsa_invocation_id;
635 0 : r->in.req->req8.naming_context = &partition->nc;
636 0 : r->in.req->req8.highwatermark = highwatermark;
637 0 : r->in.req->req8.uptodateness_vector = uptodateness_vector;
638 0 : r->in.req->req8.replica_flags = replica_flags;
639 0 : r->in.req->req8.max_object_count = 133;
640 0 : r->in.req->req8.max_ndr_size = 1336811;
641 0 : r->in.req->req8.extended_op = state->op->extended_op;
642 0 : r->in.req->req8.fsmo_info = state->op->fsmo_info;
643 0 : r->in.req->req8.partial_attribute_set = pas;
644 0 : r->in.req->req8.partial_attribute_set_ex= NULL;
645 0 : r->in.req->req8.mapping_ctr.num_mappings= mappings == NULL ? 0 : mappings->num_mappings;
646 0 : r->in.req->req8.mapping_ctr.mappings = mappings == NULL ? NULL : mappings->mappings;
647 : } else {
648 0 : r->in.level = 5;
649 0 : r->in.req->req5.destination_dsa_guid = service->ntds_guid;
650 0 : r->in.req->req5.source_dsa_invocation_id= rf1->source_dsa_invocation_id;
651 0 : r->in.req->req5.naming_context = &partition->nc;
652 0 : r->in.req->req5.highwatermark = highwatermark;
653 0 : r->in.req->req5.uptodateness_vector = uptodateness_vector;
654 0 : r->in.req->req5.replica_flags = replica_flags;
655 0 : r->in.req->req5.max_object_count = 133;
656 0 : r->in.req->req5.max_ndr_size = 1336770;
657 0 : r->in.req->req5.extended_op = state->op->extended_op;
658 0 : r->in.req->req5.fsmo_info = state->op->fsmo_info;
659 : }
660 :
661 : #if 0
662 : NDR_PRINT_IN_DEBUG(drsuapi_DsGetNCChanges, r);
663 : #endif
664 :
665 4278 : state->ndr_struct_ptr = r;
666 4278 : subreq = dcerpc_drsuapi_DsGetNCChanges_r_send(state,
667 : state->ev,
668 : drsuapi->drsuapi_handle,
669 : r);
670 4278 : if (tevent_req_nomem(subreq, req)) {
671 0 : return;
672 : }
673 4278 : tevent_req_set_callback(subreq, dreplsrv_op_pull_source_get_changes_done, req);
674 : }
675 :
676 : static void dreplsrv_op_pull_source_apply_changes_trigger(struct tevent_req *req,
677 : struct drsuapi_DsGetNCChanges *r,
678 : uint32_t ctr_level,
679 : struct drsuapi_DsGetNCChangesCtr1 *ctr1,
680 : struct drsuapi_DsGetNCChangesCtr6 *ctr6);
681 :
682 4278 : static void dreplsrv_op_pull_source_get_changes_done(struct tevent_req *subreq)
683 : {
684 4278 : struct tevent_req *req = tevent_req_callback_data(subreq,
685 : struct tevent_req);
686 4278 : struct dreplsrv_op_pull_source_state *state = tevent_req_data(req,
687 : struct dreplsrv_op_pull_source_state);
688 0 : NTSTATUS status;
689 4278 : struct drsuapi_DsGetNCChanges *r = talloc_get_type(state->ndr_struct_ptr,
690 : struct drsuapi_DsGetNCChanges);
691 4278 : uint32_t ctr_level = 0;
692 4278 : struct drsuapi_DsGetNCChangesCtr1 *ctr1 = NULL;
693 4278 : struct drsuapi_DsGetNCChangesCtr6 *ctr6 = NULL;
694 4278 : enum drsuapi_DsExtendedError extended_ret = DRSUAPI_EXOP_ERR_NONE;
695 4278 : state->ndr_struct_ptr = NULL;
696 :
697 4278 : status = dcerpc_drsuapi_DsGetNCChanges_r_recv(subreq, r);
698 4278 : TALLOC_FREE(subreq);
699 4278 : if (tevent_req_nterror(req, status)) {
700 2032 : return;
701 : }
702 :
703 4278 : if (!W_ERROR_IS_OK(r->out.result)) {
704 2032 : status = werror_to_ntstatus(r->out.result);
705 2032 : tevent_req_nterror(req, status);
706 2032 : return;
707 : }
708 :
709 2246 : if (*r->out.level_out == 1) {
710 0 : ctr_level = 1;
711 0 : ctr1 = &r->out.ctr->ctr1;
712 2246 : } else if (*r->out.level_out == 2 &&
713 0 : r->out.ctr->ctr2.mszip1.ts) {
714 0 : ctr_level = 1;
715 0 : ctr1 = &r->out.ctr->ctr2.mszip1.ts->ctr1;
716 2246 : } else if (*r->out.level_out == 6) {
717 2246 : ctr_level = 6;
718 2246 : ctr6 = &r->out.ctr->ctr6;
719 0 : } else if (*r->out.level_out == 7 &&
720 0 : r->out.ctr->ctr7.level == 6 &&
721 0 : r->out.ctr->ctr7.type == DRSUAPI_COMPRESSION_TYPE_MSZIP &&
722 0 : r->out.ctr->ctr7.ctr.mszip6.ts) {
723 0 : ctr_level = 6;
724 0 : ctr6 = &r->out.ctr->ctr7.ctr.mszip6.ts->ctr6;
725 0 : } else if (*r->out.level_out == 7 &&
726 0 : r->out.ctr->ctr7.level == 6 &&
727 0 : r->out.ctr->ctr7.type == DRSUAPI_COMPRESSION_TYPE_WIN2K3_LZ77_DIRECT2 &&
728 0 : r->out.ctr->ctr7.ctr.xpress6.ts) {
729 0 : ctr_level = 6;
730 0 : ctr6 = &r->out.ctr->ctr7.ctr.xpress6.ts->ctr6;
731 : } else {
732 0 : status = werror_to_ntstatus(WERR_BAD_NET_RESP);
733 0 : tevent_req_nterror(req, status);
734 0 : return;
735 : }
736 :
737 2246 : if (!ctr1 && !ctr6) {
738 0 : status = werror_to_ntstatus(WERR_BAD_NET_RESP);
739 0 : tevent_req_nterror(req, status);
740 0 : return;
741 : }
742 :
743 2246 : if (ctr_level == 6) {
744 2246 : if (!W_ERROR_IS_OK(ctr6->drs_error)) {
745 0 : status = werror_to_ntstatus(ctr6->drs_error);
746 0 : tevent_req_nterror(req, status);
747 0 : return;
748 : }
749 2246 : extended_ret = ctr6->extended_ret;
750 : }
751 :
752 2246 : if (ctr_level == 1) {
753 0 : extended_ret = ctr1->extended_ret;
754 : }
755 :
756 2246 : if (state->op->extended_op != DRSUAPI_EXOP_NONE) {
757 24 : state->op->extended_ret = extended_ret;
758 :
759 24 : if (extended_ret != DRSUAPI_EXOP_ERR_SUCCESS) {
760 0 : status = NT_STATUS_UNSUCCESSFUL;
761 0 : tevent_req_nterror(req, status);
762 0 : return;
763 : }
764 : }
765 :
766 2246 : dreplsrv_op_pull_source_apply_changes_trigger(req, r, ctr_level, ctr1, ctr6);
767 : }
768 :
769 : /**
770 : * If processing a chunk of replication data fails, check if it is due to a
771 : * problem that can be fixed by setting extra flags in the GetNCChanges request,
772 : * i.e. GET_ANC or GET_TGT.
773 : * @returns NT_STATUS_OK if the request was retried, and an error code if not
774 : */
775 34 : static NTSTATUS dreplsrv_op_pull_retry_with_flags(struct tevent_req *req,
776 : WERROR error_code)
777 : {
778 0 : struct dreplsrv_op_pull_source_state *state;
779 34 : NTSTATUS nt_status = NT_STATUS_OK;
780 :
781 34 : state = tevent_req_data(req, struct dreplsrv_op_pull_source_state);
782 :
783 : /*
784 : * Check if we failed to apply the records due to a missing parent or
785 : * target object. If so, try again and ask for any missing parent/target
786 : * objects to be included this time.
787 : */
788 34 : if (W_ERROR_EQUAL(error_code, WERR_DS_DRA_RECYCLED_TARGET)) {
789 :
790 9 : if (state->op->more_flags & DRSUAPI_DRS_GET_TGT) {
791 0 : DEBUG(1,("Missing target object despite setting DRSUAPI_DRS_GET_TGT flag\n"));
792 0 : nt_status = NT_STATUS_INVALID_NETWORK_RESPONSE;
793 : } else {
794 9 : state->op->more_flags |= DRSUAPI_DRS_GET_TGT;
795 9 : DEBUG(1,("Missing target object when we didn't set the DRSUAPI_DRS_GET_TGT flag, retrying\n"));
796 9 : dreplsrv_op_pull_source_get_changes_trigger(req);
797 : }
798 25 : } else if (W_ERROR_EQUAL(error_code, WERR_DS_DRA_MISSING_PARENT)) {
799 :
800 25 : if (state->op->options & DRSUAPI_DRS_GET_ANC) {
801 0 : DEBUG(1,("Missing parent object despite setting DRSUAPI_DRS_GET_ANC flag\n"));
802 0 : nt_status = NT_STATUS_INVALID_NETWORK_RESPONSE;
803 : } else {
804 25 : state->op->options |= DRSUAPI_DRS_GET_ANC;
805 25 : DEBUG(4,("Missing parent object when we didn't set the DRSUAPI_DRS_GET_ANC flag, retrying\n"));
806 25 : dreplsrv_op_pull_source_get_changes_trigger(req);
807 : }
808 : } else {
809 0 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
810 : }
811 :
812 34 : return nt_status;
813 : }
814 :
815 :
816 : static void dreplsrv_update_refs_trigger(struct tevent_req *req);
817 :
818 2246 : static void dreplsrv_op_pull_source_apply_changes_trigger(struct tevent_req *req,
819 : struct drsuapi_DsGetNCChanges *r,
820 : uint32_t ctr_level,
821 : struct drsuapi_DsGetNCChangesCtr1 *ctr1,
822 : struct drsuapi_DsGetNCChangesCtr6 *ctr6)
823 : {
824 2246 : struct dreplsrv_op_pull_source_state *state = tevent_req_data(req,
825 : struct dreplsrv_op_pull_source_state);
826 2246 : struct repsFromTo1 rf1 = *state->op->source_dsa->repsFrom1;
827 2246 : struct dreplsrv_service *service = state->op->service;
828 2246 : struct dreplsrv_partition *partition = state->op->source_dsa->partition;
829 2246 : struct dreplsrv_drsuapi_connection *drsuapi = state->op->source_dsa->conn->drsuapi;
830 2246 : struct ldb_dn *schema_dn = ldb_get_schema_basedn(service->samdb);
831 2246 : struct dreplsrv_op_pull_source_schema_cycle *sc = NULL;
832 0 : struct dsdb_schema *schema;
833 2246 : struct dsdb_schema *working_schema = NULL;
834 0 : const struct drsuapi_DsReplicaOIDMapping_Ctr *mapping_ctr;
835 0 : uint32_t object_count;
836 0 : struct drsuapi_DsReplicaObjectListItemEx *first_object;
837 0 : uint32_t linked_attributes_count;
838 0 : struct drsuapi_DsReplicaLinkedAttribute *linked_attributes;
839 0 : const struct drsuapi_DsReplicaCursor2CtrEx *uptodateness_vector;
840 0 : struct dsdb_extended_replicated_objects *objects;
841 2246 : bool more_data = false;
842 0 : WERROR status;
843 0 : NTSTATUS nt_status;
844 2246 : uint32_t dsdb_repl_flags = 0;
845 2246 : struct ldb_dn *nc_root = NULL;
846 2246 : bool was_schema = false;
847 0 : int ret;
848 :
849 2246 : switch (ctr_level) {
850 0 : case 1:
851 0 : mapping_ctr = &ctr1->mapping_ctr;
852 0 : object_count = ctr1->object_count;
853 0 : first_object = ctr1->first_object;
854 0 : linked_attributes_count = 0;
855 0 : linked_attributes = NULL;
856 0 : rf1.source_dsa_obj_guid = ctr1->source_dsa_guid;
857 0 : rf1.source_dsa_invocation_id = ctr1->source_dsa_invocation_id;
858 0 : rf1.highwatermark = ctr1->new_highwatermark;
859 0 : uptodateness_vector = NULL; /* TODO: map it */
860 0 : more_data = ctr1->more_data;
861 0 : break;
862 2246 : case 6:
863 2246 : mapping_ctr = &ctr6->mapping_ctr;
864 2246 : object_count = ctr6->object_count;
865 2246 : first_object = ctr6->first_object;
866 2246 : linked_attributes_count = ctr6->linked_attributes_count;
867 2246 : linked_attributes = ctr6->linked_attributes;
868 2246 : rf1.source_dsa_obj_guid = ctr6->source_dsa_guid;
869 2246 : rf1.source_dsa_invocation_id = ctr6->source_dsa_invocation_id;
870 2246 : rf1.highwatermark = ctr6->new_highwatermark;
871 2246 : uptodateness_vector = ctr6->uptodateness_vector;
872 2246 : more_data = ctr6->more_data;
873 2246 : break;
874 0 : default:
875 0 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
876 0 : tevent_req_nterror(req, nt_status);
877 0 : return;
878 : }
879 :
880 : /*
881 : * We need to cache the schema changes until we replicated
882 : * everything before we can apply the new schema.
883 : */
884 2246 : if (state->schema_cycle != NULL) {
885 114 : TALLOC_CTX *mem = NULL;
886 114 : struct drsuapi_DsReplicaObjectListItemEx **ptr = NULL;
887 114 : struct drsuapi_DsReplicaObjectListItemEx *l = NULL;
888 :
889 114 : was_schema = true;
890 114 : sc = state->schema_cycle;
891 :
892 114 : sc->repsFrom1 = rf1;
893 :
894 114 : if (sc->first_object == NULL) {
895 108 : mem = sc;
896 108 : ptr = &sc->first_object;
897 : } else {
898 6 : mem = sc->last_object;
899 6 : ptr = &sc->last_object->next_object;
900 : }
901 114 : *ptr = talloc_move(mem, &first_object);
902 650 : for (l = *ptr; l != NULL; l = l->next_object) {
903 596 : sc->object_count++;
904 596 : if (l->next_object == NULL) {
905 60 : sc->last_object = l;
906 60 : break;
907 : }
908 : }
909 :
910 114 : if (sc->linked_attributes_count == 0) {
911 110 : sc->linked_attributes = talloc_move(sc, &linked_attributes);
912 110 : sc->linked_attributes_count = linked_attributes_count;
913 110 : linked_attributes_count = 0;
914 4 : } else if (linked_attributes_count > 0) {
915 4 : struct drsuapi_DsReplicaLinkedAttribute *new_las = NULL;
916 4 : struct drsuapi_DsReplicaLinkedAttribute *tmp_las = NULL;
917 0 : uint64_t new_count;
918 0 : uint64_t add_size;
919 0 : uint32_t add_idx;
920 :
921 4 : new_count = sc->linked_attributes_count;
922 4 : new_count += linked_attributes_count;
923 4 : if (new_count > UINT32_MAX) {
924 0 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
925 0 : tevent_req_nterror(req, nt_status);
926 0 : return;
927 : }
928 4 : add_size = linked_attributes_count;
929 4 : add_size *= sizeof(linked_attributes[0]);
930 0 : if (add_size > SIZE_MAX) {
931 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
932 : tevent_req_nterror(req, nt_status);
933 : return;
934 : }
935 4 : add_idx = sc->linked_attributes_count;
936 :
937 4 : tmp_las = talloc_realloc(sc,
938 : sc->linked_attributes,
939 : struct drsuapi_DsReplicaLinkedAttribute,
940 : new_count);
941 4 : if (tevent_req_nomem(tmp_las, req)) {
942 0 : return;
943 : }
944 4 : new_las = talloc_move(tmp_las, &linked_attributes);
945 4 : memcpy(&tmp_las[add_idx], new_las, add_size);
946 4 : sc->linked_attributes = tmp_las;
947 4 : sc->linked_attributes_count = new_count;
948 4 : linked_attributes_count = 0;
949 : }
950 :
951 114 : if (more_data) {
952 : /* we don't need this structure anymore */
953 6 : TALLOC_FREE(r);
954 :
955 6 : dreplsrv_op_pull_source_get_changes_trigger(req);
956 6 : return;
957 : }
958 :
959 : /* detach sc from state */
960 108 : state->schema_cycle = NULL;
961 : }
962 :
963 2240 : schema = dsdb_get_schema(service->samdb, state);
964 2240 : if (!schema) {
965 0 : DEBUG(0,(__location__ ": Schema is not loaded yet!\n"));
966 0 : tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
967 0 : return;
968 : }
969 :
970 : /*
971 : * Decide what working schema to use for object conversion.
972 : * We won't need a working schema for empty replicas sent.
973 : */
974 2240 : if (sc != NULL) {
975 108 : first_object = talloc_move(r, &sc->first_object);
976 108 : object_count = sc->object_count;
977 108 : linked_attributes = talloc_move(r, &sc->linked_attributes);
978 108 : linked_attributes_count = sc->linked_attributes_count;
979 108 : TALLOC_FREE(sc);
980 :
981 108 : if (first_object != NULL) {
982 : /* create working schema to convert objects with */
983 58 : status = dsdb_repl_make_working_schema(service->samdb,
984 : schema,
985 : mapping_ctr,
986 : object_count,
987 : first_object,
988 58 : &drsuapi->gensec_skey,
989 : state, &working_schema);
990 58 : if (!W_ERROR_IS_OK(status)) {
991 0 : DEBUG(0,("Failed to create working schema: %s\n",
992 : win_errstr(status)));
993 0 : tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
994 0 : return;
995 : }
996 : }
997 : }
998 :
999 2240 : if (partition->partial_replica || partition->rodc_replica) {
1000 442 : dsdb_repl_flags |= DSDB_REPL_FLAG_PARTIAL_REPLICA;
1001 : }
1002 2240 : if (state->op->options & DRSUAPI_DRS_FULL_SYNC_IN_PROGRESS) {
1003 161 : dsdb_repl_flags |= DSDB_REPL_FLAG_PRIORITISE_INCOMING;
1004 : }
1005 2240 : if (state->op->options & DRSUAPI_DRS_SPECIAL_SECRET_PROCESSING) {
1006 0 : dsdb_repl_flags |= DSDB_REPL_FLAG_EXPECT_NO_SECRETS;
1007 : }
1008 2240 : if (state->op->options & DRSUAPI_DRS_CRITICAL_ONLY ||
1009 2240 : state->op->extended_op != DRSUAPI_EXOP_NONE) {
1010 24 : dsdb_repl_flags |= DSDB_REPL_FLAG_OBJECT_SUBSET;
1011 : }
1012 :
1013 2240 : if (state->op->more_flags & DRSUAPI_DRS_GET_TGT) {
1014 14 : dsdb_repl_flags |= DSDB_REPL_FLAG_TARGETS_UPTODATE;
1015 : }
1016 :
1017 2240 : if (state->op->extended_op != DRSUAPI_EXOP_NONE) {
1018 24 : ret = dsdb_find_nc_root(service->samdb, partition,
1019 : partition->dn, &nc_root);
1020 24 : if (ret != LDB_SUCCESS) {
1021 0 : DEBUG(0,(__location__ ": Failed to find nc_root for %s\n",
1022 : ldb_dn_get_linearized(partition->dn)));
1023 0 : tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
1024 0 : return;
1025 : }
1026 : } else {
1027 2216 : nc_root = partition->dn;
1028 : }
1029 :
1030 2240 : status = dsdb_replicated_objects_convert(service->samdb,
1031 2240 : working_schema ? working_schema : schema,
1032 : nc_root,
1033 : mapping_ctr,
1034 : object_count,
1035 : first_object,
1036 : linked_attributes_count,
1037 : linked_attributes,
1038 : &rf1,
1039 : uptodateness_vector,
1040 2240 : &drsuapi->gensec_skey,
1041 : dsdb_repl_flags,
1042 : state, &objects);
1043 :
1044 2240 : if (W_ERROR_EQUAL(status, WERR_DS_DRA_SCHEMA_MISMATCH)) {
1045 0 : struct dreplsrv_partition *p;
1046 0 : bool ok;
1047 :
1048 10 : if (was_schema) {
1049 0 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
1050 0 : DBG_ERR("Got mismatch for schema partition: %s/%s\n",
1051 : win_errstr(status), nt_errstr(nt_status));
1052 0 : tevent_req_nterror(req, nt_status);
1053 0 : return;
1054 : }
1055 :
1056 10 : if (state->retry_started) {
1057 1 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
1058 1 : DEBUG(0,("Failed to convert objects after retry: %s/%s\n",
1059 : win_errstr(status), nt_errstr(nt_status)));
1060 1 : tevent_req_nterror(req, nt_status);
1061 1 : return;
1062 : }
1063 :
1064 : /*
1065 : * Change info sync or extended operation into a fetch
1066 : * of the schema partition, so we get all the schema
1067 : * objects we need.
1068 : *
1069 : * We don't want to re-do the remote exop,
1070 : * unless it was REPL_SECRET so we set the
1071 : * fallback operation to just be a fetch of
1072 : * the relevant partition.
1073 : */
1074 :
1075 :
1076 9 : if (state->op->extended_op == DRSUAPI_EXOP_REPL_SECRET) {
1077 0 : state->extended_op_retry = state->op->extended_op;
1078 : } else {
1079 9 : state->extended_op_retry = DRSUAPI_EXOP_NONE;
1080 : }
1081 9 : state->op->extended_op = DRSUAPI_EXOP_NONE;
1082 :
1083 9 : if (ldb_dn_compare(nc_root, partition->dn) == 0) {
1084 9 : state->source_dsa_retry = state->op->source_dsa;
1085 : } else {
1086 0 : status = dreplsrv_partition_find_for_nc(service,
1087 : NULL, NULL,
1088 : ldb_dn_get_linearized(nc_root),
1089 : &p);
1090 0 : if (!W_ERROR_IS_OK(status)) {
1091 0 : DEBUG(2, ("Failed to find requested Naming Context for %s: %s\n",
1092 : ldb_dn_get_linearized(nc_root),
1093 : win_errstr(status)));
1094 0 : nt_status = werror_to_ntstatus(status);
1095 0 : tevent_req_nterror(req, nt_status);
1096 0 : return;
1097 : }
1098 0 : status = dreplsrv_partition_source_dsa_by_guid(p,
1099 0 : &state->op->source_dsa->repsFrom1->source_dsa_obj_guid,
1100 : &state->source_dsa_retry);
1101 :
1102 0 : if (!W_ERROR_IS_OK(status)) {
1103 0 : struct GUID_txt_buf str;
1104 0 : DEBUG(2, ("Failed to find requested source DSA for %s and %s: %s\n",
1105 : ldb_dn_get_linearized(nc_root),
1106 : GUID_buf_string(&state->op->source_dsa->repsFrom1->source_dsa_obj_guid, &str),
1107 : win_errstr(status)));
1108 0 : nt_status = werror_to_ntstatus(status);
1109 0 : tevent_req_nterror(req, nt_status);
1110 0 : return;
1111 : }
1112 : }
1113 :
1114 : /* Find schema naming context to be synchronized first */
1115 9 : status = dreplsrv_partition_find_for_nc(service,
1116 : NULL, NULL,
1117 : ldb_dn_get_linearized(schema_dn),
1118 : &p);
1119 9 : if (!W_ERROR_IS_OK(status)) {
1120 0 : DEBUG(2, ("Failed to find requested Naming Context for schema: %s\n",
1121 : win_errstr(status)));
1122 0 : nt_status = werror_to_ntstatus(status);
1123 0 : tevent_req_nterror(req, nt_status);
1124 0 : return;
1125 : }
1126 :
1127 9 : status = dreplsrv_partition_source_dsa_by_guid(p,
1128 9 : &state->op->source_dsa->repsFrom1->source_dsa_obj_guid,
1129 9 : &state->op->source_dsa);
1130 9 : if (!W_ERROR_IS_OK(status)) {
1131 0 : struct GUID_txt_buf str;
1132 0 : DEBUG(2, ("Failed to find requested source DSA for %s and %s: %s\n",
1133 : ldb_dn_get_linearized(schema_dn),
1134 : GUID_buf_string(&state->op->source_dsa->repsFrom1->source_dsa_obj_guid, &str),
1135 : win_errstr(status)));
1136 0 : nt_status = werror_to_ntstatus(status);
1137 0 : tevent_req_nterror(req, nt_status);
1138 0 : return;
1139 : }
1140 9 : DEBUG(4,("Wrong schema when applying reply GetNCChanges, retrying\n"));
1141 :
1142 9 : state->retry_started = true;
1143 :
1144 9 : ok = dreplsrv_op_pull_source_detect_schema_cycle(req);
1145 9 : if (!ok) {
1146 0 : return;
1147 : }
1148 :
1149 9 : dreplsrv_op_pull_source_get_changes_trigger(req);
1150 9 : return;
1151 :
1152 2230 : } else if (!W_ERROR_IS_OK(status)) {
1153 0 : nt_status = werror_to_ntstatus(WERR_BAD_NET_RESP);
1154 0 : DEBUG(0,("Failed to convert objects: %s/%s\n",
1155 : win_errstr(status), nt_errstr(nt_status)));
1156 0 : tevent_req_nterror(req, nt_status);
1157 0 : return;
1158 : }
1159 :
1160 2230 : status = dsdb_replicated_objects_commit(service->samdb,
1161 : working_schema,
1162 : objects,
1163 2230 : &state->op->source_dsa->notify_uSN);
1164 2230 : talloc_free(objects);
1165 :
1166 2230 : if (!W_ERROR_IS_OK(status)) {
1167 :
1168 : /*
1169 : * Check if this error can be fixed by resending the GetNCChanges
1170 : * request with extra flags set (i.e. GET_ANC/GET_TGT)
1171 : */
1172 34 : nt_status = dreplsrv_op_pull_retry_with_flags(req, status);
1173 :
1174 34 : if (NT_STATUS_IS_OK(nt_status)) {
1175 :
1176 : /*
1177 : * We resent the request. Don't update the highwatermark,
1178 : * we'll start this part of the cycle again.
1179 : */
1180 34 : return;
1181 : }
1182 :
1183 0 : DEBUG(0,("Failed to commit objects: %s/%s\n",
1184 : win_errstr(status), nt_errstr(nt_status)));
1185 0 : tevent_req_nterror(req, nt_status);
1186 0 : return;
1187 : }
1188 :
1189 2196 : if (state->op->extended_op == DRSUAPI_EXOP_NONE) {
1190 : /* if it applied fine, we need to update the highwatermark */
1191 2172 : *state->op->source_dsa->repsFrom1 = rf1;
1192 : }
1193 :
1194 : /* we don't need this maybe very large structure anymore */
1195 2196 : TALLOC_FREE(r);
1196 :
1197 2196 : if (more_data) {
1198 394 : dreplsrv_op_pull_source_get_changes_trigger(req);
1199 394 : return;
1200 : }
1201 :
1202 : /*
1203 : * If we had to divert via doing some other thing, such as
1204 : * pulling the schema, then go back and do the original
1205 : * operation once we are done.
1206 : */
1207 1802 : if (state->source_dsa_retry != NULL) {
1208 9 : state->op->source_dsa = state->source_dsa_retry;
1209 9 : state->op->extended_op = state->extended_op_retry;
1210 9 : state->source_dsa_retry = NULL;
1211 9 : dreplsrv_op_pull_source_get_changes_trigger(req);
1212 9 : return;
1213 : }
1214 :
1215 1793 : if (state->op->extended_op != DRSUAPI_EXOP_NONE ||
1216 1769 : state->op->service->am_rodc) {
1217 : /*
1218 : we don't do the UpdateRefs for extended ops or if we
1219 : are a RODC
1220 : */
1221 388 : tevent_req_done(req);
1222 388 : return;
1223 : }
1224 :
1225 : /* now we need to update the repsTo record for this partition
1226 : on the server. These records are initially established when
1227 : we join the domain, but they quickly expire. We do it here
1228 : so we can use the already established DRSUAPI pipe
1229 : */
1230 1405 : dreplsrv_update_refs_trigger(req);
1231 : }
1232 :
1233 : static void dreplsrv_update_refs_done(struct tevent_req *subreq);
1234 :
1235 : /*
1236 : send a UpdateRefs request to refresh our repsTo record on the server
1237 : */
1238 1405 : static void dreplsrv_update_refs_trigger(struct tevent_req *req)
1239 : {
1240 1405 : struct dreplsrv_op_pull_source_state *state = tevent_req_data(req,
1241 : struct dreplsrv_op_pull_source_state);
1242 1405 : struct dreplsrv_service *service = state->op->service;
1243 1405 : struct dreplsrv_partition *partition = state->op->source_dsa->partition;
1244 1405 : struct dreplsrv_drsuapi_connection *drsuapi = state->op->source_dsa->conn->drsuapi;
1245 0 : struct drsuapi_DsReplicaUpdateRefs *r;
1246 0 : char *ntds_dns_name;
1247 0 : struct tevent_req *subreq;
1248 :
1249 1405 : r = talloc(state, struct drsuapi_DsReplicaUpdateRefs);
1250 1405 : if (tevent_req_nomem(r, req)) {
1251 0 : return;
1252 : }
1253 :
1254 1405 : ntds_dns_name = samdb_ntds_msdcs_dns_name(service->samdb, r, &service->ntds_guid);
1255 1405 : if (tevent_req_nomem(ntds_dns_name, req)) {
1256 0 : talloc_free(r);
1257 0 : return;
1258 : }
1259 :
1260 1405 : r->in.bind_handle = &drsuapi->bind_handle;
1261 1405 : r->in.level = 1;
1262 1405 : r->in.req.req1.naming_context = &partition->nc;
1263 1405 : r->in.req.req1.dest_dsa_dns_name = ntds_dns_name;
1264 1405 : r->in.req.req1.dest_dsa_guid = service->ntds_guid;
1265 1405 : r->in.req.req1.options = DRSUAPI_DRS_ADD_REF | DRSUAPI_DRS_DEL_REF;
1266 1405 : if (!service->am_rodc) {
1267 1405 : r->in.req.req1.options |= DRSUAPI_DRS_WRIT_REP;
1268 : }
1269 :
1270 1405 : state->ndr_struct_ptr = r;
1271 1405 : subreq = dcerpc_drsuapi_DsReplicaUpdateRefs_r_send(state,
1272 : state->ev,
1273 : drsuapi->drsuapi_handle,
1274 : r);
1275 1405 : if (tevent_req_nomem(subreq, req)) {
1276 0 : talloc_free(r);
1277 0 : return;
1278 : }
1279 1405 : tevent_req_set_callback(subreq, dreplsrv_update_refs_done, req);
1280 : }
1281 :
1282 : /*
1283 : receive a UpdateRefs reply
1284 : */
1285 1405 : static void dreplsrv_update_refs_done(struct tevent_req *subreq)
1286 : {
1287 1405 : struct tevent_req *req = tevent_req_callback_data(subreq,
1288 : struct tevent_req);
1289 1405 : struct dreplsrv_op_pull_source_state *state = tevent_req_data(req,
1290 : struct dreplsrv_op_pull_source_state);
1291 1405 : struct drsuapi_DsReplicaUpdateRefs *r = talloc_get_type(state->ndr_struct_ptr,
1292 : struct drsuapi_DsReplicaUpdateRefs);
1293 0 : NTSTATUS status;
1294 :
1295 1405 : state->ndr_struct_ptr = NULL;
1296 :
1297 1405 : status = dcerpc_drsuapi_DsReplicaUpdateRefs_r_recv(subreq, r);
1298 1405 : TALLOC_FREE(subreq);
1299 1405 : if (!NT_STATUS_IS_OK(status)) {
1300 0 : DEBUG(0,("UpdateRefs failed with %s\n",
1301 : nt_errstr(status)));
1302 0 : tevent_req_nterror(req, status);
1303 0 : return;
1304 : }
1305 :
1306 1405 : if (!W_ERROR_IS_OK(r->out.result)) {
1307 0 : status = werror_to_ntstatus(r->out.result);
1308 0 : DEBUG(0,("UpdateRefs failed with %s/%s for %s %s\n",
1309 : win_errstr(r->out.result),
1310 : nt_errstr(status),
1311 : r->in.req.req1.dest_dsa_dns_name,
1312 : r->in.req.req1.naming_context->dn));
1313 : /*
1314 : * TODO we are currently not sending the
1315 : * DsReplicaUpdateRefs at the correct moment,
1316 : * we do it just after a GetNcChanges which is
1317 : * not always correct.
1318 : * Especially when another DC is trying to demote
1319 : * it will sends us a DsReplicaSync that will trigger a getNcChanges
1320 : * this call will succeed but the DsRecplicaUpdateRefs that we send
1321 : * just after will not because the DC is in a demote state and
1322 : * will reply us a WERR_DS_DRA_BUSY, this error will cause us to
1323 : * answer to the DsReplicaSync with a non OK status, the other DC
1324 : * will stop the demote due to this error.
1325 : * In order to cope with this we will for the moment consider
1326 : * a DS_DRA_BUSY not as an error.
1327 : * It's not ideal but it should not have a too huge impact for
1328 : * running production as this error otherwise never happen and
1329 : * due to the fact the send a DsReplicaUpdateRefs after each getNcChanges
1330 : */
1331 0 : if (!W_ERROR_EQUAL(r->out.result, WERR_DS_DRA_BUSY)) {
1332 0 : tevent_req_nterror(req, status);
1333 0 : return;
1334 : }
1335 : }
1336 :
1337 1405 : DEBUG(4,("UpdateRefs OK for %s %s\n",
1338 : r->in.req.req1.dest_dsa_dns_name,
1339 : r->in.req.req1.naming_context->dn));
1340 :
1341 1405 : tevent_req_done(req);
1342 : }
1343 :
1344 3846 : WERROR dreplsrv_op_pull_source_recv(struct tevent_req *req)
1345 : {
1346 0 : NTSTATUS status;
1347 :
1348 3846 : if (tevent_req_is_nterror(req, &status)) {
1349 2053 : tevent_req_received(req);
1350 2053 : return ntstatus_to_werror(status);
1351 : }
1352 :
1353 1793 : tevent_req_received(req);
1354 1793 : return WERR_OK;
1355 : }
1356 :
|