For tied file handle calls, use PUSH* when we know that the stack has space.
[p5sagit/p5-mst-13.2.git] / lib / h2xs.t
1 #!./perl -w
2
3 # Some quick tests to see if h2xs actually runs and creates files as 
4 # expected.  File contents include date stamps and/or usernames
5 # hence are not checked.  File existence is checked with -e though.
6 # This test depends on File::Path::rmtree() to clean up with.
7 #  - pvhp
8 #
9 # We are now checking that the correct use $version; is present in
10 # Makefile.PL and $module.pm
11
12 BEGIN {
13     chdir 't' if -d 't';
14     @INC = '../lib';
15     # FIXME (or rather FIXh2xs)
16     require Config;
17     if (($Config::Config{'extensions'} !~ m!\bDevel/PPPort\b!) ){
18         print "1..0 # Skip -- Perl configured without Devel::PPPort module\n";
19         exit 0;
20     }
21 }
22
23 # use strict; # we are not really testing this
24 use File::Path;  # for cleaning up with rmtree()
25 use Test::More;
26 use File::Spec;
27 use File::Find;
28 use ExtUtils::Manifest;
29 # Don't want its diagnostics getting in the way of ours.
30 $ExtUtils::Manifest::Quiet=1;
31 my $up = File::Spec->updir();
32
33 my $extracted_program = '../utils/h2xs'; # unix, nt, ...
34
35 my $Is_VMS_traildot = 0;
36 if ($^O eq 'VMS') {
37     $extracted_program = '[-.utils]h2xs.com';
38
39     # We have to know if VMS is in UNIX mode.  In UNIX mode, trailing dots
40     # should not be present.  There are actually two settings that control this.
41
42     $Is_VMS_traildot = 1;
43     my $unix_rpt = 0;
44     my $drop_dot = 0;
45     if (eval 'require VMS::Feature') {
46         $unix_rpt = VMS::Feature::current('filename_unix_report');
47         $drop_dot = VMS::Feature::current('readdir_dropdotnotype');
48     } else {
49         my $unix_report = $ENV{'DECC$FILENAME_UNIX_REPORT'} || '';
50         $unix_rpt = $unix_report =~ /^[ET1]/i; 
51         my $drop_dot_notype = $ENV{'DECC$READDIR_DROPDOTNOTYPE'} || '';
52         $drop_dot = $drop_dot_notype =~ /^[ET1]/i;
53     }
54     $Is_VMS_traildot = 0 if $drop_dot && unix_rpt;
55 }
56 if (!(-e $extracted_program)) {
57     print "1..0 # Skip: $extracted_program was not built\n";
58     exit 0;
59 }
60 # You might also wish to bail out if your perl platform does not
61 # do `$^X -e 'warn "Writing h2xst"' 2>&1`; duplicity.
62
63 # ok on unix, nt, VMS, ...
64 my $dupe = '2>&1';
65 # ok on unix, nt, The extra \" are for VMS
66 my $lib = '"-I../lib" "-I../../lib"';
67 # $name should differ from system header file names and must
68 # not already be found in the t/ subdirectory for perl.
69 my $name = 'h2xst';
70 my $header = "$name.h";
71 my $thisversion = sprintf "%vd", $^V;
72 $thisversion =~ s/^v//;
73
74 # If this test has failed previously a copy may be left.
75 rmtree($name);
76
77 my @tests = (
78 "-f -n $name", $], <<"EOXSFILES",
79 Defaulting to backwards compatibility with perl $thisversion
80 If you intend this module to be compatible with earlier perl versions, please
81 specify a minimum perl version with the -b option.
82
83 Writing $name/ppport.h
84 Writing $name/lib/$name.pm
85 Writing $name/$name.xs
86 Writing $name/fallback/const-c.inc
87 Writing $name/fallback/const-xs.inc
88 Writing $name/Makefile.PL
89 Writing $name/README
90 Writing $name/t/$name.t
91 Writing $name/Changes
92 Writing $name/MANIFEST
93 EOXSFILES
94
95 "-f -n $name -b $thisversion", $], <<"EOXSFILES",
96 Writing $name/ppport.h
97 Writing $name/lib/$name.pm
98 Writing $name/$name.xs
99 Writing $name/fallback/const-c.inc
100 Writing $name/fallback/const-xs.inc
101 Writing $name/Makefile.PL
102 Writing $name/README
103 Writing $name/t/$name.t
104 Writing $name/Changes
105 Writing $name/MANIFEST
106 EOXSFILES
107
108 "-f -n $name -b 5.6.1", "5.006001", <<"EOXSFILES",
109 Writing $name/ppport.h
110 Writing $name/lib/$name.pm
111 Writing $name/$name.xs
112 Writing $name/fallback/const-c.inc
113 Writing $name/fallback/const-xs.inc
114 Writing $name/Makefile.PL
115 Writing $name/README
116 Writing $name/t/$name.t
117 Writing $name/Changes
118 Writing $name/MANIFEST
119 EOXSFILES
120
121 "-f -n $name -b 5.5.3", "5.00503", <<"EOXSFILES",
122 Writing $name/ppport.h
123 Writing $name/lib/$name.pm
124 Writing $name/$name.xs
125 Writing $name/fallback/const-c.inc
126 Writing $name/fallback/const-xs.inc
127 Writing $name/Makefile.PL
128 Writing $name/README
129 Writing $name/t/$name.t
130 Writing $name/Changes
131 Writing $name/MANIFEST
132 EOXSFILES
133
134 "\"-X\" -f -n $name -b $thisversion", $], <<"EONOXSFILES",
135 Writing $name/lib/$name.pm
136 Writing $name/Makefile.PL
137 Writing $name/README
138 Writing $name/t/$name.t
139 Writing $name/Changes
140 Writing $name/MANIFEST
141 EONOXSFILES
142
143 "-f -n $name -b $thisversion $header", $], <<"EOXSFILES",
144 Writing $name/ppport.h
145 Writing $name/lib/$name.pm
146 Writing $name/$name.xs
147 Writing $name/fallback/const-c.inc
148 Writing $name/fallback/const-xs.inc
149 Writing $name/Makefile.PL
150 Writing $name/README
151 Writing $name/t/$name.t
152 Writing $name/Changes
153 Writing $name/MANIFEST
154 EOXSFILES
155 );
156
157 my $total_tests = 3; # opening, closing and deleting the header file.
158 for (my $i = $#tests; $i > 0; $i-=3) {
159   # 1 test for running it, 1 test for the expected result, and 1 for each file
160   # plus 1 to open and 1 to check for the use in lib/$name.pm and Makefile.PL
161   # And 1 more for our check for "bonus" files, 2 more for ExtUtil::Manifest.
162   # use the () to force list context and hence count the number of matches.
163   $total_tests += 9 + (() = $tests[$i] =~ /(Writing)/sg);
164 }
165
166 plan tests => $total_tests;
167
168 ok (open (HEADER, ">$header"), "open '$header'");
169 print HEADER <<HEADER or die $!;
170 #define Camel 2
171 #define Dromedary 1
172 HEADER
173 ok (close (HEADER), "close '$header'");
174
175 while (my ($args, $version, $expectation) = splice @tests, 0, 3) {
176   # h2xs warns about what it is writing hence the (possibly unportable)
177   # 2>&1 dupe:
178   # does it run?
179   my $prog = "$^X $lib $extracted_program $args $dupe";
180   @result = `$prog`;
181   cmp_ok ($?, "==", 0, "running $prog ");
182   $result = join("",@result);
183
184   #print "# expectation is >$expectation<\n";
185   #print "# result is >$result<\n";
186   # Was the output the list of files that were expected?
187   is ($result, $expectation, "running $prog");
188
189   my (%got);
190   find (sub {$got{$File::Find::name}++ unless -d $_}, $name);
191
192   foreach ($expectation =~ /Writing\s+(\S+)/gm) {
193     if ($^O eq 'VMS') {
194       if ($Is_VMS_traildot) {
195           $_ .= '.' unless $_ =~ m/\./;
196       }
197       $_ = lc($_) unless exists $got{$_};
198     }
199     ok (-e $_, "check for $_") and delete $got{$_};
200   }
201   my @extra = keys %got;
202   unless (ok (!@extra, "Are any extra files present?")) {
203     print "# These files are unexpectedly present:\n";
204     print "# $_\n" foreach sort @extra;
205   }
206
207   chdir ($name) or die "chdir $name failed: $!";
208   # Aargh. Something wants to load a bit of regexp. And we have to chdir
209   # for ExtUtils::Manifest. Caught between a rock and a hard place, so this
210   # seems the least evil thing to do:
211   push @INC, "../../lib";
212   my ($missing, $extra) = ExtUtils::Manifest::fullcheck();
213   is_deeply ($missing, [], "No files in the MANIFEST should be missing");
214   is_deeply ($extra, [],   "and all files present should be in the MANIFEST");
215   pop @INC;
216   chdir ($up) or die "chdir $up failed: $!";
217  
218   foreach my $leaf (File::Spec->catfile('lib', "$name.pm"), 'Makefile.PL') {
219     my $file = File::Spec->catfile($name, $leaf);
220     if (ok (open (FILE, $file), "open $file")) {
221       my $match = qr/use $version;/;
222       my $found;
223       while (<FILE>) {
224         last if $found = /$match/;
225       }
226       ok ($found, "looking for /$match/ in $file");
227       close FILE or die "close $file: $!";
228     }
229   }
230   # clean up
231   rmtree($name);
232 }
233
234 cmp_ok (unlink ($header), "==", 1, "unlink '$header'") or die "\$! is $!";