1#!/usr/bin/perl
2
3#
4# Check SPLICE function's effect on the file
5# (07_rv_splice.t checks its return value)
6#
7# Each call to 'check_contents' actually performs two tests.
8# First, it calls the tied object's own 'check_integrity' method,
9# which makes sure that the contents of the read cache and offset tables
10# accurately reflect the contents of the file.
11# Then, it checks the actual contents of the file against the expected
12# contents.
13
14
15$| = 1;
16my $file = "tf$$.txt";
17$: = Tie::File::_default_recsep();
18my $data = "rec0$:rec1$:rec2$:";
19print "1..118\n";
20
21init_file($data);
22
23my $N = 1;
24use Tie::File;
25print "ok $N\n"; $N++;  # partial credit just for showing up
26
27my $o = tie @a, 'Tie::File', $file;
28print $o ? "ok $N\n" : "not ok $N\n";
29$N++;
30
31$: = $o->{recsep};
32my $n;
33
34# (3-22) splicing at the beginning
35splice(@a, 0, 0, "rec4");
36check_contents("rec4$:$data");
37splice(@a, 0, 1, "rec5");       # same length
38check_contents("rec5$:$data");
39splice(@a, 0, 1, "record5");    # longer
40check_contents("record5$:$data");
41
42splice(@a, 0, 1, "r5");         # shorter
43check_contents("r5$:$data");
44splice(@a, 0, 1);               # removal
45check_contents("$data");
46splice(@a, 0, 0);               # no-op
47check_contents("$data");
48splice(@a, 0, 0, 'r7', 'rec8'); # insert more than one
49check_contents("r7$:rec8$:$data");
50splice(@a, 0, 2, 'rec7', 'record8', 'rec9'); # insert more than delete
51check_contents("rec7$:record8$:rec9$:$data");
52
53splice(@a, 0, 3, 'record9', 'rec10'); # delete more than insert
54check_contents("record9$:rec10$:$data");
55splice(@a, 0, 2);               # delete more than one
56check_contents("$data");
57
58
59# (23-42) splicing in the middle
60splice(@a, 1, 0, "rec4");
61check_contents("rec0$:rec4$:rec1$:rec2$:");
62splice(@a, 1, 1, "rec5");       # same length
63check_contents("rec0$:rec5$:rec1$:rec2$:");
64splice(@a, 1, 1, "record5");    # longer
65check_contents("rec0$:record5$:rec1$:rec2$:");
66
67splice(@a, 1, 1, "r5");         # shorter
68check_contents("rec0$:r5$:rec1$:rec2$:");
69splice(@a, 1, 1);               # removal
70check_contents("$data");
71splice(@a, 1, 0);               # no-op
72check_contents("$data");
73splice(@a, 1, 0, 'r7', 'rec8'); # insert more than one
74check_contents("rec0$:r7$:rec8$:rec1$:rec2$:");
75splice(@a, 1, 2, 'rec7', 'record8', 'rec9'); # insert more than delete
76check_contents("rec0$:rec7$:record8$:rec9$:rec1$:rec2$:");
77
78splice(@a, 1, 3, 'record9', 'rec10'); # delete more than insert
79check_contents("rec0$:record9$:rec10$:rec1$:rec2$:");
80splice(@a, 1, 2);               # delete more than one
81check_contents("$data");
82
83# (43-62) splicing at the end
84splice(@a, 3, 0, "rec4");
85check_contents("$ {data}rec4$:");
86splice(@a, 3, 1, "rec5");       # same length
87check_contents("$ {data}rec5$:");
88splice(@a, 3, 1, "record5");    # longer
89check_contents("$ {data}record5$:");
90
91splice(@a, 3, 1, "r5");         # shorter
92check_contents("$ {data}r5$:");
93splice(@a, 3, 1);               # removal
94check_contents("$data");
95splice(@a, 3, 0);               # no-op
96check_contents("$data");
97splice(@a, 3, 0, 'r7', 'rec8'); # insert more than one
98check_contents("$ {data}r7$:rec8$:");
99splice(@a, 3, 2, 'rec7', 'record8', 'rec9'); # insert more than delete
100check_contents("$ {data}rec7$:record8$:rec9$:");
101
102splice(@a, 3, 3, 'record9', 'rec10'); # delete more than insert
103check_contents("$ {data}record9$:rec10$:");
104splice(@a, 3, 2);               # delete more than one
105check_contents("$data");
106
107# (63-82) splicing with negative subscript
108splice(@a, -1, 0, "rec4");
109check_contents("rec0$:rec1$:rec4$:rec2$:");
110splice(@a, -1, 1, "rec5");       # same length
111check_contents("rec0$:rec1$:rec4$:rec5$:");
112splice(@a, -1, 1, "record5");    # longer
113check_contents("rec0$:rec1$:rec4$:record5$:");
114
115splice(@a, -1, 1, "r5");         # shorter
116check_contents("rec0$:rec1$:rec4$:r5$:");
117splice(@a, -1, 1);               # removal
118check_contents("rec0$:rec1$:rec4$:");
119splice(@a, -1, 0);               # no-op
120check_contents("rec0$:rec1$:rec4$:");
121splice(@a, -1, 0, 'r7', 'rec8'); # insert more than one
122check_contents("rec0$:rec1$:r7$:rec8$:rec4$:");
123splice(@a, -1, 2, 'rec7', 'record8', 'rec9'); # insert more than delete
124check_contents("rec0$:rec1$:r7$:rec8$:rec7$:record8$:rec9$:");
125
126splice(@a, -3, 3, 'record9', 'rec10'); # delete more than insert
127check_contents("rec0$:rec1$:r7$:rec8$:record9$:rec10$:");
128splice(@a, -4, 3);               # delete more than one
129check_contents("rec0$:rec1$:rec10$:");
130
131# (83-84) scrub it all out
132splice(@a, 0, 3);
133check_contents("");
134
135# (85-86) put some back in
136splice(@a, 0, 0, "rec0", "rec1");
137check_contents("rec0$:rec1$:");
138
139# (87-88) what if we remove too many records?
140splice(@a, 0, 17);
141check_contents("");
142
143# (89-92) In the past, splicing past the end was not correctly detected
144# (1.14)
145splice(@a, 89, 3);
146check_contents("");
147splice(@a, @a, 3);
148check_contents("");
149
150# (93-96) Also we did not emulate splice's freaky behavior when inserting
151# past the end of the array (1.14)
152splice(@a, 89, 0, "I", "like", "pie");
153check_contents("I$:like$:pie$:");
154splice(@a, 89, 0, "pie pie pie");
155check_contents("I$:like$:pie$:pie pie pie$:");
156
157# (97) Splicing with too large a negative number should be fatal
158# This test ignored because it causes 5.6.1 and 5.7.3 to dump core
159# It also garbles the stack under 5.005_03 (20020401)
160# NOT MY FAULT
161if ($] > 5.007003) {
162  eval { splice(@a, -7, 0) };
163  print $@ =~ /^Modification of non-creatable array value attempted, subscript -7/
164      ? "ok $N\n" : "not ok $N \# \$\@ was '$@'\n";
165} else {
166  print "ok $N \# skipped (versions through 5.7.3 dump core here.)\n";
167}
168$N++;
169
170# (98-101) Test default arguments
171splice @a, 0, 0, (0..11);
172splice @a, 4;
173check_contents("0$:1$:2$:3$:");
174splice @a;
175check_contents("");
176
177# (102-103) I think there's a bug here---it will fail to clear the EOF flag
178@a = (0..11);
179splice @a, -1, 1000;
180check_contents("0$:1$:2$:3$:4$:5$:6$:7$:8$:9$:10$:");
181
182# (104-106) make sure that undefs are treated correctly---they should
183# be converted to empty records, and should not raise any warnings.
184# (Some of these failed in 0.90.  The change to _fixrec fixed them.)
185# 20020331
186{
187  my $good = 1; my $warn;
188  # If any of these raise warnings, we have a problem.
189  local $SIG{__WARN__} = sub { $good = 0; $warn = shift(); ctrlfix($warn)};
190  local $^W = 1;
191  @a = (1);
192  splice @a, 1, 0, undef, undef, undef;
193  print $good ? "ok $N\n" : "not ok $N # $warn\n";
194  $N++; $good = 1;
195  print defined($a[2]) ? "ok $N\n" : "not ok $N\n";
196  $N++; $good = 1;
197  my @r = splice @a, 2;
198  print defined($r[0]) ? "ok $N\n" : "not ok $N\n";
199  $N++; $good = 1;
200}
201
202# (107-118) splice with negative length was treated wrong
203# 20020402 Reported by Juerd Waalboer
204@a = (0..8) ;
205splice @a, 0, -3;
206check_contents("6$:7$:8$:");
207@a = (0..8) ;
208splice @a, 1, -3;
209check_contents("0$:6$:7$:8$:");
210@a = (0..8) ;
211splice @a, 7, -3;
212check_contents("0$:1$:2$:3$:4$:5$:6$:7$:8$:");
213@a = (0..2) ;
214splice @a, 0, -3;
215check_contents("0$:1$:2$:");
216@a = (0..2) ;
217splice @a, 1, -3;
218check_contents("0$:1$:2$:");
219@a = (0..2) ;
220splice @a, 7, -3;
221check_contents("0$:1$:2$:");
222
223sub init_file {
224  my $data = shift;
225  open F, "> $file" or die $!;
226  binmode F;
227  print F $data;
228  close F;
229}
230
231use POSIX 'SEEK_SET';
232sub check_contents {
233  my $x = shift;
234  my $integrity = $o->_check_integrity($file, $ENV{INTEGRITY});
235  local *FH = $o->{fh};
236  seek FH, 0, SEEK_SET;
237  print $integrity ? "ok $N\n" : "not ok $N\n";
238  $N++;
239  my $a;
240  { local $/; $a = <FH> }
241  $a = "" unless defined $a;
242  if ($a eq $x) {
243    print "ok $N\n";
244  } else {
245    ctrlfix($a, $x);
246    print "not ok $N\n# expected <$x>, got <$a>\n";
247  }
248  $N++;
249}
250
251
252sub ctrlfix {
253  for (@_) {
254    s/\n/\\n/g;
255    s/\r/\\r/g;
256  }
257}
258
259END {
260  undef $o;
261  untie @a;
262  1 while unlink $file;
263}
264
265