2 Unix SMB/CIFS implementation.
5 Copyright (C) Stefan Metzmacher <metze@samba.org> 2006-2007
6 Copyright (C) Andrew Bartlett <abartlet@samba.org> 2006-2008
7 Copyright (C) Matthieu Patou <mat@matws.net> 2011
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3 of the License, or
12 (at your option) any later version.
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "lib/util/dlinklist.h"
26 #include "dsdb/samdb/samdb.h"
27 #include <ldb_module.h>
28 #include "param/param.h"
29 #include "librpc/ndr/libndr.h"
30 #include "librpc/gen_ndr/ndr_misc.h"
31 #include "lib/util/tsort.h"
34 override the name to attribute handler function
36 const struct ldb_schema_attribute
*dsdb_attribute_handler_override(struct ldb_context
*ldb
,
40 struct dsdb_schema
*schema
= talloc_get_type_abort(private_data
, struct dsdb_schema
);
41 const struct dsdb_attribute
*a
= dsdb_attribute_by_lDAPDisplayName(schema
, name
);
43 /* this will fall back to ldb internal handling */
46 return a
->ldb_schema_attribute
;
49 static int dsdb_schema_set_attributes(struct ldb_context
*ldb
, struct dsdb_schema
*schema
, bool write_attributes
)
51 int ret
= LDB_SUCCESS
;
52 struct ldb_result
*res
;
53 struct ldb_result
*res_idx
;
54 struct dsdb_attribute
*attr
;
55 struct ldb_message
*mod_msg
;
57 struct ldb_message
*msg
;
58 struct ldb_message
*msg_idx
;
60 /* setup our own attribute name to schema handler */
61 ldb_schema_attribute_set_override_handler(ldb
, dsdb_attribute_handler_override
, schema
);
63 if (!write_attributes
) {
67 mem_ctx
= talloc_new(ldb
);
72 msg
= ldb_msg_new(mem_ctx
);
77 msg_idx
= ldb_msg_new(mem_ctx
);
82 msg
->dn
= ldb_dn_new(msg
, ldb
, "@ATTRIBUTES");
87 msg_idx
->dn
= ldb_dn_new(msg_idx
, ldb
, "@INDEXLIST");
93 ret
= ldb_msg_add_string(msg_idx
, "@IDXONE", "1");
94 if (ret
!= LDB_SUCCESS
) {
98 for (attr
= schema
->attributes
; attr
; attr
= attr
->next
) {
99 const char *syntax
= attr
->syntax
->ldb_syntax
;
102 syntax
= attr
->syntax
->ldap_oid
;
106 * Write out a rough approximation of the schema
107 * as an @ATTRIBUTES value, for bootstrapping
109 if (strcmp(syntax
, LDB_SYNTAX_INTEGER
) == 0) {
110 ret
= ldb_msg_add_string(msg
, attr
->lDAPDisplayName
, "INTEGER");
111 } else if (strcmp(syntax
, LDB_SYNTAX_DIRECTORY_STRING
) == 0) {
112 ret
= ldb_msg_add_string(msg
, attr
->lDAPDisplayName
, "CASE_INSENSITIVE");
114 if (ret
!= LDB_SUCCESS
) {
118 if (attr
->searchFlags
& SEARCH_FLAG_ATTINDEX
) {
119 ret
= ldb_msg_add_string(msg_idx
, "@IDXATTR", attr
->lDAPDisplayName
);
120 if (ret
!= LDB_SUCCESS
) {
126 if (ret
!= LDB_SUCCESS
) {
127 talloc_free(mem_ctx
);
132 * Try to avoid churning the attributes too much,
133 * we only want to do this if they have changed
135 ret
= ldb_search(ldb
, mem_ctx
, &res
, msg
->dn
, LDB_SCOPE_BASE
, NULL
,
137 if (ret
== LDB_ERR_NO_SUCH_OBJECT
) {
138 ret
= ldb_add(ldb
, msg
);
139 } else if (ret
!= LDB_SUCCESS
) {
140 } else if (res
->count
!= 1) {
141 ret
= ldb_add(ldb
, msg
);
144 /* Annoyingly added to our search results */
145 ldb_msg_remove_attr(res
->msgs
[0], "distinguishedName");
147 ret
= ldb_msg_difference(ldb
, mem_ctx
,
148 res
->msgs
[0], msg
, &mod_msg
);
149 if (ret
!= LDB_SUCCESS
) {
152 if (mod_msg
->num_elements
> 0) {
153 ret
= dsdb_replace(ldb
, mod_msg
, 0);
155 talloc_free(mod_msg
);
158 if (ret
== LDB_ERR_OPERATIONS_ERROR
|| ret
== LDB_ERR_INSUFFICIENT_ACCESS_RIGHTS
|| ret
== LDB_ERR_INVALID_DN_SYNTAX
) {
159 /* We might be on a read-only DB or LDAP */
162 if (ret
!= LDB_SUCCESS
) {
163 talloc_free(mem_ctx
);
167 /* Now write out the indexes, as found in the schema (if they have changed) */
169 ret
= ldb_search(ldb
, mem_ctx
, &res_idx
, msg_idx
->dn
, LDB_SCOPE_BASE
,
171 if (ret
== LDB_ERR_NO_SUCH_OBJECT
) {
172 ret
= ldb_add(ldb
, msg_idx
);
173 } else if (ret
!= LDB_SUCCESS
) {
174 } else if (res_idx
->count
!= 1) {
175 ret
= ldb_add(ldb
, msg_idx
);
178 /* Annoyingly added to our search results */
179 ldb_msg_remove_attr(res_idx
->msgs
[0], "distinguishedName");
181 ret
= ldb_msg_difference(ldb
, mem_ctx
,
182 res_idx
->msgs
[0], msg_idx
, &mod_msg
);
183 if (ret
!= LDB_SUCCESS
) {
186 if (mod_msg
->num_elements
> 0) {
187 ret
= dsdb_replace(ldb
, mod_msg
, 0);
189 talloc_free(mod_msg
);
191 if (ret
== LDB_ERR_OPERATIONS_ERROR
|| ret
== LDB_ERR_INSUFFICIENT_ACCESS_RIGHTS
|| ret
== LDB_ERR_INVALID_DN_SYNTAX
) {
192 /* We might be on a read-only DB */
195 talloc_free(mem_ctx
);
199 talloc_free(mem_ctx
);
200 return ldb_operr(ldb
);
205 create extra attribute shortcuts
207 static void dsdb_setup_attribute_shortcuts(struct ldb_context
*ldb
, struct dsdb_schema
*schema
)
209 struct dsdb_attribute
*attribute
;
211 /* setup fast access to one_way_link and DN format */
212 for (attribute
=schema
->attributes
; attribute
; attribute
=attribute
->next
) {
213 attribute
->dn_format
= dsdb_dn_oid_to_format(attribute
->syntax
->ldap_oid
);
215 if (attribute
->dn_format
== DSDB_INVALID_DN
) {
216 attribute
->one_way_link
= false;
220 /* these are not considered to be one way links for
221 the purpose of DN link fixups */
222 if (ldb_attr_cmp("distinguishedName", attribute
->lDAPDisplayName
) == 0 ||
223 ldb_attr_cmp("objectCategory", attribute
->lDAPDisplayName
) == 0) {
224 attribute
->one_way_link
= false;
228 if (attribute
->linkID
== 0) {
229 attribute
->one_way_link
= true;
232 /* handle attributes with a linkID but no backlink */
233 if (dsdb_attribute_by_linkID(schema
, attribute
->linkID
) == NULL
) {
234 attribute
->one_way_link
= true;
237 attribute
->one_way_link
= false;
241 static int uint32_cmp(uint32_t c1
, uint32_t c2
)
243 if (c1
== c2
) return 0;
244 return c1
> c2
? 1 : -1;
247 static int dsdb_compare_class_by_lDAPDisplayName(struct dsdb_class
**c1
, struct dsdb_class
**c2
)
249 return strcasecmp((*c1
)->lDAPDisplayName
, (*c2
)->lDAPDisplayName
);
251 static int dsdb_compare_class_by_governsID_id(struct dsdb_class
**c1
, struct dsdb_class
**c2
)
253 return uint32_cmp((*c1
)->governsID_id
, (*c2
)->governsID_id
);
255 static int dsdb_compare_class_by_governsID_oid(struct dsdb_class
**c1
, struct dsdb_class
**c2
)
257 return strcasecmp((*c1
)->governsID_oid
, (*c2
)->governsID_oid
);
259 static int dsdb_compare_class_by_cn(struct dsdb_class
**c1
, struct dsdb_class
**c2
)
261 return strcasecmp((*c1
)->cn
, (*c2
)->cn
);
264 static int dsdb_compare_attribute_by_lDAPDisplayName(struct dsdb_attribute
**a1
, struct dsdb_attribute
**a2
)
266 return strcasecmp((*a1
)->lDAPDisplayName
, (*a2
)->lDAPDisplayName
);
268 static int dsdb_compare_attribute_by_attributeID_id(struct dsdb_attribute
**a1
, struct dsdb_attribute
**a2
)
270 return uint32_cmp((*a1
)->attributeID_id
, (*a2
)->attributeID_id
);
272 static int dsdb_compare_attribute_by_msDS_IntId(struct dsdb_attribute
**a1
, struct dsdb_attribute
**a2
)
274 return uint32_cmp((*a1
)->msDS_IntId
, (*a2
)->msDS_IntId
);
276 static int dsdb_compare_attribute_by_attributeID_oid(struct dsdb_attribute
**a1
, struct dsdb_attribute
**a2
)
278 return strcasecmp((*a1
)->attributeID_oid
, (*a2
)->attributeID_oid
);
280 static int dsdb_compare_attribute_by_linkID(struct dsdb_attribute
**a1
, struct dsdb_attribute
**a2
)
282 return uint32_cmp((*a1
)->linkID
, (*a2
)->linkID
);
286 * Clean up Classes and Attributes accessor arrays
288 static void dsdb_sorted_accessors_free(struct dsdb_schema
*schema
)
290 /* free classes accessors */
291 TALLOC_FREE(schema
->classes_by_lDAPDisplayName
);
292 TALLOC_FREE(schema
->classes_by_governsID_id
);
293 TALLOC_FREE(schema
->classes_by_governsID_oid
);
294 TALLOC_FREE(schema
->classes_by_cn
);
295 /* free attribute accessors */
296 TALLOC_FREE(schema
->attributes_by_lDAPDisplayName
);
297 TALLOC_FREE(schema
->attributes_by_attributeID_id
);
298 TALLOC_FREE(schema
->attributes_by_msDS_IntId
);
299 TALLOC_FREE(schema
->attributes_by_attributeID_oid
);
300 TALLOC_FREE(schema
->attributes_by_linkID
);
304 create the sorted accessor arrays for the schema
306 int dsdb_setup_sorted_accessors(struct ldb_context
*ldb
,
307 struct dsdb_schema
*schema
)
309 struct dsdb_class
*cur
;
310 struct dsdb_attribute
*a
;
312 unsigned int num_int_id
;
314 /* free all caches */
315 dsdb_sorted_accessors_free(schema
);
317 /* count the classes */
318 for (i
=0, cur
=schema
->classes
; cur
; i
++, cur
=cur
->next
) /* noop */ ;
319 schema
->num_classes
= i
;
321 /* setup classes_by_* */
322 schema
->classes_by_lDAPDisplayName
= talloc_array(schema
, struct dsdb_class
*, i
);
323 schema
->classes_by_governsID_id
= talloc_array(schema
, struct dsdb_class
*, i
);
324 schema
->classes_by_governsID_oid
= talloc_array(schema
, struct dsdb_class
*, i
);
325 schema
->classes_by_cn
= talloc_array(schema
, struct dsdb_class
*, i
);
326 if (schema
->classes_by_lDAPDisplayName
== NULL
||
327 schema
->classes_by_governsID_id
== NULL
||
328 schema
->classes_by_governsID_oid
== NULL
||
329 schema
->classes_by_cn
== NULL
) {
333 for (i
=0, cur
=schema
->classes
; cur
; i
++, cur
=cur
->next
) {
334 schema
->classes_by_lDAPDisplayName
[i
] = cur
;
335 schema
->classes_by_governsID_id
[i
] = cur
;
336 schema
->classes_by_governsID_oid
[i
] = cur
;
337 schema
->classes_by_cn
[i
] = cur
;
340 /* sort the arrays */
341 TYPESAFE_QSORT(schema
->classes_by_lDAPDisplayName
, schema
->num_classes
, dsdb_compare_class_by_lDAPDisplayName
);
342 TYPESAFE_QSORT(schema
->classes_by_governsID_id
, schema
->num_classes
, dsdb_compare_class_by_governsID_id
);
343 TYPESAFE_QSORT(schema
->classes_by_governsID_oid
, schema
->num_classes
, dsdb_compare_class_by_governsID_oid
);
344 TYPESAFE_QSORT(schema
->classes_by_cn
, schema
->num_classes
, dsdb_compare_class_by_cn
);
346 /* now build the attribute accessor arrays */
348 /* count the attributes
349 * and attributes with msDS-IntId set */
351 for (i
=0, a
=schema
->attributes
; a
; i
++, a
=a
->next
) {
352 if (a
->msDS_IntId
!= 0) {
356 schema
->num_attributes
= i
;
357 schema
->num_int_id_attr
= num_int_id
;
359 /* setup attributes_by_* */
360 schema
->attributes_by_lDAPDisplayName
= talloc_array(schema
, struct dsdb_attribute
*, i
);
361 schema
->attributes_by_attributeID_id
= talloc_array(schema
, struct dsdb_attribute
*, i
);
362 schema
->attributes_by_msDS_IntId
= talloc_array(schema
,
363 struct dsdb_attribute
*, num_int_id
);
364 schema
->attributes_by_attributeID_oid
= talloc_array(schema
, struct dsdb_attribute
*, i
);
365 schema
->attributes_by_linkID
= talloc_array(schema
, struct dsdb_attribute
*, i
);
366 if (schema
->attributes_by_lDAPDisplayName
== NULL
||
367 schema
->attributes_by_attributeID_id
== NULL
||
368 schema
->attributes_by_msDS_IntId
== NULL
||
369 schema
->attributes_by_attributeID_oid
== NULL
||
370 schema
->attributes_by_linkID
== NULL
) {
375 for (i
=0, a
=schema
->attributes
; a
; i
++, a
=a
->next
) {
376 schema
->attributes_by_lDAPDisplayName
[i
] = a
;
377 schema
->attributes_by_attributeID_id
[i
] = a
;
378 schema
->attributes_by_attributeID_oid
[i
] = a
;
379 schema
->attributes_by_linkID
[i
] = a
;
380 /* append attr-by-msDS-IntId values */
381 if (a
->msDS_IntId
!= 0) {
382 schema
->attributes_by_msDS_IntId
[num_int_id
] = a
;
386 SMB_ASSERT(num_int_id
== schema
->num_int_id_attr
);
388 /* sort the arrays */
389 TYPESAFE_QSORT(schema
->attributes_by_lDAPDisplayName
, schema
->num_attributes
, dsdb_compare_attribute_by_lDAPDisplayName
);
390 TYPESAFE_QSORT(schema
->attributes_by_attributeID_id
, schema
->num_attributes
, dsdb_compare_attribute_by_attributeID_id
);
391 TYPESAFE_QSORT(schema
->attributes_by_msDS_IntId
, schema
->num_int_id_attr
, dsdb_compare_attribute_by_msDS_IntId
);
392 TYPESAFE_QSORT(schema
->attributes_by_attributeID_oid
, schema
->num_attributes
, dsdb_compare_attribute_by_attributeID_oid
);
393 TYPESAFE_QSORT(schema
->attributes_by_linkID
, schema
->num_attributes
, dsdb_compare_attribute_by_linkID
);
395 dsdb_setup_attribute_shortcuts(ldb
, schema
);
400 dsdb_sorted_accessors_free(schema
);
404 int dsdb_setup_schema_inversion(struct ldb_context
*ldb
, struct dsdb_schema
*schema
)
406 /* Walk the list of schema classes */
408 /* For each subClassOf, add us to subclasses of the parent */
410 /* collect these subclasses into a recursive list of total subclasses, preserving order */
412 /* For each subclass under 'top', write the index from it's
413 * order as an integer in the dsdb_class (for sorting
414 * objectClass lists efficiently) */
416 /* Walk the list of schema classes */
418 /* Create a 'total possible superiors' on each class */
423 * Attach the schema to an opaque pointer on the ldb,
424 * so ldb modules can find it
426 int dsdb_set_schema(struct ldb_context
*ldb
, struct dsdb_schema
*schema
)
428 struct dsdb_schema
*old_schema
;
431 ret
= dsdb_setup_sorted_accessors(ldb
, schema
);
432 if (ret
!= LDB_SUCCESS
) {
436 ret
= schema_fill_constructed(schema
);
437 if (ret
!= LDB_SUCCESS
) {
441 old_schema
= ldb_get_opaque(ldb
, "dsdb_schema");
443 ret
= ldb_set_opaque(ldb
, "dsdb_schema", schema
);
444 if (ret
!= LDB_SUCCESS
) {
448 /* Remove the reference to the schema we just overwrote - if there was
449 * none, NULL is harmless here */
450 if (old_schema
!= schema
) {
451 talloc_unlink(ldb
, old_schema
);
452 talloc_steal(ldb
, schema
);
455 ret
= ldb_set_opaque(ldb
, "dsdb_use_global_schema", NULL
);
456 if (ret
!= LDB_SUCCESS
) {
460 /* Set the new attributes based on the new schema */
461 ret
= dsdb_schema_set_attributes(ldb
, schema
, true);
462 if (ret
!= LDB_SUCCESS
) {
470 * Global variable to hold one copy of the schema, used to avoid memory bloat
472 static struct dsdb_schema
*global_schema
;
475 * Make this ldb use a specified schema, already fully calculated and belonging to another ldb
477 int dsdb_reference_schema(struct ldb_context
*ldb
, struct dsdb_schema
*schema
,
478 bool write_attributes
)
481 struct dsdb_schema
*old_schema
;
482 old_schema
= ldb_get_opaque(ldb
, "dsdb_schema");
483 ret
= ldb_set_opaque(ldb
, "dsdb_schema", schema
);
484 if (ret
!= LDB_SUCCESS
) {
488 /* Remove the reference to the schema we just overwrote - if there was
489 * none, NULL is harmless here */
490 talloc_unlink(ldb
, old_schema
);
492 if (talloc_reference(ldb
, schema
) == NULL
) {
496 /* Make this ldb use local schema preferably */
497 ret
= ldb_set_opaque(ldb
, "dsdb_use_global_schema", NULL
);
498 if (ret
!= LDB_SUCCESS
) {
502 ret
= dsdb_schema_set_attributes(ldb
, schema
, write_attributes
);
503 if (ret
!= LDB_SUCCESS
) {
511 * Make this ldb use the 'global' schema, setup to avoid having multiple copies in this process
513 int dsdb_set_global_schema(struct ldb_context
*ldb
)
516 void *use_global_schema
= (void *)1;
517 if (!global_schema
) {
520 ret
= ldb_set_opaque(ldb
, "dsdb_use_global_schema", use_global_schema
);
521 if (ret
!= LDB_SUCCESS
) {
525 /* Set the new attributes based on the new schema */
526 ret
= dsdb_schema_set_attributes(ldb
, global_schema
, false /* Don't write attributes, it's expensive */);
527 if (ret
== LDB_SUCCESS
) {
528 /* Keep a reference to this schema, just in case the original copy is replaced */
529 if (talloc_reference(ldb
, global_schema
) == NULL
) {
537 bool dsdb_uses_global_schema(struct ldb_context
*ldb
)
539 return (ldb_get_opaque(ldb
, "dsdb_use_global_schema") != NULL
);
543 * Find the schema object for this ldb
545 * If reference_ctx is not NULL, then talloc_reference onto that context
548 struct dsdb_schema
*dsdb_get_schema(struct ldb_context
*ldb
, TALLOC_CTX
*reference_ctx
)
551 struct dsdb_schema
*schema_out
;
552 struct dsdb_schema
*schema_in
;
553 bool use_global_schema
;
554 TALLOC_CTX
*tmp_ctx
= talloc_new(reference_ctx
);
559 /* see if we have a cached copy */
560 use_global_schema
= dsdb_uses_global_schema(ldb
);
561 if (use_global_schema
) {
562 schema_in
= global_schema
;
564 p
= ldb_get_opaque(ldb
, "dsdb_schema");
566 schema_in
= talloc_get_type(p
, struct dsdb_schema
);
568 talloc_free(tmp_ctx
);
573 if (schema_in
->refresh_fn
&& !schema_in
->refresh_in_progress
) {
574 if (!talloc_reference(tmp_ctx
, schema_in
)) {
576 * ensure that the schema_in->refresh_in_progress
577 * remains valid for the right amount of time
579 talloc_free(tmp_ctx
);
582 schema_in
->refresh_in_progress
= true;
583 /* This may change schema, if it needs to reload it from disk */
584 schema_out
= schema_in
->refresh_fn(schema_in
->loaded_from_module
,
587 schema_in
->refresh_in_progress
= false;
589 schema_out
= schema_in
;
592 /* This removes the extra reference above */
593 talloc_free(tmp_ctx
);
594 if (!reference_ctx
) {
597 return talloc_reference(reference_ctx
, schema_out
);
602 * Make the schema found on this ldb the 'global' schema
605 void dsdb_make_schema_global(struct ldb_context
*ldb
, struct dsdb_schema
*schema
)
612 talloc_unlink(talloc_autofree_context(), global_schema
);
615 /* we want the schema to be around permanently */
616 talloc_reparent(ldb
, talloc_autofree_context(), schema
);
617 global_schema
= schema
;
619 /* This calls the talloc_reference() of the global schema back onto the ldb */
620 dsdb_set_global_schema(ldb
);
624 * When loading the schema from LDIF files, we don't get the extended DNs.
626 * We need to set these up, so that from the moment we start the provision,
627 * the defaultObjectCategory links are set up correctly.
629 int dsdb_schema_fill_extended_dn(struct ldb_context
*ldb
, struct dsdb_schema
*schema
)
631 struct dsdb_class
*cur
;
632 const struct dsdb_class
*target_class
;
633 for (cur
= schema
->classes
; cur
; cur
= cur
->next
) {
634 const struct ldb_val
*rdn
;
637 struct ldb_dn
*dn
= ldb_dn_new(NULL
, ldb
, cur
->defaultObjectCategory
);
640 return LDB_ERR_INVALID_DN_SYNTAX
;
642 rdn
= ldb_dn_get_component_val(dn
, 0);
645 return LDB_ERR_INVALID_DN_SYNTAX
;
647 target_class
= dsdb_class_by_cn_ldb_val(schema
, rdn
);
650 return LDB_ERR_CONSTRAINT_VIOLATION
;
653 status
= GUID_to_ndr_blob(&target_class
->objectGUID
, dn
, &guid
);
654 if (!NT_STATUS_IS_OK(status
)) {
656 return ldb_operr(ldb
);
658 ldb_dn_set_extended_component(dn
, "GUID", &guid
);
660 cur
->defaultObjectCategory
= ldb_dn_get_extended_linearized(cur
, dn
, 1);
667 * Add an element to the schema (attribute or class) from an LDB message
669 WERROR
dsdb_schema_set_el_from_ldb_msg(struct ldb_context
*ldb
, struct dsdb_schema
*schema
,
670 struct ldb_message
*msg
)
672 if (samdb_find_attribute(ldb
, msg
,
673 "objectclass", "attributeSchema") != NULL
) {
674 return dsdb_attribute_from_ldb(ldb
, schema
, msg
);
675 } else if (samdb_find_attribute(ldb
, msg
,
676 "objectclass", "classSchema") != NULL
) {
677 return dsdb_class_from_ldb(schema
, msg
);
680 /* Don't fail on things not classes or attributes */
685 * Rather than read a schema from the LDB itself, read it from an ldif
686 * file. This allows schema to be loaded and used while adding the
687 * schema itself to the directory.
690 WERROR
dsdb_set_schema_from_ldif(struct ldb_context
*ldb
, const char *pf
, const char *df
)
692 struct ldb_ldif
*ldif
;
693 struct ldb_message
*msg
;
697 struct dsdb_schema
*schema
;
698 const struct ldb_val
*prefix_val
;
699 const struct ldb_val
*info_val
;
700 struct ldb_val info_val_default
;
703 mem_ctx
= talloc_new(ldb
);
708 schema
= dsdb_new_schema(mem_ctx
);
710 schema
->fsmo
.we_are_master
= true;
711 schema
->fsmo
.master_dn
= ldb_dn_new_fmt(schema
, ldb
, "@PROVISION_SCHEMA_MASTER");
712 if (!schema
->fsmo
.master_dn
) {
717 * load the prefixMap attribute from pf
719 ldif
= ldb_ldif_read_string(ldb
, &pf
);
721 status
= WERR_INVALID_PARAM
;
724 talloc_steal(mem_ctx
, ldif
);
726 ret
= ldb_msg_normalize(ldb
, mem_ctx
, ldif
->msg
, &msg
);
727 if (ret
!= LDB_SUCCESS
) {
732 prefix_val
= ldb_msg_find_ldb_val(msg
, "prefixMap");
734 status
= WERR_INVALID_PARAM
;
738 info_val
= ldb_msg_find_ldb_val(msg
, "schemaInfo");
740 status
= dsdb_schema_info_blob_new(mem_ctx
, &info_val_default
);
741 W_ERROR_NOT_OK_GOTO(status
, failed
);
742 info_val
= &info_val_default
;
745 status
= dsdb_load_oid_mappings_ldb(schema
, prefix_val
, info_val
);
746 if (!W_ERROR_IS_OK(status
)) {
747 DEBUG(0,("ERROR: dsdb_load_oid_mappings_ldb() failed with %s\n", win_errstr(status
)));
751 /* load the attribute and class definitions out of df */
752 while ((ldif
= ldb_ldif_read_string(ldb
, &df
))) {
753 talloc_steal(mem_ctx
, ldif
);
755 ret
= ldb_msg_normalize(ldb
, ldif
, ldif
->msg
, &msg
);
756 if (ret
!= LDB_SUCCESS
) {
760 status
= dsdb_schema_set_el_from_ldb_msg(ldb
, schema
, msg
);
762 if (!W_ERROR_IS_OK(status
)) {
767 ret
= dsdb_set_schema(ldb
, schema
);
768 if (ret
!= LDB_SUCCESS
) {
769 status
= WERR_FOOBAR
;
773 ret
= dsdb_schema_fill_extended_dn(ldb
, schema
);
774 if (ret
!= LDB_SUCCESS
) {
775 status
= WERR_FOOBAR
;
785 talloc_free(mem_ctx
);