4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
23 * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
27 * Processes name2sid & sid2name batched lookups for a given user or
28 * computer from an AD Directory server using GSSAPI authentication
38 #include <sasl/sasl.h>
48 #include <sys/u8_textprep.h>
49 #include "libadutils.h"
50 #include "nldaputils.h"
53 /* Attribute names and filter format strings */
54 #define SAN "sAMAccountName"
55 #define OBJSID "objectSid"
56 #define OBJCLASS "objectClass"
57 #define UIDNUMBER "uidNumber"
58 #define GIDNUMBER "gidNumber"
59 #define UIDNUMBERFILTER "(&(objectclass=user)(uidNumber=%u))"
60 #define GIDNUMBERFILTER "(&(objectclass=group)(gidNumber=%u))"
61 #define SANFILTER "(sAMAccountName=%s)"
62 #define OBJSIDFILTER "(objectSid=%s)"
64 void idmap_ldap_res_search_cb(LDAP
*ld
, LDAPMessage
**res
, int rc
,
68 * A place to put the results of a batched (async) query
70 * There is one of these for every query added to a batch object
71 * (idmap_query_state, see below).
73 typedef struct idmap_q
{
75 * data used for validating search result entries for name->SID
78 char *ecanonname
; /* expected canon name */
79 char *edomain
; /* expected domain name */
80 idmap_id_type esidtype
; /* expected SID type */
82 char **canonname
; /* actual canon name */
83 char **domain
; /* name of domain of object */
84 char **sid
; /* stringified SID */
86 idmap_id_type
*sid_type
; /* user or group SID? */
87 char **unixname
; /* unixname for name mapping */
88 char **dn
; /* DN of entry */
89 char **attr
; /* Attr for name mapping */
90 char **value
; /* value for name mapping */
91 posix_id_t
*pid
; /* Posix ID found via IDMU */
94 adutils_result_t
*result
;
97 * The LDAP search entry result is placed here to be processed
98 * when the search done result is received.
100 LDAPMessage
*search_res
; /* The LDAP search result */
103 /* Batch context structure; typedef is in header file */
104 struct idmap_query_state
{
105 adutils_query_state_t
*qs
;
106 int qsize
; /* Queue size */
107 uint32_t qcount
; /* Number of queued requests */
108 const char *ad_unixuser_attr
;
109 const char *ad_unixgroup_attr
;
110 int directory_based_mapping
; /* enum */
111 char *default_domain
;
112 idmap_q_t queries
[1]; /* array of query results */
115 static pthread_t reaperid
= 0;
118 * Keep connection management simple for now, extend or replace later
119 * with updated libsldap code.
121 #define ADREAPERSLEEP 60
124 * Idle connection reaping side of connection management
126 * Every minute wake up and look for connections that have been idle for
127 * five minutes or more and close them.
136 ts
.tv_sec
= ADREAPERSLEEP
;
141 * nanosleep(3RT) is thead-safe (no SIGALRM) and more
142 * portable than usleep(3C)
144 (void) nanosleep(&ts
, NULL
);
145 adutils_reap_idle_connections();
150 * Take ad_host_config_t information, create a ad_host_t,
151 * populate it and add it to the list of hosts.
155 idmap_add_ds(adutils_ad_t
*ad
, const char *host
, int port
)
159 if (adutils_add_ds(ad
, host
, port
) == ADUTILS_SUCCESS
)
162 /* Start reaper if it doesn't exist */
163 if (ret
== 0 && reaperid
== 0)
164 (void) pthread_create(&reaperid
, NULL
,
165 (void *(*)(void *))adreaper
, NULL
);
171 map_adrc2idmaprc(adutils_rc adrc
)
174 case ADUTILS_SUCCESS
:
175 return (IDMAP_SUCCESS
);
176 case ADUTILS_ERR_NOTFOUND
:
177 return (IDMAP_ERR_NOTFOUND
);
178 case ADUTILS_ERR_MEMORY
:
179 return (IDMAP_ERR_MEMORY
);
180 case ADUTILS_ERR_DOMAIN
:
181 return (IDMAP_ERR_DOMAIN
);
182 case ADUTILS_ERR_OTHER
:
183 return (IDMAP_ERR_OTHER
);
184 case ADUTILS_ERR_RETRIABLE_NET_ERR
:
185 return (IDMAP_ERR_RETRIABLE_NET_ERR
);
187 return (IDMAP_ERR_INTERNAL
);
193 idmap_lookup_batch_start(adutils_ad_t
*ad
, int nqueries
,
194 int directory_based_mapping
, const char *default_domain
,
195 idmap_query_state_t
**state
)
197 idmap_query_state_t
*new_state
;
204 new_state
= calloc(1, sizeof (idmap_query_state_t
) +
205 (nqueries
- 1) * sizeof (idmap_q_t
));
206 if (new_state
== NULL
)
207 return (IDMAP_ERR_MEMORY
);
209 if ((rc
= adutils_lookup_batch_start(ad
, nqueries
,
210 idmap_ldap_res_search_cb
, new_state
, &new_state
->qs
))
211 != ADUTILS_SUCCESS
) {
212 idmap_lookup_release_batch(&new_state
);
213 return (map_adrc2idmaprc(rc
));
216 new_state
->default_domain
= strdup(default_domain
);
217 if (new_state
->default_domain
== NULL
) {
218 idmap_lookup_release_batch(&new_state
);
219 return (IDMAP_ERR_MEMORY
);
222 new_state
->directory_based_mapping
= directory_based_mapping
;
223 new_state
->qsize
= nqueries
;
225 return (IDMAP_SUCCESS
);
229 * Set unixuser_attr and unixgroup_attr for AD-based name mapping
232 idmap_lookup_batch_set_unixattr(idmap_query_state_t
*state
,
233 const char *unixuser_attr
, const char *unixgroup_attr
)
235 state
->ad_unixuser_attr
= unixuser_attr
;
236 state
->ad_unixgroup_attr
= unixgroup_attr
;
240 * Take parsed attribute values from a search result entry and check if
241 * it is the result that was desired and, if so, set the result fields
242 * of the given idmap_q_t.
244 * Except for dn and attr, all strings are consumed, either by transferring
245 * them over into the request results (where the caller will eventually free
246 * them) or by freeing them here. Note that this aligns with the "const"
247 * declarations below.
268 if ((domain
= adutils_dn2dns(dn
)) == NULL
)
271 if (q
->ecanonname
!= NULL
&& san
!= NULL
) {
272 /* Check that this is the canonname that we were looking for */
273 if (u8_strcmp(q
->ecanonname
, san
, 0,
274 U8_STRCMP_CI_LOWER
, /* no normalization, for now */
275 U8_UNICODE_LATEST
, &err1
) != 0 || err1
!= 0)
279 if (q
->edomain
!= NULL
) {
280 /* Check that this is the domain that we were looking for */
281 if (!domain_eq(q
->edomain
, domain
))
285 /* Copy the DN and attr and value */
289 if (q
->attr
!= NULL
&& attr
!= NULL
)
290 *q
->attr
= strdup(attr
);
292 if (q
->value
!= NULL
&& value
!= NULL
) {
305 *q
->sid_type
= sid_type
;
307 *q
->unixname
= unixname
;
310 if (q
->domain
!= NULL
) {
314 if (q
->canonname
!= NULL
) {
316 * The caller may be replacing the given winname by its
317 * canonical name and therefore free any old name before
318 * overwriting the field by the canonical name.
325 if (q
->pid
!= NULL
&& pid
!= IDMAP_SENTINEL_PID
) {
329 q
->ad_rc
= ADUTILS_SUCCESS
;
332 /* Free unused attribute values */
340 #define BVAL_CASEEQ(bv, str) \
341 (((*(bv))->bv_len == (sizeof (str) - 1)) && \
342 strncasecmp((*(bv))->bv_val, str, (*(bv))->bv_len) == 0)
345 * Extract the class of the result entry. Returns 1 on success, 0 on
350 idmap_bv_objclass2sidtype(BerValue
**bvalues
, int *sid_type
)
354 *sid_type
= IDMAP_SID
;
359 * We consider Computer to be a subclass of User, so we can just
360 * ignore Computer entries and pay attention to the accompanying
363 for (cbval
= bvalues
; *cbval
!= NULL
; cbval
++) {
364 if (BVAL_CASEEQ(cbval
, "group")) {
365 *sid_type
= IDMAP_GSID
;
367 } else if (BVAL_CASEEQ(cbval
, "user")) {
368 *sid_type
= IDMAP_USID
;
372 * "else if (*sid_type = IDMAP_USID)" then this is a
373 * new sub-class of user -- what to do with it??
381 * Handle a given search result entry
385 idmap_extract_object(idmap_query_state_t
*state
, idmap_q_t
*q
,
386 LDAPMessage
*res
, LDAP
*ld
)
389 const char *attr
= NULL
;
391 char *unix_name
= NULL
;
398 posix_id_t pid
= IDMAP_SENTINEL_PID
;
400 assert(q
->rc
!= NULL
);
401 assert(q
->domain
== NULL
|| *q
->domain
== NULL
);
403 if ((dn
= ldap_get_dn(ld
, res
)) == NULL
)
406 bvalues
= ldap_get_values_len(ld
, res
, OBJCLASS
);
407 if (bvalues
== NULL
) {
409 * Didn't find objectclass. Something's wrong with our
412 idmapdlog(LOG_ERR
, "%s has no %s", dn
, OBJCLASS
);
415 ok
= idmap_bv_objclass2sidtype(bvalues
, &sid_type
);
416 ldap_value_free_len(bvalues
);
419 * Didn't understand objectclass. Something's wrong with our
422 idmapdlog(LOG_ERR
, "%s has unexpected %s", dn
, OBJCLASS
);
426 if (state
->directory_based_mapping
== DIRECTORY_MAPPING_IDMU
&&
428 if (sid_type
== IDMAP_USID
)
430 else if (sid_type
== IDMAP_GSID
)
433 bvalues
= ldap_get_values_len(ld
, res
, attr
);
434 if (bvalues
!= NULL
) {
435 value
= adutils_bv_str(bvalues
[0]);
436 if (!adutils_bv_uint(bvalues
[0], &pid
)) {
438 "%s has Invalid %s value \"%s\"",
441 ldap_value_free_len(bvalues
);
446 if (state
->directory_based_mapping
== DIRECTORY_MAPPING_NAME
&&
447 q
->unixname
!= NULL
) {
449 * If the caller has requested unixname then determine the
450 * AD attribute name that will have the unixname, and retrieve
453 idmap_id_type esidtype
;
455 * Determine the target type.
457 * If the caller specified one, use that. Otherwise, give the
458 * same type that as we found for the Windows user.
460 esidtype
= q
->esidtype
;
461 if (esidtype
== IDMAP_SID
)
464 if (esidtype
== IDMAP_USID
)
465 attr
= state
->ad_unixuser_attr
;
466 else if (esidtype
== IDMAP_GSID
)
467 attr
= state
->ad_unixgroup_attr
;
470 bvalues
= ldap_get_values_len(ld
, res
, attr
);
471 if (bvalues
!= NULL
) {
472 unix_name
= adutils_bv_str(bvalues
[0]);
473 ldap_value_free_len(bvalues
);
474 value
= strdup(unix_name
);
479 bvalues
= ldap_get_values_len(ld
, res
, SAN
);
480 if (bvalues
!= NULL
) {
481 san
= adutils_bv_str(bvalues
[0]);
482 ldap_value_free_len(bvalues
);
485 if (q
->sid
!= NULL
) {
486 bvalues
= ldap_get_values_len(ld
, res
, OBJSID
);
487 if (bvalues
!= NULL
) {
488 sid
= adutils_bv_objsid2sidstr(bvalues
[0], &rid
);
489 ldap_value_free_len(bvalues
);
493 idmap_setqresults(q
, san
, dn
,
503 idmap_ldap_res_search_cb(LDAP
*ld
, LDAPMessage
**res
, int rc
, int qid
,
506 idmap_query_state_t
*state
= (idmap_query_state_t
*)argp
;
507 idmap_q_t
*q
= &(state
->queries
[qid
]);
510 case LDAP_RES_SEARCH_RESULT
:
511 if (q
->search_res
!= NULL
) {
512 idmap_extract_object(state
, q
, q
->search_res
, ld
);
513 (void) ldap_msgfree(q
->search_res
);
514 q
->search_res
= NULL
;
516 q
->ad_rc
= ADUTILS_ERR_NOTFOUND
;
518 case LDAP_RES_SEARCH_ENTRY
:
519 if (q
->search_res
== NULL
) {
520 q
->search_res
= *res
;
531 idmap_cleanup_batch(idmap_query_state_t
*batch
)
535 for (i
= 0; i
< batch
->qcount
; i
++) {
536 free(batch
->queries
[i
].ecanonname
);
537 batch
->queries
[i
].ecanonname
= NULL
;
538 free(batch
->queries
[i
].edomain
);
539 batch
->queries
[i
].edomain
= NULL
;
544 * This routine frees the idmap_query_state_t structure
547 idmap_lookup_release_batch(idmap_query_state_t
**state
)
549 if (state
== NULL
|| *state
== NULL
)
551 adutils_lookup_batch_release(&(*state
)->qs
);
552 idmap_cleanup_batch(*state
);
553 free((*state
)->default_domain
);
559 idmap_lookup_batch_end(idmap_query_state_t
**state
)
563 idmap_query_state_t
*id_qs
= *state
;
565 ad_rc
= adutils_lookup_batch_end(&id_qs
->qs
);
568 * Map adutils rc to idmap_retcode in each
569 * query because consumers in dbutils.c
570 * expects idmap_retcode.
572 for (i
= 0; i
< id_qs
->qcount
; i
++) {
573 *id_qs
->queries
[i
].rc
=
574 map_adrc2idmaprc(id_qs
->queries
[i
].ad_rc
);
576 idmap_lookup_release_batch(state
);
577 return (map_adrc2idmaprc(ad_rc
));
581 * Send one prepared search, queue up msgid, process what results are
586 idmap_batch_add1(idmap_query_state_t
*state
, const char *filter
,
587 char *ecanonname
, char *edomain
, idmap_id_type esidtype
,
588 char **dn
, char **attr
, char **value
,
589 char **canonname
, char **dname
,
590 char **sid
, rid_t
*rid
, idmap_id_type
*sid_type
, char **unixname
,
597 char *attrs
[20]; /* Plenty */
599 qid
= atomic_inc_32_nv(&state
->qcount
) - 1;
600 q
= &(state
->queries
[qid
]);
602 assert(qid
< state
->qsize
);
605 * Remember the expected canonname, domainname and unix type
606 * so we can check the results * against it
608 q
->ecanonname
= ecanonname
;
609 q
->edomain
= edomain
;
610 q
->esidtype
= esidtype
;
612 /* Remember where to put the results */
613 q
->canonname
= canonname
;
617 q
->sid_type
= sid_type
;
619 q
->unixname
= unixname
;
625 /* Add attributes that are not always needed */
629 attrs
[i
++] = OBJCLASS
;
631 if (unixname
!= NULL
) {
632 /* Add unixuser/unixgroup attribute names to the attrs list */
633 if (esidtype
!= IDMAP_GSID
&&
634 state
->ad_unixuser_attr
!= NULL
)
635 attrs
[i
++] = (char *)state
->ad_unixuser_attr
;
636 if (esidtype
!= IDMAP_USID
&&
637 state
->ad_unixgroup_attr
!= NULL
)
638 attrs
[i
++] = (char *)state
->ad_unixgroup_attr
;
642 if (esidtype
!= IDMAP_GSID
)
643 attrs
[i
++] = UIDNUMBER
;
644 if (esidtype
!= IDMAP_USID
)
645 attrs
[i
++] = GIDNUMBER
;
651 * Provide sane defaults for the results in case we never hear
652 * back from the DS before closing the connection.
654 * In particular we default the result to indicate a retriable
655 * error. The first complete matching result entry will cause
656 * this to be set to IDMAP_SUCCESS, and the end of the results
657 * for this search will cause this to indicate "not found" if no
658 * result entries arrived or no complete ones matched the lookup
661 *rc
= IDMAP_ERR_RETRIABLE_NET_ERR
;
662 if (sid_type
!= NULL
)
663 *sid_type
= IDMAP_SID
;
678 * Don't set *canonname to NULL because it may be pointing to the
679 * given winname. Later on if we get a canonical name from AD the
680 * old name if any will be freed before assigning the new name.
684 * Invoke the mother of all APIs i.e. the adutils API
686 ad_rc
= adutils_lookup_batch_add(state
->qs
, filter
,
687 (const char **)attrs
,
688 edomain
, &q
->result
, &q
->ad_rc
);
689 return (map_adrc2idmaprc(ad_rc
));
693 idmap_name2sid_batch_add1(idmap_query_state_t
*state
,
694 const char *name
, const char *dname
, idmap_id_type esidtype
,
695 char **dn
, char **attr
, char **value
,
696 char **canonname
, char **sid
, rid_t
*rid
,
697 idmap_id_type
*sid_type
, char **unixname
,
698 posix_id_t
*pid
, idmap_retcode
*rc
)
700 idmap_retcode retcode
;
701 char *filter
, *s_name
;
702 char *ecanonname
, *edomain
; /* expected canonname */
705 * Strategy: search the global catalog for user/group by
706 * sAMAccountName = user/groupname with "" as the base DN and by
707 * userPrincipalName = user/groupname@domain. The result
708 * entries will be checked to conform to the name and domain
709 * name given here. The DN, sAMAccountName, userPrincipalName,
710 * objectSid and objectClass of the result entries are all we
711 * need to figure out which entries match the lookup, the SID of
712 * the user/group and whether it is a user or a group.
715 if ((ecanonname
= strdup(name
)) == NULL
)
716 return (IDMAP_ERR_MEMORY
);
718 if (dname
== NULL
|| *dname
== '\0') {
719 /* 'name' not qualified and dname not given */
720 dname
= state
->default_domain
;
721 edomain
= strdup(dname
);
722 if (edomain
== NULL
) {
724 return (IDMAP_ERR_MEMORY
);
727 if ((edomain
= strdup(dname
)) == NULL
) {
729 return (IDMAP_ERR_MEMORY
);
733 if (!adutils_lookup_check_domain(state
->qs
, dname
)) {
736 return (IDMAP_ERR_DOMAIN_NOTFOUND
);
739 s_name
= sanitize_for_ldap_filter(name
);
740 if (s_name
== NULL
) {
743 return (IDMAP_ERR_MEMORY
);
746 /* Assemble filter */
747 (void) asprintf(&filter
, SANFILTER
, s_name
);
750 if (filter
== NULL
) {
753 return (IDMAP_ERR_MEMORY
);
756 retcode
= idmap_batch_add1(state
, filter
, ecanonname
, edomain
,
757 esidtype
, dn
, attr
, value
, canonname
, NULL
, sid
, rid
, sid_type
,
766 idmap_sid2name_batch_add1(idmap_query_state_t
*state
,
767 const char *sid
, const rid_t
*rid
, idmap_id_type esidtype
,
768 char **dn
, char **attr
, char **value
,
769 char **name
, char **dname
, idmap_id_type
*sid_type
,
770 char **unixname
, posix_id_t
*pid
, idmap_retcode
*rc
)
772 idmap_retcode retcode
;
775 char cbinsid
[ADUTILS_MAXHEXBINSID
+ 1];
778 * Strategy: search [the global catalog] for user/group by
779 * objectSid = SID with empty base DN. The DN, sAMAccountName
780 * and objectClass of the result are all we need to figure out
781 * the name of the SID and whether it is a user, a group or a
785 if (!adutils_lookup_check_sid_prefix(state
->qs
, sid
))
786 return (IDMAP_ERR_DOMAIN_NOTFOUND
);
788 ret
= adutils_txtsid2hexbinsid(sid
, rid
, &cbinsid
[0], sizeof (cbinsid
));
790 return (IDMAP_ERR_SID
);
792 /* Assemble filter */
793 (void) asprintf(&filter
, OBJSIDFILTER
, cbinsid
);
795 return (IDMAP_ERR_MEMORY
);
797 retcode
= idmap_batch_add1(state
, filter
, NULL
, NULL
, esidtype
,
798 dn
, attr
, value
, name
, dname
, NULL
, NULL
, sid_type
, unixname
,
807 idmap_unixname2sid_batch_add1(idmap_query_state_t
*state
,
808 const char *unixname
, int is_user
, int is_wuser
,
809 char **dn
, char **attr
, char **value
,
810 char **sid
, rid_t
*rid
, char **name
,
811 char **dname
, idmap_id_type
*sid_type
, idmap_retcode
*rc
)
813 idmap_retcode retcode
;
814 char *filter
, *s_unixname
;
815 const char *attrname
;
817 /* Get unixuser or unixgroup AD attribute name */
818 attrname
= (is_user
) ?
819 state
->ad_unixuser_attr
: state
->ad_unixgroup_attr
;
820 if (attrname
== NULL
)
821 return (IDMAP_ERR_NOTFOUND
);
823 s_unixname
= sanitize_for_ldap_filter(unixname
);
824 if (s_unixname
== NULL
)
825 return (IDMAP_ERR_MEMORY
);
827 /* Assemble filter */
828 (void) asprintf(&filter
, "(&(objectclass=%s)(%s=%s))",
829 is_wuser
? "user" : "group", attrname
, s_unixname
);
830 if (s_unixname
!= unixname
)
832 if (filter
== NULL
) {
833 return (IDMAP_ERR_MEMORY
);
836 retcode
= idmap_batch_add1(state
, filter
, NULL
, NULL
,
837 IDMAP_POSIXID
, dn
, NULL
, NULL
, name
, dname
, sid
, rid
, sid_type
,
840 if (retcode
== IDMAP_SUCCESS
&& attr
!= NULL
) {
841 if ((*attr
= strdup(attrname
)) == NULL
)
842 retcode
= IDMAP_ERR_MEMORY
;
845 if (retcode
== IDMAP_SUCCESS
&& value
!= NULL
) {
846 if ((*value
= strdup(unixname
)) == NULL
)
847 retcode
= IDMAP_ERR_MEMORY
;
856 idmap_pid2sid_batch_add1(idmap_query_state_t
*state
,
857 posix_id_t pid
, int is_user
,
858 char **dn
, char **attr
, char **value
,
859 char **sid
, rid_t
*rid
, char **name
,
860 char **dname
, idmap_id_type
*sid_type
, idmap_retcode
*rc
)
862 idmap_retcode retcode
;
864 const char *attrname
;
866 /* Assemble filter */
868 (void) asprintf(&filter
, UIDNUMBERFILTER
, pid
);
869 attrname
= UIDNUMBER
;
871 (void) asprintf(&filter
, GIDNUMBERFILTER
, pid
);
872 attrname
= GIDNUMBER
;
875 return (IDMAP_ERR_MEMORY
);
877 retcode
= idmap_batch_add1(state
, filter
, NULL
, NULL
,
878 IDMAP_POSIXID
, dn
, NULL
, NULL
, name
, dname
, sid
, rid
, sid_type
,
881 if (retcode
== IDMAP_SUCCESS
&& attr
!= NULL
) {
882 if ((*attr
= strdup(attrname
)) == NULL
)
883 retcode
= IDMAP_ERR_MEMORY
;
886 if (retcode
== IDMAP_SUCCESS
&& value
!= NULL
) {
887 (void) asprintf(value
, "%u", pid
);
889 retcode
= IDMAP_ERR_MEMORY
;