* Sync up to trunk head (r60691).
[reactos.git] / dll / win32 / wldap32 / init.c
1 /*
2 * WLDAP32 - LDAP support for Wine
3 *
4 * Copyright 2005 Hans Leidekker
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library 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 GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20
21 #include <config.h>
22 //#include "wine/port.h"
23
24 //#include <stdio.h>
25 //#include <stdarg.h>
26 #ifdef HAVE_LDAP_H
27 #include <ldap.h>
28 #endif
29
30 #include <windef.h>
31 //#include "winbase.h"
32 //#include "winnls.h"
33
34 #include "winldap_private.h"
35 //#include "wldap32.h"
36 #include <wine/debug.h>
37
38 #ifdef HAVE_LDAP
39 /* Should eventually be determined by the algorithm documented on MSDN. */
40 static const WCHAR defaulthost[] = { 'l','o','c','a','l','h','o','s','t',0 };
41
42 /* Split a space separated string of hostnames into a string array */
43 static char **split_hostnames( const char *hostnames )
44 {
45 char **res, *str, *p, *q;
46 unsigned int i = 0;
47
48 str = strdupU( hostnames );
49 if (!str) return NULL;
50
51 p = str;
52 while (isspace( *p )) p++;
53 if (*p) i++;
54
55 while (*p)
56 {
57 if (isspace( *p ))
58 {
59 while (isspace( *p )) p++;
60 if (*p) i++;
61 }
62 p++;
63 }
64
65 res = HeapAlloc( GetProcessHeap(), 0, (i + 1) * sizeof(char *) );
66 if (!res)
67 {
68 HeapFree( GetProcessHeap(), 0, str );
69 return NULL;
70 }
71
72 p = str;
73 while (isspace( *p )) p++;
74
75 q = p;
76 i = 0;
77
78 while (*p)
79 {
80 if (p[1] != '\0')
81 {
82 if (isspace( *p ))
83 {
84 *p = '\0'; p++;
85 res[i] = strdupU( q );
86 if (!res[i]) goto oom;
87 i++;
88
89 while (isspace( *p )) p++;
90 q = p;
91 }
92 }
93 else
94 {
95 res[i] = strdupU( q );
96 if (!res[i]) goto oom;
97 i++;
98 }
99 p++;
100 }
101 res[i] = NULL;
102
103 HeapFree( GetProcessHeap(), 0, str );
104 return res;
105
106 oom:
107 while (i > 0) strfreeU( res[--i] );
108
109 HeapFree( GetProcessHeap(), 0, res );
110 HeapFree( GetProcessHeap(), 0, str );
111
112 return NULL;
113 }
114
115 /* Determine if a URL starts with a known LDAP scheme */
116 static int has_ldap_scheme( char *url )
117 {
118 if (!strncasecmp( url, "ldap://", 7 ) ||
119 !strncasecmp( url, "ldaps://", 8 ) ||
120 !strncasecmp( url, "ldapi://", 8 ) ||
121 !strncasecmp( url, "cldap://", 8 )) return 1;
122 return 0;
123 }
124
125 /* Flatten an array of hostnames into a space separated string of URLs.
126 * Prepend a given scheme and append a given port number to each hostname
127 * if necessary.
128 */
129 static char *join_hostnames( const char *scheme, char **hostnames, ULONG portnumber )
130 {
131 char *res, *p, *q, **v;
132 unsigned int i = 0, size = 0;
133 static const char sep[] = " ", fmt[] = ":%d";
134 char port[7];
135
136 sprintf( port, fmt, portnumber );
137
138 for (v = hostnames; *v; v++)
139 {
140 if (!has_ldap_scheme( *v ))
141 {
142 size += strlen( scheme );
143 q = *v;
144 }
145 else
146 /* skip past colon in scheme prefix */
147 q = strchr( *v, '/' );
148
149 size += strlen( *v );
150
151 if (!strchr( q, ':' ))
152 size += strlen( port );
153
154 i++;
155 }
156
157 size += (i - 1) * strlen( sep );
158
159 res = HeapAlloc( GetProcessHeap(), 0, size + 1 );
160 if (!res) return NULL;
161
162 p = res;
163 for (v = hostnames; *v; v++)
164 {
165 if (v != hostnames)
166 {
167 strcpy( p, sep );
168 p += strlen( sep );
169 }
170
171 if (!has_ldap_scheme( *v ))
172 {
173 strcpy( p, scheme );
174 p += strlen( scheme );
175 q = *v;
176 }
177 else
178 /* skip past colon in scheme prefix */
179 q = strchr( *v, '/' );
180
181 strcpy( p, *v );
182 p += strlen( *v );
183
184 if (!strchr( q, ':' ))
185 {
186 strcpy( p, port );
187 p += strlen( port );
188 }
189 }
190 return res;
191 }
192
193 static char *urlify_hostnames( const char *scheme, char *hostnames, ULONG port )
194 {
195 char *url = NULL, **strarray;
196
197 strarray = split_hostnames( hostnames );
198 if (strarray)
199 url = join_hostnames( scheme, strarray, port );
200 else
201 return NULL;
202
203 strarrayfreeU( strarray );
204 return url;
205 }
206 #endif
207
208 WINE_DEFAULT_DEBUG_CHANNEL(wldap32);
209
210 /***********************************************************************
211 * cldap_openA (WLDAP32.@)
212 *
213 * See cldap_openW.
214 */
215 WLDAP32_LDAP * CDECL cldap_openA( PCHAR hostname, ULONG portnumber )
216 {
217 #ifdef HAVE_LDAP
218 WLDAP32_LDAP *ld = NULL;
219 WCHAR *hostnameW = NULL;
220
221 TRACE( "(%s, %d)\n", debugstr_a(hostname), portnumber );
222
223 if (hostname) {
224 hostnameW = strAtoW( hostname );
225 if (!hostnameW) goto exit;
226 }
227
228 ld = cldap_openW( hostnameW, portnumber );
229
230 exit:
231 strfreeW( hostnameW );
232 return ld;
233
234 #else
235 return NULL;
236 #endif
237 }
238
239 /***********************************************************************
240 * cldap_openW (WLDAP32.@)
241 *
242 * Initialize an LDAP context and create a UDP connection.
243 *
244 * PARAMS
245 * hostname [I] Name of the host to connect to.
246 * portnumber [I] Port number to use.
247 *
248 * RETURNS
249 * Success: Pointer to an LDAP context.
250 * Failure: NULL
251 *
252 * NOTES
253 * The hostname string can be a space separated string of hostnames,
254 * in which case the LDAP runtime will try to connect to the hosts
255 * in order, until a connection can be made. A hostname may have a
256 * trailing port number (separated from the hostname by a ':'), which
257 * will take precedence over the port number supplied as a parameter
258 * to this function.
259 */
260 WLDAP32_LDAP * CDECL cldap_openW( PWCHAR hostname, ULONG portnumber )
261 {
262 #ifdef HAVE_LDAP
263 LDAP *ld = NULL;
264 char *hostnameU = NULL, *url = NULL;
265
266 TRACE( "(%s, %d)\n", debugstr_w(hostname), portnumber );
267
268 if (hostname) {
269 hostnameU = strWtoU( hostname );
270 if (!hostnameU) goto exit;
271 }
272 else {
273 hostnameU = strWtoU( defaulthost );
274 if (!hostnameU) goto exit;
275 }
276
277 url = urlify_hostnames( "cldap://", hostnameU, portnumber );
278 if (!url) goto exit;
279
280 ldap_initialize( &ld, url );
281
282 exit:
283 strfreeU( hostnameU );
284 strfreeU( url );
285 return ld;
286
287 #else
288 return NULL;
289 #endif
290 }
291
292 /***********************************************************************
293 * ldap_connect (WLDAP32.@)
294 *
295 * Connect to an LDAP server.
296 *
297 * PARAMS
298 * ld [I] Pointer to an LDAP context.
299 * timeout [I] Pointer to an l_timeval structure specifying the
300 * timeout in seconds.
301 *
302 * RETURNS
303 * Success: LDAP_SUCCESS
304 * Failure: An LDAP error code.
305 *
306 * NOTES
307 * The timeout parameter may be NULL in which case a default timeout
308 * value will be used.
309 */
310 ULONG CDECL ldap_connect( WLDAP32_LDAP *ld, struct l_timeval *timeout )
311 {
312 TRACE( "(%p, %p)\n", ld, timeout );
313
314 if (!ld) return WLDAP32_LDAP_PARAM_ERROR;
315 return WLDAP32_LDAP_SUCCESS; /* FIXME: do something, e.g. ping the host */
316 }
317
318 /***********************************************************************
319 * ldap_initA (WLDAP32.@)
320 *
321 * See ldap_initW.
322 */
323 WLDAP32_LDAP * CDECL ldap_initA( PCHAR hostname, ULONG portnumber )
324 {
325 #ifdef HAVE_LDAP
326 WLDAP32_LDAP *ld = NULL;
327 WCHAR *hostnameW = NULL;
328
329 TRACE( "(%s, %d)\n", debugstr_a(hostname), portnumber );
330
331 if (hostname) {
332 hostnameW = strAtoW( hostname );
333 if (!hostnameW) goto exit;
334 }
335
336 ld = ldap_initW( hostnameW, portnumber );
337
338 exit:
339 strfreeW( hostnameW );
340 return ld;
341
342 #else
343 return NULL;
344 #endif
345 }
346
347 /***********************************************************************
348 * ldap_initW (WLDAP32.@)
349 *
350 * Initialize an LDAP context and create a TCP connection.
351 *
352 * PARAMS
353 * hostname [I] Name of the host to connect to.
354 * portnumber [I] Port number to use.
355 *
356 * RETURNS
357 * Success: Pointer to an LDAP context.
358 * Failure: NULL
359 *
360 * NOTES
361 * The hostname string can be a space separated string of hostnames,
362 * in which case the LDAP runtime will try to connect to the hosts
363 * in order, until a connection can be made. A hostname may have a
364 * trailing port number (separated from the hostname by a ':'), which
365 * will take precedence over the port number supplied as a parameter
366 * to this function. The connection will not be made until the first
367 * LDAP function that needs it is called.
368 */
369 WLDAP32_LDAP * CDECL ldap_initW( PWCHAR hostname, ULONG portnumber )
370 {
371 #ifdef HAVE_LDAP
372 LDAP *ld = NULL;
373 char *hostnameU = NULL, *url = NULL;
374
375 TRACE( "(%s, %d)\n", debugstr_w(hostname), portnumber );
376
377 if (hostname) {
378 hostnameU = strWtoU( hostname );
379 if (!hostnameU) goto exit;
380 }
381 else {
382 hostnameU = strWtoU( defaulthost );
383 if (!hostnameU) goto exit;
384 }
385
386 url = urlify_hostnames( "ldap://", hostnameU, portnumber );
387 if (!url) goto exit;
388
389 ldap_initialize( &ld, url );
390
391 exit:
392 strfreeU( hostnameU );
393 strfreeU( url );
394 return ld;
395
396 #else
397 return NULL;
398 #endif
399 }
400
401 /***********************************************************************
402 * ldap_openA (WLDAP32.@)
403 *
404 * See ldap_openW.
405 */
406 WLDAP32_LDAP * CDECL ldap_openA( PCHAR hostname, ULONG portnumber )
407 {
408 #ifdef HAVE_LDAP
409 WLDAP32_LDAP *ld = NULL;
410 WCHAR *hostnameW = NULL;
411
412 TRACE( "(%s, %d)\n", debugstr_a(hostname), portnumber );
413
414 if (hostname) {
415 hostnameW = strAtoW( hostname );
416 if (!hostnameW) goto exit;
417 }
418
419 ld = ldap_openW( hostnameW, portnumber );
420
421 exit:
422 strfreeW( hostnameW );
423 return ld;
424
425 #else
426 return NULL;
427 #endif
428 }
429
430 /***********************************************************************
431 * ldap_openW (WLDAP32.@)
432 *
433 * Initialize an LDAP context and create a TCP connection.
434 *
435 * PARAMS
436 * hostname [I] Name of the host to connect to.
437 * portnumber [I] Port number to use.
438 *
439 * RETURNS
440 * Success: Pointer to an LDAP context.
441 * Failure: NULL
442 *
443 * NOTES
444 * The hostname string can be a space separated string of hostnames,
445 * in which case the LDAP runtime will try to connect to the hosts
446 * in order, until a connection can be made. A hostname may have a
447 * trailing port number (separated from the hostname by a ':'), which
448 * will take precedence over the port number supplied as a parameter
449 * to this function.
450 */
451 WLDAP32_LDAP * CDECL ldap_openW( PWCHAR hostname, ULONG portnumber )
452 {
453 #ifdef HAVE_LDAP
454 LDAP *ld = NULL;
455 char *hostnameU = NULL, *url = NULL;
456
457 TRACE( "(%s, %d)\n", debugstr_w(hostname), portnumber );
458
459 if (hostname) {
460 hostnameU = strWtoU( hostname );
461 if (!hostnameU) goto exit;
462 }
463 else {
464 hostnameU = strWtoU( defaulthost );
465 if (!hostnameU) goto exit;
466 }
467
468 url = urlify_hostnames( "ldap://", hostnameU, portnumber );
469 if (!url) goto exit;
470
471 ldap_initialize( &ld, url );
472
473 exit:
474 strfreeU( hostnameU );
475 strfreeU( url );
476 return ld;
477
478 #else
479 return NULL;
480 #endif
481 }
482
483 /***********************************************************************
484 * ldap_sslinitA (WLDAP32.@)
485 *
486 * See ldap_sslinitW.
487 */
488 WLDAP32_LDAP * CDECL ldap_sslinitA( PCHAR hostname, ULONG portnumber, int secure )
489 {
490 #ifdef HAVE_LDAP
491 WLDAP32_LDAP *ld;
492 WCHAR *hostnameW = NULL;
493
494 TRACE( "(%s, %d, 0x%08x)\n", debugstr_a(hostname), portnumber, secure );
495
496 if (hostname) {
497 hostnameW = strAtoW( hostname );
498 if (!hostnameW) return NULL;
499 }
500
501 ld = ldap_sslinitW( hostnameW, portnumber, secure );
502
503 strfreeW( hostnameW );
504 return ld;
505
506 #else
507 return NULL;
508 #endif
509 }
510
511 /***********************************************************************
512 * ldap_sslinitW (WLDAP32.@)
513 *
514 * Initialize an LDAP context and create a secure TCP connection.
515 *
516 * PARAMS
517 * hostname [I] Name of the host to connect to.
518 * portnumber [I] Port number to use.
519 * secure [I] Ask the server to create an SSL connection.
520 *
521 * RETURNS
522 * Success: Pointer to an LDAP context.
523 * Failure: NULL
524 *
525 * NOTES
526 * The hostname string can be a space separated string of hostnames,
527 * in which case the LDAP runtime will try to connect to the hosts
528 * in order, until a connection can be made. A hostname may have a
529 * trailing port number (separated from the hostname by a ':'), which
530 * will take precedence over the port number supplied as a parameter
531 * to this function. The connection will not be made until the first
532 * LDAP function that needs it is called.
533 */
534 WLDAP32_LDAP * CDECL ldap_sslinitW( PWCHAR hostname, ULONG portnumber, int secure )
535 {
536 #ifdef HAVE_LDAP
537 WLDAP32_LDAP *ld = NULL;
538 char *hostnameU = NULL, *url = NULL;
539
540 TRACE( "(%s, %d, 0x%08x)\n", debugstr_w(hostname), portnumber, secure );
541
542 if (hostname) {
543 hostnameU = strWtoU( hostname );
544 if (!hostnameU) goto exit;
545 }
546 else {
547 hostnameU = strWtoU( defaulthost );
548 if (!hostnameU) goto exit;
549 }
550
551 if (secure)
552 url = urlify_hostnames( "ldaps://", hostnameU, portnumber );
553 else
554 url = urlify_hostnames( "ldap://", hostnameU, portnumber );
555
556 if (!url) goto exit;
557 ldap_initialize( &ld, url );
558
559 exit:
560 strfreeU( hostnameU );
561 strfreeU( url );
562 return ld;
563
564 #else
565 return NULL;
566 #endif
567 }
568
569 /***********************************************************************
570 * ldap_start_tls_sA (WLDAP32.@)
571 *
572 * See ldap_start_tls_sW.
573 */
574 ULONG CDECL ldap_start_tls_sA( WLDAP32_LDAP *ld, PULONG retval, WLDAP32_LDAPMessage **result,
575 PLDAPControlA *serverctrls, PLDAPControlA *clientctrls )
576 {
577 ULONG ret = WLDAP32_LDAP_NOT_SUPPORTED;
578 #ifdef HAVE_LDAP
579 LDAPControlW **serverctrlsW = NULL, **clientctrlsW = NULL;
580
581 ret = WLDAP32_LDAP_NO_MEMORY;
582
583 TRACE( "(%p, %p, %p, %p, %p)\n", ld, retval, result, serverctrls, clientctrls );
584
585 if (!ld) return ~0u;
586
587 if (serverctrls) {
588 serverctrlsW = controlarrayAtoW( serverctrls );
589 if (!serverctrlsW) goto exit;
590 }
591 if (clientctrls) {
592 clientctrlsW = controlarrayAtoW( clientctrls );
593 if (!clientctrlsW) goto exit;
594 }
595
596 ret = ldap_start_tls_sW( ld, retval, result, serverctrlsW, clientctrlsW );
597
598 exit:
599 controlarrayfreeW( serverctrlsW );
600 controlarrayfreeW( clientctrlsW );
601
602 #endif
603 return ret;
604 }
605
606 /***********************************************************************
607 * ldap_start_tls_s (WLDAP32.@)
608 *
609 * Start TLS encryption on an LDAP connection.
610 *
611 * PARAMS
612 * ld [I] Pointer to an LDAP context.
613 * retval [I] Return value from the server.
614 * result [I] Response message from the server.
615 * serverctrls [I] Array of LDAP server controls.
616 * clientctrls [I] Array of LDAP client controls.
617 *
618 * RETURNS
619 * Success: LDAP_SUCCESS
620 * Failure: An LDAP error code.
621 *
622 * NOTES
623 * LDAP function that needs it is called.
624 */
625 ULONG CDECL ldap_start_tls_sW( WLDAP32_LDAP *ld, PULONG retval, WLDAP32_LDAPMessage **result,
626 PLDAPControlW *serverctrls, PLDAPControlW *clientctrls )
627 {
628 ULONG ret = WLDAP32_LDAP_NOT_SUPPORTED;
629 #ifdef HAVE_LDAP
630 LDAPControl **serverctrlsU = NULL, **clientctrlsU = NULL;
631
632 ret = WLDAP32_LDAP_NO_MEMORY;
633
634 TRACE( "(%p, %p, %p, %p, %p)\n", ld, retval, result, serverctrls, clientctrls );
635
636 if (!ld) return ~0u;
637
638 if (serverctrls) {
639 serverctrlsU = controlarrayWtoU( serverctrls );
640 if (!serverctrlsU) goto exit;
641 }
642 if (clientctrls) {
643 clientctrlsU = controlarrayWtoU( clientctrls );
644 if (!clientctrlsU) goto exit;
645 }
646
647 ret = map_error( ldap_start_tls_s( ld, serverctrlsU, clientctrlsU ));
648
649 exit:
650 controlarrayfreeU( serverctrlsU );
651 controlarrayfreeU( clientctrlsU );
652
653 #endif
654 return ret;
655 }
656
657 /***********************************************************************
658 * ldap_startup (WLDAP32.@)
659 */
660 ULONG CDECL ldap_startup( PLDAP_VERSION_INFO version, HANDLE *instance )
661 {
662 TRACE( "(%p, %p)\n", version, instance );
663 return WLDAP32_LDAP_SUCCESS;
664 }
665
666 /***********************************************************************
667 * ldap_stop_tls_s (WLDAP32.@)
668 *
669 * Stop TLS encryption on an LDAP connection.
670 *
671 * PARAMS
672 * ld [I] Pointer to an LDAP context.
673 *
674 * RETURNS
675 * Success: TRUE
676 * Failure: FALSE
677 */
678 BOOLEAN CDECL ldap_stop_tls_s( WLDAP32_LDAP *ld )
679 {
680 TRACE( "(%p)\n", ld );
681 return TRUE; /* FIXME: find a way to stop tls on a connection */
682 }