A user might belong to only a single group
[p5sagit/p5-mst-13.2.git] / t / op / groups.t
CommitLineData
fe14fcc3 1#!./perl
2
61ae2fbf 3$ENV{PATH} ="/bin:/usr/bin:/usr/xpg4/bin:/usr/ucb" .
4 exists $ENV{PATH} ? ":$ENV{PATH}" : "";
5$ENV{LC_ALL} = "C"; # so that external utilities speak English
b598356e 6$ENV{LANGUAGE} = 'C'; # GNU locale extension
b9416812 7
13d7cbc1 8sub quit {
45c0de28 9 print "1..0 # Skip: no `id` or `groups`\n";
13d7cbc1 10 exit 0;
11}
12
2986a63f 13quit() if (($^O eq 'MSWin32' || $^O eq 'NetWare') or $^O =~ /lynxos/i);
13d7cbc1 14
d0f88fcc 15# We have to find a command that prints all (effective
16# and real) group names (not ids). The known commands are:
17# groups
18# id -Gn
19# id -a
20# Beware 1: some systems do just 'id -G' even when 'id -Gn' is used.
98cfb1fc 21# Beware 2: id -Gn or id -a format might be id(name) or name(id).
22# Beware 3: the groups= might be anywhere in the id output.
f62c0cf2 23# Beware 4: groups can have spaces ('id -a' being the only defense against this)
702a0e5a 24# Beware 5: id -a might not contain the groups= part.
98cfb1fc 25#
26# That is, we might meet the following:
27#
f62c0cf2 28# foo bar zot # accept
29# foo 22 42 bar zot # accept
30# 1 22 42 2 3 # reject
31# groups=(42),foo(1),bar(2),zot me(3) # parse
32# groups=22,42,1(foo),2(bar),3(zot me) # parse
98cfb1fc 33#
34# and the groups= might be after, before, or between uid=... and gid=...
d0f88fcc 35
36GROUPS: {
f62c0cf2 37 # prefer 'id' over 'groups' (is this ever wrong anywhere?)
38 # and 'id -a' over 'id -Gn' (the former is good about spaces in group names)
39 if (($groups = `id -a 2>/dev/null`) ne '') {
40 # $groups is of the form:
41 # uid=39957(gsar) gid=22(users) groups=33536,39181,22(users),0(root),1067(dev)
702a0e5a 42 last GROUPS if $groups =~ /groups=/;
f62c0cf2 43 }
44 if (($groups = `id -Gn 2>/dev/null`) ne '') {
45 # $groups could be of the form:
46 # users 33536 39181 root dev
47 last GROUPS if $groups !~ /^(\d|\s)+$/;
d0f88fcc 48 }
f62c0cf2 49 if (($groups = `groups 2>/dev/null`) ne '') {
50 # may not reflect all groups in some places, so do a sanity check
51 if (-d '/afs') {
52 print <<EOM;
53# These test results *may* be bogus, as you appear to have AFS,
54# and I can't find a working 'id' in your PATH (which I have set
55# to '$ENV{PATH}').
56#
57# If these tests fail, report the particular incantation you use
58# on this platform to find *all* the groups that an arbitrary
59# luser may belong to, using the 'perlbug' program.
60EOM
61 }
98cfb1fc 62 last GROUPS;
d0f88fcc 63 }
64 # Okay, not today.
13d7cbc1 65 quit();
fe14fcc3 66}
67
4435c477 68unless (eval { getgrgid(0); 1 }) {
69 print "1..0 # Skip: getgrgid() not implemented\n";
70 exit 0;
71}
72
dd570ea6 73chomp($groups);
74
75print "# groups = $groups\n";
76
98cfb1fc 77# Remember that group names can contain whitespace, '-', et cetera.
78# That is: do not \w, do not \S.
f62c0cf2 79if ($groups =~ /groups=(.+)( [ug]id=|$)/) {
98cfb1fc 80 my $gr = $1;
f62c0cf2 81 my @g0 = split /,/, $gr;
82 my @g1;
83 # prefer names over numbers
84 for (@g0) {
85 # 42(zot me)
9461e3d0 86 if (/^(\d+)(?:\(([^)]+)\))?/) {
f62c0cf2 87 push @g1, ($2 || $1);
88 }
89 # zot me(42)
9461e3d0 90 elsif (/^([^(]*)\((\d+)\)/) {
f62c0cf2 91 push @g1, ($1 || $2);
92 }
93 else {
94 print "# ignoring group entry [$_]\n";
95 }
96 }
97 print "# groups=$gr\n";
98cfb1fc 98 print "# g0 = @g0\n";
99 print "# g1 = @g1\n";
f62c0cf2 100 $groups = "@g1";
98cfb1fc 101}
102
988174c1 103print "1..2\n";
104
105$pwgid = $( + 0;
106($pwgnam) = getgrgid($pwgid);
f8da21f9 107if ($^O eq 'cygwin') { # basegroup on Cygwin has id = 0.
108 @basegroup{$pwgid,$pwgnam} = (0,0);
109} else {
110 @basegroup{$pwgid,$pwgnam} = (1,1);
111}
988174c1 112$seen{$pwgid}++;
fe14fcc3 113
dd570ea6 114print "# pwgid = $pwgid, pwgnam = $pwgnam\n";
115
fe14fcc3 116for (split(' ', $()) {
117 next if $seen{$_}++;
6e21c824 118 ($group) = getgrgid($_);
119 if (defined $group) {
120 push(@gr, $group);
121 }
122 else {
123 push(@gr, $_);
124 }
fe14fcc3 125}
988174c1 126
dd570ea6 127print "# gr = @gr\n";
128
27b4d0f8 129if ($^O =~ /^(?:uwin|solaris)$/) {
130 # Or anybody else who can have spaces in group names.
72720e3c 131 $gr1 = join(' ', grep(!$did{$_}++, sort split(' ', join(' ', @gr))));
132} else {
133 $gr1 = join(' ', sort @gr);
134}
988174c1 135
b9416812 136$gr2 = join(' ', grep(!$basegroup{$_}++, sort split(' ',$groups)));
988174c1 137
dd570ea6 138if ($gr1 eq $gr2 || ($gr1 eq '' && $gr2 eq $pwgid)) {
988174c1 139 print "ok 1\n";
140}
141else {
142 print "#gr1 is <$gr1>\n";
143 print "#gr2 is <$gr2>\n";
144 print "not ok 1\n";
145}
146
147# multiple 0's indicate GROUPSTYPE is currently long but should be short
148
149if ($pwgid == 0 || $seen{0} < 2) {
150 print "ok 2\n";
151}
152else {
153 print "not ok 2 (groupstype should be type short, not long)\n";
154}