source: trunk/server/lib/nss_wrapper/testsuite.c

Last change on this file was 745, checked in by Silvan Scherrer, 13 years ago

Samba Server: updated trunk to 3.6.0

File size: 25.0 KB
Line 
1/*
2 Unix SMB/CIFS implementation.
3
4 local testing of the nss wrapper
5
6 Copyright (C) Guenther Deschner 2009-2010
7
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>.
20*/
21
22#include "includes.h"
23
24#ifndef NSS_WRAPPER
25#define NSS_WRAPPER
26#endif
27
28#include "torture/torture.h"
29#include "lib/replace/system/passwd.h"
30
31static bool copy_passwd(struct torture_context *tctx,
32 const struct passwd *pwd,
33 struct passwd *p)
34{
35 p->pw_name = talloc_strdup(tctx, pwd->pw_name);
36 p->pw_passwd = talloc_strdup(tctx, pwd->pw_passwd);
37 p->pw_uid = pwd->pw_uid;
38 p->pw_gid = pwd->pw_gid;
39 p->pw_gecos = talloc_strdup(tctx, pwd->pw_gecos);
40 p->pw_dir = talloc_strdup(tctx, pwd->pw_dir);
41 p->pw_shell = talloc_strdup(tctx, pwd->pw_shell);
42
43 return true;
44}
45
46static void print_passwd(struct passwd *pwd)
47{
48 printf("%s:%s:%lu:%lu:%s:%s:%s\n",
49 pwd->pw_name,
50 pwd->pw_passwd,
51 (unsigned long)pwd->pw_uid,
52 (unsigned long)pwd->pw_gid,
53 pwd->pw_gecos,
54 pwd->pw_dir,
55 pwd->pw_shell);
56}
57
58
59static bool test_nwrap_getpwnam(struct torture_context *tctx,
60 const char *name,
61 struct passwd *pwd_p)
62{
63 struct passwd *pwd;
64
65 torture_comment(tctx, "Testing getpwnam: %s\n", name);
66
67 pwd = getpwnam(name);
68 if (pwd) {
69 print_passwd(pwd);
70 }
71
72 if (pwd_p) {
73 copy_passwd(tctx, pwd, pwd_p);
74 }
75
76 return pwd ? true : false;
77}
78
79static bool test_nwrap_getpwnam_r(struct torture_context *tctx,
80 const char *name,
81 struct passwd *pwd_p)
82{
83 struct passwd pwd, *pwdp;
84 char buffer[4096];
85 int ret;
86
87 torture_comment(tctx, "Testing getpwnam_r: %s\n", name);
88
89 ret = getpwnam_r(name, &pwd, buffer, sizeof(buffer), &pwdp);
90 if (ret != 0) {
91 if (ret != ENOENT) {
92 torture_comment(tctx, "got %d return code\n", ret);
93 }
94 return false;
95 }
96
97 print_passwd(&pwd);
98
99 if (pwd_p) {
100 copy_passwd(tctx, &pwd, pwd_p);
101 }
102
103 return true;
104}
105
106static bool test_nwrap_getpwuid(struct torture_context *tctx,
107 uid_t uid,
108 struct passwd *pwd_p)
109{
110 struct passwd *pwd;
111
112 torture_comment(tctx, "Testing getpwuid: %lu\n", (unsigned long)uid);
113
114 pwd = getpwuid(uid);
115 if (pwd) {
116 print_passwd(pwd);
117 }
118
119 if (pwd_p) {
120 copy_passwd(tctx, pwd, pwd_p);
121 }
122
123 return pwd ? true : false;
124}
125
126static bool test_nwrap_getpwuid_r(struct torture_context *tctx,
127 uid_t uid,
128 struct passwd *pwd_p)
129{
130 struct passwd pwd, *pwdp;
131 char buffer[4096];
132 int ret;
133
134 torture_comment(tctx, "Testing getpwuid_r: %lu\n", (unsigned long)uid);
135
136 ret = getpwuid_r(uid, &pwd, buffer, sizeof(buffer), &pwdp);
137 if (ret != 0) {
138 if (ret != ENOENT) {
139 torture_comment(tctx, "got %d return code\n", ret);
140 }
141 return false;
142 }
143
144 print_passwd(&pwd);
145
146 if (pwd_p) {
147 copy_passwd(tctx, &pwd, pwd_p);
148 }
149
150 return true;
151}
152
153
154static bool copy_group(struct torture_context *tctx,
155 const struct group *grp,
156 struct group *g)
157{
158 int i;
159
160 g->gr_name = talloc_strdup(tctx, grp->gr_name);
161 g->gr_passwd = talloc_strdup(tctx, grp->gr_passwd);
162 g->gr_gid = grp->gr_gid;
163 g->gr_mem = NULL;
164
165 for (i=0; grp->gr_mem && grp->gr_mem[i]; i++) {
166 g->gr_mem = talloc_realloc(tctx, g->gr_mem, char *, i + 2);
167 g->gr_mem[i] = talloc_strdup(g->gr_mem, grp->gr_mem[i]);
168 g->gr_mem[i+1] = NULL;
169 }
170
171 return true;
172}
173
174static void print_group(struct group *grp)
175{
176 int i;
177 printf("%s:%s:%lu:",
178 grp->gr_name,
179 grp->gr_passwd,
180 (unsigned long)grp->gr_gid);
181
182 if ((grp->gr_mem == NULL) || !grp->gr_mem[0]) {
183 printf("\n");
184 return;
185 }
186
187 for (i=0; grp->gr_mem[i+1]; i++) {
188 printf("%s,", grp->gr_mem[i]);
189 }
190 printf("%s\n", grp->gr_mem[i]);
191}
192
193static bool test_nwrap_getgrnam(struct torture_context *tctx,
194 const char *name,
195 struct group *grp_p)
196{
197 struct group *grp;
198
199 torture_comment(tctx, "Testing getgrnam: %s\n", name);
200
201 grp = getgrnam(name);
202 if (grp) {
203 print_group(grp);
204 }
205
206 if (grp_p) {
207 copy_group(tctx, grp, grp_p);
208 }
209
210 return grp ? true : false;
211}
212
213static bool test_nwrap_getgrnam_r(struct torture_context *tctx,
214 const char *name,
215 struct group *grp_p)
216{
217 struct group grp, *grpp;
218 char buffer[4096];
219 int ret;
220
221 torture_comment(tctx, "Testing getgrnam_r: %s\n", name);
222
223 ret = getgrnam_r(name, &grp, buffer, sizeof(buffer), &grpp);
224 if (ret != 0) {
225 if (ret != ENOENT) {
226 torture_comment(tctx, "got %d return code\n", ret);
227 }
228 return false;
229 }
230
231 print_group(&grp);
232
233 if (grp_p) {
234 copy_group(tctx, &grp, grp_p);
235 }
236
237 return true;
238}
239
240
241static bool test_nwrap_getgrgid(struct torture_context *tctx,
242 gid_t gid,
243 struct group *grp_p)
244{
245 struct group *grp;
246
247 torture_comment(tctx, "Testing getgrgid: %lu\n", (unsigned long)gid);
248
249 grp = getgrgid(gid);
250 if (grp) {
251 print_group(grp);
252 }
253
254 if (grp_p) {
255 copy_group(tctx, grp, grp_p);
256 }
257
258 return grp ? true : false;
259}
260
261static bool test_nwrap_getgrgid_r(struct torture_context *tctx,
262 gid_t gid,
263 struct group *grp_p)
264{
265 struct group grp, *grpp;
266 char buffer[4096];
267 int ret;
268
269 torture_comment(tctx, "Testing getgrgid_r: %lu\n", (unsigned long)gid);
270
271 ret = getgrgid_r(gid, &grp, buffer, sizeof(buffer), &grpp);
272 if (ret != 0) {
273 if (ret != ENOENT) {
274 torture_comment(tctx, "got %d return code\n", ret);
275 }
276 return false;
277 }
278
279 print_group(&grp);
280
281 if (grp_p) {
282 copy_group(tctx, &grp, grp_p);
283 }
284
285 return true;
286}
287
288static bool test_nwrap_enum_passwd(struct torture_context *tctx,
289 struct passwd **pwd_array_p,
290 size_t *num_pwd_p)
291{
292 struct passwd *pwd;
293 struct passwd *pwd_array = NULL;
294 size_t num_pwd = 0;
295
296 torture_comment(tctx, "Testing setpwent\n");
297 setpwent();
298
299 while ((pwd = getpwent()) != NULL) {
300 torture_comment(tctx, "Testing getpwent\n");
301
302 print_passwd(pwd);
303 if (pwd_array_p && num_pwd_p) {
304 pwd_array = talloc_realloc(tctx, pwd_array, struct passwd, num_pwd+1);
305 torture_assert(tctx, pwd_array, "out of memory");
306 copy_passwd(tctx, pwd, &pwd_array[num_pwd]);
307 num_pwd++;
308 }
309 }
310
311 torture_comment(tctx, "Testing endpwent\n");
312 endpwent();
313
314 if (pwd_array_p) {
315 *pwd_array_p = pwd_array;
316 }
317 if (num_pwd_p) {
318 *num_pwd_p = num_pwd;
319 }
320
321 return true;
322}
323
324static bool test_nwrap_enum_r_passwd(struct torture_context *tctx,
325 struct passwd **pwd_array_p,
326 size_t *num_pwd_p)
327{
328 struct passwd pwd, *pwdp;
329 struct passwd *pwd_array = NULL;
330 size_t num_pwd = 0;
331 char buffer[4096];
332 int ret;
333
334 torture_comment(tctx, "Testing setpwent\n");
335 setpwent();
336
337 while (1) {
338 torture_comment(tctx, "Testing getpwent_r\n");
339
340 ret = getpwent_r(&pwd, buffer, sizeof(buffer), &pwdp);
341 if (ret != 0) {
342 if (ret != ENOENT) {
343 torture_comment(tctx, "got %d return code\n", ret);
344 }
345 break;
346 }
347 print_passwd(&pwd);
348 if (pwd_array_p && num_pwd_p) {
349 pwd_array = talloc_realloc(tctx, pwd_array, struct passwd, num_pwd+1);
350 torture_assert(tctx, pwd_array, "out of memory");
351 copy_passwd(tctx, &pwd, &pwd_array[num_pwd]);
352 num_pwd++;
353 }
354 }
355
356 torture_comment(tctx, "Testing endpwent\n");
357 endpwent();
358
359 if (pwd_array_p) {
360 *pwd_array_p = pwd_array;
361 }
362 if (num_pwd_p) {
363 *num_pwd_p = num_pwd;
364 }
365
366 return true;
367}
368
369static bool torture_assert_passwd_equal(struct torture_context *tctx,
370 const struct passwd *p1,
371 const struct passwd *p2,
372 const char *comment)
373{
374 torture_assert_str_equal(tctx, p1->pw_name, p2->pw_name, comment);
375 torture_assert_str_equal(tctx, p1->pw_passwd, p2->pw_passwd, comment);
376 torture_assert_int_equal(tctx, p1->pw_uid, p2->pw_uid, comment);
377 torture_assert_int_equal(tctx, p1->pw_gid, p2->pw_gid, comment);
378 torture_assert_str_equal(tctx, p1->pw_gecos, p2->pw_gecos, comment);
379 torture_assert_str_equal(tctx, p1->pw_dir, p2->pw_dir, comment);
380 torture_assert_str_equal(tctx, p1->pw_shell, p2->pw_shell, comment);
381
382 return true;
383}
384
385static bool test_nwrap_passwd(struct torture_context *tctx)
386{
387 int i;
388 struct passwd *pwd, pwd1, pwd2;
389 size_t num_pwd;
390
391 torture_assert(tctx, test_nwrap_enum_passwd(tctx, &pwd, &num_pwd),
392 "failed to enumerate passwd");
393
394 for (i=0; i < num_pwd; i++) {
395 torture_assert(tctx, test_nwrap_getpwnam(tctx, pwd[i].pw_name, &pwd1),
396 "failed to call getpwnam for enumerated user");
397 torture_assert_passwd_equal(tctx, &pwd[i], &pwd1,
398 "getpwent and getpwnam gave different results");
399 torture_assert(tctx, test_nwrap_getpwuid(tctx, pwd[i].pw_uid, &pwd2),
400 "failed to call getpwuid for enumerated user");
401 torture_assert_passwd_equal(tctx, &pwd[i], &pwd2,
402 "getpwent and getpwuid gave different results");
403 torture_assert_passwd_equal(tctx, &pwd1, &pwd2,
404 "getpwnam and getpwuid gave different results");
405 }
406
407 return true;
408}
409
410static bool test_nwrap_passwd_r(struct torture_context *tctx)
411{
412 int i;
413 struct passwd *pwd, pwd1, pwd2;
414 size_t num_pwd;
415
416 torture_assert(tctx, test_nwrap_enum_r_passwd(tctx, &pwd, &num_pwd),
417 "failed to enumerate passwd");
418
419 for (i=0; i < num_pwd; i++) {
420 torture_assert(tctx, test_nwrap_getpwnam_r(tctx, pwd[i].pw_name, &pwd1),
421 "failed to call getpwnam_r for enumerated user");
422 torture_assert_passwd_equal(tctx, &pwd[i], &pwd1,
423 "getpwent_r and getpwnam_r gave different results");
424 torture_assert(tctx, test_nwrap_getpwuid_r(tctx, pwd[i].pw_uid, &pwd2),
425 "failed to call getpwuid_r for enumerated user");
426 torture_assert_passwd_equal(tctx, &pwd[i], &pwd2,
427 "getpwent_r and getpwuid_r gave different results");
428 torture_assert_passwd_equal(tctx, &pwd1, &pwd2,
429 "getpwnam_r and getpwuid_r gave different results");
430 }
431
432 return true;
433}
434
435static bool test_nwrap_passwd_r_cross(struct torture_context *tctx)
436{
437 int i;
438 struct passwd *pwd, pwd1, pwd2, pwd3, pwd4;
439 size_t num_pwd;
440
441 torture_assert(tctx, test_nwrap_enum_r_passwd(tctx, &pwd, &num_pwd),
442 "failed to enumerate passwd");
443
444 for (i=0; i < num_pwd; i++) {
445 torture_assert(tctx, test_nwrap_getpwnam_r(tctx, pwd[i].pw_name, &pwd1),
446 "failed to call getpwnam_r for enumerated user");
447 torture_assert_passwd_equal(tctx, &pwd[i], &pwd1,
448 "getpwent_r and getpwnam_r gave different results");
449 torture_assert(tctx, test_nwrap_getpwuid_r(tctx, pwd[i].pw_uid, &pwd2),
450 "failed to call getpwuid_r for enumerated user");
451 torture_assert_passwd_equal(tctx, &pwd[i], &pwd2,
452 "getpwent_r and getpwuid_r gave different results");
453 torture_assert_passwd_equal(tctx, &pwd1, &pwd2,
454 "getpwnam_r and getpwuid_r gave different results");
455 torture_assert(tctx, test_nwrap_getpwnam(tctx, pwd[i].pw_name, &pwd3),
456 "failed to call getpwnam for enumerated user");
457 torture_assert_passwd_equal(tctx, &pwd[i], &pwd3,
458 "getpwent_r and getpwnam gave different results");
459 torture_assert(tctx, test_nwrap_getpwuid(tctx, pwd[i].pw_uid, &pwd4),
460 "failed to call getpwuid for enumerated user");
461 torture_assert_passwd_equal(tctx, &pwd[i], &pwd4,
462 "getpwent_r and getpwuid gave different results");
463 torture_assert_passwd_equal(tctx, &pwd3, &pwd4,
464 "getpwnam and getpwuid gave different results");
465 }
466
467 return true;
468}
469
470static bool test_nwrap_enum_group(struct torture_context *tctx,
471 struct group **grp_array_p,
472 size_t *num_grp_p)
473{
474 struct group *grp;
475 struct group *grp_array = NULL;
476 size_t num_grp = 0;
477
478 torture_comment(tctx, "Testing setgrent\n");
479 setgrent();
480
481 while ((grp = getgrent()) != NULL) {
482 torture_comment(tctx, "Testing getgrent\n");
483
484 print_group(grp);
485 if (grp_array_p && num_grp_p) {
486 grp_array = talloc_realloc(tctx, grp_array, struct group, num_grp+1);
487 torture_assert(tctx, grp_array, "out of memory");
488 copy_group(tctx, grp, &grp_array[num_grp]);
489 num_grp++;
490 }
491 }
492
493 torture_comment(tctx, "Testing endgrent\n");
494 endgrent();
495
496 if (grp_array_p) {
497 *grp_array_p = grp_array;
498 }
499 if (num_grp_p) {
500 *num_grp_p = num_grp;
501 }
502
503 return true;
504}
505
506static bool test_nwrap_enum_r_group(struct torture_context *tctx,
507 struct group **grp_array_p,
508 size_t *num_grp_p)
509{
510 struct group grp, *grpp;
511 struct group *grp_array = NULL;
512 size_t num_grp = 0;
513 char buffer[4096];
514 int ret;
515
516 torture_comment(tctx, "Testing setgrent\n");
517 setgrent();
518
519 while (1) {
520 torture_comment(tctx, "Testing getgrent_r\n");
521
522 ret = getgrent_r(&grp, buffer, sizeof(buffer), &grpp);
523 if (ret != 0) {
524 if (ret != ENOENT) {
525 torture_comment(tctx, "got %d return code\n", ret);
526 }
527 break;
528 }
529 print_group(&grp);
530 if (grp_array_p && num_grp_p) {
531 grp_array = talloc_realloc(tctx, grp_array, struct group, num_grp+1);
532 torture_assert(tctx, grp_array, "out of memory");
533 copy_group(tctx, &grp, &grp_array[num_grp]);
534 num_grp++;
535 }
536 }
537
538 torture_comment(tctx, "Testing endgrent\n");
539 endgrent();
540
541 if (grp_array_p) {
542 *grp_array_p = grp_array;
543 }
544 if (num_grp_p) {
545 *num_grp_p = num_grp;
546 }
547
548 return true;
549}
550
551static bool torture_assert_group_equal(struct torture_context *tctx,
552 const struct group *g1,
553 const struct group *g2,
554 const char *comment)
555{
556 int i;
557 torture_assert_str_equal(tctx, g1->gr_name, g2->gr_name, comment);
558 torture_assert_str_equal(tctx, g1->gr_passwd, g2->gr_passwd, comment);
559 torture_assert_int_equal(tctx, g1->gr_gid, g2->gr_gid, comment);
560 if (g1->gr_mem && !g2->gr_mem) {
561 return false;
562 }
563 if (!g1->gr_mem && g2->gr_mem) {
564 return false;
565 }
566 if (!g1->gr_mem && !g2->gr_mem) {
567 return true;
568 }
569 for (i=0; g1->gr_mem[i] && g2->gr_mem[i]; i++) {
570 torture_assert_str_equal(tctx, g1->gr_mem[i], g2->gr_mem[i], comment);
571 }
572
573 return true;
574}
575
576static bool test_nwrap_group(struct torture_context *tctx)
577{
578 int i;
579 struct group *grp, grp1, grp2;
580 size_t num_grp;
581
582 torture_assert(tctx, test_nwrap_enum_group(tctx, &grp, &num_grp),
583 "failed to enumerate group");
584
585 for (i=0; i < num_grp; i++) {
586 torture_assert(tctx, test_nwrap_getgrnam(tctx, grp[i].gr_name, &grp1),
587 "failed to call getgrnam for enumerated user");
588 torture_assert_group_equal(tctx, &grp[i], &grp1,
589 "getgrent and getgrnam gave different results");
590 torture_assert(tctx, test_nwrap_getgrgid(tctx, grp[i].gr_gid, &grp2),
591 "failed to call getgrgid for enumerated user");
592 torture_assert_group_equal(tctx, &grp[i], &grp2,
593 "getgrent and getgrgid gave different results");
594 torture_assert_group_equal(tctx, &grp1, &grp2,
595 "getgrnam and getgrgid gave different results");
596 }
597
598 return true;
599}
600
601static bool test_nwrap_group_r(struct torture_context *tctx)
602{
603 int i;
604 struct group *grp, grp1, grp2;
605 size_t num_grp;
606
607 torture_assert(tctx, test_nwrap_enum_r_group(tctx, &grp, &num_grp),
608 "failed to enumerate group");
609
610 for (i=0; i < num_grp; i++) {
611 torture_assert(tctx, test_nwrap_getgrnam_r(tctx, grp[i].gr_name, &grp1),
612 "failed to call getgrnam_r for enumerated user");
613 torture_assert_group_equal(tctx, &grp[i], &grp1,
614 "getgrent_r and getgrnam_r gave different results");
615 torture_assert(tctx, test_nwrap_getgrgid_r(tctx, grp[i].gr_gid, &grp2),
616 "failed to call getgrgid_r for enumerated user");
617 torture_assert_group_equal(tctx, &grp[i], &grp2,
618 "getgrent_r and getgrgid_r gave different results");
619 torture_assert_group_equal(tctx, &grp1, &grp2,
620 "getgrnam_r and getgrgid_r gave different results");
621 }
622
623 return true;
624}
625
626static bool test_nwrap_group_r_cross(struct torture_context *tctx)
627{
628 int i;
629 struct group *grp, grp1, grp2, grp3, grp4;
630 size_t num_grp;
631
632 torture_assert(tctx, test_nwrap_enum_r_group(tctx, &grp, &num_grp),
633 "failed to enumerate group");
634
635 for (i=0; i < num_grp; i++) {
636 torture_assert(tctx, test_nwrap_getgrnam_r(tctx, grp[i].gr_name, &grp1),
637 "failed to call getgrnam_r for enumerated user");
638 torture_assert_group_equal(tctx, &grp[i], &grp1,
639 "getgrent_r and getgrnam_r gave different results");
640 torture_assert(tctx, test_nwrap_getgrgid_r(tctx, grp[i].gr_gid, &grp2),
641 "failed to call getgrgid_r for enumerated user");
642 torture_assert_group_equal(tctx, &grp[i], &grp2,
643 "getgrent_r and getgrgid_r gave different results");
644 torture_assert_group_equal(tctx, &grp1, &grp2,
645 "getgrnam_r and getgrgid_r gave different results");
646 torture_assert(tctx, test_nwrap_getgrnam(tctx, grp[i].gr_name, &grp3),
647 "failed to call getgrnam for enumerated user");
648 torture_assert_group_equal(tctx, &grp[i], &grp3,
649 "getgrent_r and getgrnam gave different results");
650 torture_assert(tctx, test_nwrap_getgrgid(tctx, grp[i].gr_gid, &grp4),
651 "failed to call getgrgid for enumerated user");
652 torture_assert_group_equal(tctx, &grp[i], &grp4,
653 "getgrent_r and getgrgid gave different results");
654 torture_assert_group_equal(tctx, &grp3, &grp4,
655 "getgrnam and getgrgid gave different results");
656 }
657
658 return true;
659}
660
661static bool test_nwrap_getgrouplist(struct torture_context *tctx,
662 const char *user,
663 gid_t gid,
664 gid_t **gids_p,
665 int *num_gids_p)
666{
667 int ret;
668 int num_groups = 0;
669 gid_t *groups = NULL;
670
671 torture_comment(tctx, "Testing getgrouplist: %s\n", user);
672
673 ret = getgrouplist(user, gid, NULL, &num_groups);
674 if (ret == -1 || num_groups != 0) {
675
676 groups = talloc_array(tctx, gid_t, num_groups);
677 torture_assert(tctx, groups, "out of memory\n");
678
679 ret = getgrouplist(user, gid, groups, &num_groups);
680 }
681
682 torture_assert(tctx, (ret != -1), "failed to call getgrouplist");
683
684 torture_comment(tctx, "%s is member in %d groups\n", user, num_groups);
685
686 if (gids_p) {
687 *gids_p = groups;
688 }
689 if (num_gids_p) {
690 *num_gids_p = num_groups;
691 }
692
693 return true;
694}
695
696static bool test_nwrap_user_in_group(struct torture_context *tctx,
697 const struct passwd *pwd,
698 const struct group *grp)
699{
700 int i;
701
702 for (i=0; grp->gr_mem && grp->gr_mem[i] != NULL; i++) {
703 if (strequal(grp->gr_mem[i], pwd->pw_name)) {
704 return true;
705 }
706 }
707
708 return false;
709}
710
711static bool test_nwrap_membership_user(struct torture_context *tctx,
712 const struct passwd *pwd,
713 struct group *grp_array,
714 size_t num_grp)
715{
716 int num_user_groups = 0;
717 int num_user_groups_from_enum = 0;
718 gid_t *user_groups = NULL;
719 int g, i;
720 bool primary_group_had_user_member = false;
721
722 torture_assert(tctx, test_nwrap_getgrouplist(tctx,
723 pwd->pw_name,
724 pwd->pw_gid,
725 &user_groups,
726 &num_user_groups),
727 "failed to test getgrouplist");
728
729 for (g=0; g < num_user_groups; g++) {
730 torture_assert(tctx, test_nwrap_getgrgid(tctx, user_groups[g], NULL),
731 "failed to find the group the user is a member of");
732 }
733
734
735 for (i=0; i < num_grp; i++) {
736
737 struct group grp = grp_array[i];
738
739 if (test_nwrap_user_in_group(tctx, pwd, &grp)) {
740
741 struct group current_grp;
742 num_user_groups_from_enum++;
743
744 torture_assert(tctx, test_nwrap_getgrnam(tctx, grp.gr_name, &current_grp),
745 "failed to find the group the user is a member of");
746
747 if (current_grp.gr_gid == pwd->pw_gid) {
748 torture_comment(tctx, "primary group %s of user %s lists user as member\n",
749 current_grp.gr_name,
750 pwd->pw_name);
751 primary_group_had_user_member = true;
752 }
753
754 continue;
755 }
756 }
757
758 if (!primary_group_had_user_member) {
759 num_user_groups_from_enum++;
760 }
761
762 torture_assert_int_equal(tctx, num_user_groups, num_user_groups_from_enum,
763 "getgrouplist and real inspection of grouplist gave different results\n");
764
765 return true;
766}
767
768static bool test_nwrap_membership(struct torture_context *tctx)
769{
770 const char *old_pwd = getenv("NSS_WRAPPER_PASSWD");
771 const char *old_group = getenv("NSS_WRAPPER_GROUP");
772 struct passwd *pwd;
773 size_t num_pwd;
774 struct group *grp;
775 size_t num_grp;
776 int i;
777
778 if (!old_pwd || !old_group) {
779 torture_comment(tctx, "ENV NSS_WRAPPER_PASSWD or NSS_WRAPPER_GROUP not set\n");
780 torture_skip(tctx, "nothing to test\n");
781 }
782
783 torture_assert(tctx, test_nwrap_enum_passwd(tctx, &pwd, &num_pwd),
784 "failed to enumerate passwd");
785 torture_assert(tctx, test_nwrap_enum_group(tctx, &grp, &num_grp),
786 "failed to enumerate group");
787
788 for (i=0; i < num_pwd; i++) {
789
790 torture_assert(tctx, test_nwrap_membership_user(tctx, &pwd[i], grp, num_grp),
791 "failed to test membership for user");
792
793 }
794
795 return true;
796}
797
798static bool test_nwrap_enumeration(struct torture_context *tctx)
799{
800 const char *old_pwd = getenv("NSS_WRAPPER_PASSWD");
801 const char *old_group = getenv("NSS_WRAPPER_GROUP");
802
803 if (!old_pwd || !old_group) {
804 torture_comment(tctx, "ENV NSS_WRAPPER_PASSWD or NSS_WRAPPER_GROUP not set\n");
805 torture_skip(tctx, "nothing to test\n");
806 }
807
808 torture_assert(tctx, test_nwrap_passwd(tctx),
809 "failed to test users");
810 torture_assert(tctx, test_nwrap_group(tctx),
811 "failed to test groups");
812
813 return true;
814}
815
816static bool test_nwrap_reentrant_enumeration(struct torture_context *tctx)
817{
818 const char *old_pwd = getenv("NSS_WRAPPER_PASSWD");
819 const char *old_group = getenv("NSS_WRAPPER_GROUP");
820
821 if (!old_pwd || !old_group) {
822 torture_comment(tctx, "ENV NSS_WRAPPER_PASSWD or NSS_WRAPPER_GROUP not set\n");
823 torture_skip(tctx, "nothing to test\n");
824 }
825
826 torture_comment(tctx, "Testing re-entrant calls\n");
827
828 torture_assert(tctx, test_nwrap_passwd_r(tctx),
829 "failed to test users");
830 torture_assert(tctx, test_nwrap_group_r(tctx),
831 "failed to test groups");
832
833 return true;
834}
835
836static bool test_nwrap_reentrant_enumeration_crosschecks(struct torture_context *tctx)
837{
838 const char *old_pwd = getenv("NSS_WRAPPER_PASSWD");
839 const char *old_group = getenv("NSS_WRAPPER_GROUP");
840
841 if (!old_pwd || !old_group) {
842 torture_comment(tctx, "ENV NSS_WRAPPER_PASSWD or NSS_WRAPPER_GROUP not set\n");
843 torture_skip(tctx, "nothing to test\n");
844 }
845
846 torture_comment(tctx, "Testing re-entrant calls with cross checks\n");
847
848 torture_assert(tctx, test_nwrap_passwd_r_cross(tctx),
849 "failed to test users");
850 torture_assert(tctx, test_nwrap_group_r_cross(tctx),
851 "failed to test groups");
852
853 return true;
854}
855
856static bool test_nwrap_passwd_duplicates(struct torture_context *tctx)
857{
858 int i, d;
859 struct passwd *pwd;
860 size_t num_pwd;
861 int duplicates = 0;
862
863 torture_assert(tctx, test_nwrap_enum_passwd(tctx, &pwd, &num_pwd),
864 "failed to enumerate passwd");
865
866 for (i=0; i < num_pwd; i++) {
867 const char *current_name = pwd[i].pw_name;
868 for (d=0; d < num_pwd; d++) {
869 const char *dup_name = pwd[d].pw_name;
870 if (d == i) {
871 continue;
872 }
873 if (!strequal(current_name, dup_name)) {
874 continue;
875 }
876
877 torture_warning(tctx, "found duplicate names:");
878 print_passwd(&pwd[d]);
879 print_passwd(&pwd[i]);
880 duplicates++;
881 }
882 }
883
884 if (duplicates) {
885 torture_fail(tctx, talloc_asprintf(tctx, "found %d duplicate names", duplicates));
886 }
887
888 return true;
889}
890
891static bool test_nwrap_group_duplicates(struct torture_context *tctx)
892{
893 int i, d;
894 struct group *grp;
895 size_t num_grp;
896 int duplicates = 0;
897
898 torture_assert(tctx, test_nwrap_enum_group(tctx, &grp, &num_grp),
899 "failed to enumerate group");
900
901 for (i=0; i < num_grp; i++) {
902 const char *current_name = grp[i].gr_name;
903 for (d=0; d < num_grp; d++) {
904 const char *dup_name = grp[d].gr_name;
905 if (d == i) {
906 continue;
907 }
908 if (!strequal(current_name, dup_name)) {
909 continue;
910 }
911
912 torture_warning(tctx, "found duplicate names:");
913 print_group(&grp[d]);
914 print_group(&grp[i]);
915 duplicates++;
916 }
917 }
918
919 if (duplicates) {
920 torture_fail(tctx, talloc_asprintf(tctx, "found %d duplicate names", duplicates));
921 }
922
923 return true;
924}
925
926
927static bool test_nwrap_duplicates(struct torture_context *tctx)
928{
929 const char *old_pwd = getenv("NSS_WRAPPER_PASSWD");
930 const char *old_group = getenv("NSS_WRAPPER_GROUP");
931
932 if (!old_pwd || !old_group) {
933 torture_comment(tctx, "ENV NSS_WRAPPER_PASSWD or NSS_WRAPPER_GROUP not set\n");
934 torture_skip(tctx, "nothing to test\n");
935 }
936
937 torture_assert(tctx, test_nwrap_passwd_duplicates(tctx),
938 "failed to test users");
939 torture_assert(tctx, test_nwrap_group_duplicates(tctx),
940 "failed to test groups");
941
942 return true;
943}
944
945
946struct torture_suite *torture_local_nss_wrapper(TALLOC_CTX *mem_ctx)
947{
948 struct torture_suite *suite = torture_suite_create(mem_ctx, "nss-wrapper");
949
950 torture_suite_add_simple_test(suite, "enumeration", test_nwrap_enumeration);
951 torture_suite_add_simple_test(suite, "reentrant enumeration", test_nwrap_reentrant_enumeration);
952 torture_suite_add_simple_test(suite, "reentrant enumeration crosschecks", test_nwrap_reentrant_enumeration_crosschecks);
953 torture_suite_add_simple_test(suite, "membership", test_nwrap_membership);
954 torture_suite_add_simple_test(suite, "duplicates", test_nwrap_duplicates);
955
956 return suite;
957}
Note: See TracBrowser for help on using the repository browser.