]> granicus.if.org Git - sudo/blob - lib/util/regress/parse_gids/parse_gids_test.c
674bd96af32b78759e81b0536a7532155f0894c8
[sudo] / lib / util / regress / parse_gids / parse_gids_test.c
1 /*
2  * Copyright (c) 2015 Todd C. Miller <Todd.Miller@sudo.ws>
3  *
4  * Permission to use, copy, modify, and distribute this software for any
5  * purpose with or without fee is hereby granted, provided that the above
6  * copyright notice and this permission notice appear in all copies.
7  *
8  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15  */
16
17 #include <config.h>
18
19 #include <sys/types.h>
20 #include <stdio.h>
21 #include <stdlib.h>
22 #ifdef HAVE_STRING_H
23 # include <string.h>
24 #endif /* HAVE_STRING_H */
25 #ifdef HAVE_STRINGS_H
26 # include <strings.h>
27 #endif /* HAVE_STRINGS_H */
28 #ifdef HAVE_STDBOOL_H
29 # include <stdbool.h>
30 #else
31 # include "compat/stdbool.h"
32 #endif
33
34 #include "sudo_compat.h"
35 #include "sudo_fatal.h"
36 #include "sudo_util.h"
37
38 __dso_public int main(int argc, char *argv[]);
39
40 /*
41  * Test that sudo_parse_gids() works as expected.
42  */
43
44 struct parse_gids_test {
45     const char *gids;
46     gid_t *baseptr;
47     gid_t basegid;
48     int ngids;
49     const GETGROUPS_T *gidlist;
50 };
51
52 static const GETGROUPS_T test1_out[] = { 0, 1, 2, 3, 4 };
53 static const GETGROUPS_T test2_out[] = { 1, 2, 3, 4 };
54 static const GETGROUPS_T test3_out[] = { 0, 1, (gid_t)-2, 3, 4 };
55
56 /* XXX - test syntax errors too */
57 static struct parse_gids_test test_data[] = {
58     { "1,2,3,4", &test_data[0].basegid, 0, 5, test1_out },
59     { "1,2,3,4", NULL, 0, 4, test2_out },
60     { "1,-2,3,4", &test_data[2].basegid, 0, 5, test3_out },
61     { NULL, false, 0, 0, NULL }
62 };
63
64 static void
65 dump_gids(const char *prefix, int ngids, const GETGROUPS_T *gidlist)
66 {
67     int i;
68
69     fprintf(stderr, "%s: %s: ", getprogname(), prefix);
70     for (i = 0; i < ngids; i++) {
71         fprintf(stderr, "%s%d", i ? ", " : "", (int)gidlist[i]);
72     }
73     fputc('\n', stderr);
74 }
75
76 int
77 main(int argc, char *argv[])
78 {
79     GETGROUPS_T *gidlist = NULL;
80     int i, j, errors = 0, ntests = 0;
81     int ngids;
82     initprogname(argc > 0 ? argv[0] : "parse_gids_test");
83
84     for (i = 0; test_data[i].gids != NULL; i++) {
85         free(gidlist);
86         ngids = sudo_parse_gids(test_data[i].gids, test_data[i].baseptr, &gidlist);
87         if (ngids == -1)
88             exit(1);    /* out of memory? */
89         ntests++;
90         if (ngids != test_data[i].ngids) {
91             sudo_warnx_nodebug("test #%d: expected %d gids, got %d",
92                 ntests, test_data[i].ngids, ngids);
93             dump_gids("expected", test_data[i].ngids, test_data[i].gidlist);
94             dump_gids("received", ngids, gidlist);
95             errors++;
96             continue;
97         }
98         ntests++;
99         for (j = 0; j < ngids; j++) {
100             if (test_data[i].gidlist[j] != gidlist[j]) {
101                 sudo_warnx_nodebug("test #%d: gid mismatch", ntests);
102                 dump_gids("expected", test_data[i].ngids, test_data[i].gidlist);
103                 dump_gids("received", ngids, gidlist);
104                 errors++;
105                 break;
106             }
107         }
108     }
109     if (ntests != 0) {
110         printf("%s: %d tests run, %d errors, %d%% success rate\n",
111             getprogname(), ntests, errors, (ntests - errors) * 100 / ntests);
112     }
113     exit(errors);
114 }