3 # This file is part of GNU Stow.
5 # GNU Stow is free software: you can redistribute it and/or modify it
6 # under the terms of the GNU General Public License as published by
7 # the Free Software Foundation, either version 3 of the License, or
8 # (at your option) any later version.
10 # GNU Stow is distributed in the hope that it will be useful, but
11 # WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 # General Public License for more details.
15 # You should have received a copy of the GNU General Public License
16 # along with this program. If not, see https://www.gnu.org/licenses/.
19 # Test unstowing packages
25 use Test
::More tests
=> 39;
27 use English
qw(-no_match_vars);
30 use Stow
::Util
qw(canon_path);
33 cd
("$TEST_DIR/target");
35 # Note that each of the following tests use a distinct set of files
41 # unstow a simple tree minimally
45 make_path
('../stow/pkg1/bin1');
46 make_file
('../stow/pkg1/bin1/file1');
47 make_link
('bin1', '../stow/pkg1/bin1');
49 $stow->plan_unstow('pkg1');
50 $stow->process_tasks();
52 $stow->get_conflict_count == 0 &&
53 -f
'../stow/pkg1/bin1/file1' && ! -e
'bin1'
54 => 'unstow a simple tree'
58 # unstow a simple tree from an existing directory
63 make_path
('../stow/pkg2/lib2');
64 make_file
('../stow/pkg2/lib2/file2');
65 make_link
('lib2/file2', '../../stow/pkg2/lib2/file2');
66 $stow->plan_unstow('pkg2');
67 $stow->process_tasks();
69 $stow->get_conflict_count == 0 &&
70 -f
'../stow/pkg2/lib2/file2' && -d
'lib2'
71 => 'unstow simple tree from a pre-existing directory'
75 # fold tree after unstowing
81 make_path
('../stow/pkg3a/bin3');
82 make_file
('../stow/pkg3a/bin3/file3a');
83 make_link
('bin3/file3a' => '../../stow/pkg3a/bin3/file3a'); # emulate stow
85 make_path
('../stow/pkg3b/bin3');
86 make_file
('../stow/pkg3b/bin3/file3b');
87 make_link
('bin3/file3b' => '../../stow/pkg3b/bin3/file3b'); # emulate stow
88 $stow->plan_unstow('pkg3b');
89 $stow->process_tasks();
91 $stow->get_conflict_count == 0 &&
93 readlink('bin3') eq '../stow/pkg3a/bin3'
94 => 'fold tree after unstowing'
98 # existing link is owned by stow but is invalid so it gets removed anyway
103 make_path
('../stow/pkg4/bin4');
104 make_file
('../stow/pkg4/bin4/file4');
105 make_invalid_link
('bin4/file4', '../../stow/pkg4/bin4/does-not-exist');
107 $stow->plan_unstow('pkg4');
108 $stow->process_tasks();
110 $stow->get_conflict_count == 0 &&
112 => q
(remove invalid
link owned by stow
)
116 # Existing link is not owned by stow
120 make_path
('../stow/pkg5/bin5');
121 make_invalid_link
('bin5', '../not-stow');
123 $stow->plan_unstow('pkg5');
124 %conflicts = $stow->get_conflicts;
126 $conflicts{unstow
}{pkg5
}[-1],
127 qr
(existing target is
not owned by stow
)
128 => q
(existing
link not owned by stow
)
132 # Target already exists, is owned by stow, but points to a different package
137 make_path
('../stow/pkg6a/bin6');
138 make_file
('../stow/pkg6a/bin6/file6');
139 make_link
('bin6/file6', '../../stow/pkg6a/bin6/file6');
141 make_path
('../stow/pkg6b/bin6');
142 make_file
('../stow/pkg6b/bin6/file6');
144 $stow->plan_unstow('pkg6b');
146 $stow->get_conflict_count == 0 &&
148 readlink('bin6/file6') eq '../../stow/pkg6a/bin6/file6'
149 => q
(ignore existing
link that points to a different
package)
153 # Don't unlink anything under the stow directory
155 make_path
('stow'); # make out stow dir a subdir of target
156 $stow = new_Stow
(dir
=> 'stow');
158 # emulate stowing into ourself (bizarre corner case or accident)
159 make_path
('stow/pkg7a/stow/pkg7b');
160 make_file
('stow/pkg7a/stow/pkg7b/file7b');
161 make_link
('stow/pkg7b', '../stow/pkg7a/stow/pkg7b');
163 $stow->plan_unstow('pkg7b');
164 is
($stow->get_tasks, 0, 'no tasks to process when unstowing pkg7b');
166 $stow->get_conflict_count == 0 &&
168 readlink('stow/pkg7b') eq '../stow/pkg7a/stow/pkg7b'
169 => q
(don
't unlink any nodes under the stow directory)
174 # Don't
unlink any nodes under another stow directory
176 $stow = new_Stow
(dir
=> 'stow');
178 make_path
('stow2'); # make our alternate stow dir a subdir of target
179 make_file
('stow2/.stow');
181 # emulate stowing into ourself (bizarre corner case or accident)
182 make_path
('stow/pkg8a/stow2/pkg8b');
183 make_file
('stow/pkg8a/stow2/pkg8b/file8b');
184 make_link
('stow2/pkg8b', '../stow/pkg8a/stow2/pkg8b');
187 sub { $stow->plan_unstow('pkg8a'); },
188 qr/WARNING: skipping marked Stow directory stow2/
189 => "unstowing from ourself should skip stow"
191 is
($stow->get_tasks, 0, 'no tasks to process when unstowing pkg8a');
193 $stow->get_conflict_count == 0 &&
195 readlink('stow2/pkg8b') eq '../stow/pkg8a/stow2/pkg8b'
196 => q
(don
't unlink any nodes under another stow directory)
200 # overriding already stowed documentation
202 $stow = new_Stow(override => ['man9
', 'info9
']);
203 make_file('stow
/.stow
');
205 make_path('../stow/pkg
9a
/man9/man
1');
206 make_file('../stow/pkg
9a
/man9/man
1/file9
.1
');
207 make_path('man9
/man1
');
208 make_link('man9
/man1/file9
.1
' => '../../../stow/pkg
9a
/man9/man
1/file9
.1
'); # emulate stow
210 make_path('../stow/pkg
9b
/man9/man
1');
211 make_file('../stow/pkg
9b
/man9/man
1/file9
.1
');
212 $stow->plan_unstow('pkg9b
');
213 $stow->process_tasks();
215 $stow->get_conflict_count == 0 &&
216 !-l 'man9
/man1/file9
.1
'
217 => 'overriding existing documentation files
'
221 # deferring to already stowed documentation
223 $stow = new_Stow(defer => ['man10
', 'info10
']);
225 make_path('../stow/pkg
10a
/man10/man
1');
226 make_file('../stow/pkg
10a
/man10/man
1/file10a
.1');
227 make_path('man10
/man1
');
228 make_link('man10
/man1/file10a
.1' => '../../../stow/pkg
10a
/man10/man
1/file10a
.1');
230 # need this to block folding
231 make_path('../stow/pkg
10b
/man10/man
1');
232 make_file('../stow/pkg
10b
/man10/man
1/file10b
.1');
233 make_link('man10
/man1/file10b
.1' => '../../../stow/pkg
10b
/man10/man
1/file10b
.1');
236 make_path('../stow/pkg
10c
/man10/man
1');
237 make_file('../stow/pkg
10c
/man10/man
1/file10a
.1');
238 $stow->plan_unstow('pkg10c
');
239 is($stow->get_tasks, 0, 'no tasks to process
when unstowing pkg10c
');
241 $stow->get_conflict_count == 0 &&
242 readlink('man10
/man1/file10a
.1') eq '../../../stow/pkg
10a
/man10/man
1/file10a
.1'
243 => 'defer to existing documentation files
'
249 $stow = new_Stow(ignore => ['~', '\
.#.*']);
251 make_path
('../stow/pkg12/man12/man1');
252 make_file
('../stow/pkg12/man12/man1/file12.1');
253 make_file
('../stow/pkg12/man12/man1/file12.1~');
254 make_file
('../stow/pkg12/man12/man1/.#file12.1');
255 make_path
('man12/man1');
256 make_link
('man12/man1/file12.1' => '../../../stow/pkg12/man12/man1/file12.1');
258 $stow->plan_unstow('pkg12');
259 $stow->process_tasks();
261 $stow->get_conflict_count == 0 &&
262 !-e
'man12/man1/file12.1'
263 => 'ignore temp files'
267 # Unstow an already unstowed package
270 $stow->plan_unstow('pkg12');
271 is
($stow->get_tasks, 0, 'no tasks to process when unstowing pkg12');
273 $stow->get_conflict_count == 0
274 => 'unstow already unstowed package pkg12'
278 # Unstow a never stowed package
281 eval { remove_dir
("$TEST_DIR/target"); };
282 mkdir("$TEST_DIR/target");
285 $stow->plan_unstow('pkg12');
286 is
($stow->get_tasks, 0, 'no tasks to process when unstowing pkg12 which was never stowed');
288 $stow->get_conflict_count == 0
289 => 'unstow never stowed package pkg12'
293 # Unstowing when target contains a real file shouldn't be an issue.
295 make_file
('man12/man1/file12.1');
298 $stow->plan_unstow('pkg12');
299 is
($stow->get_tasks, 0, 'no tasks to process when unstowing pkg12 for third time');
300 %conflicts = $stow->get_conflicts;
302 $stow->get_conflict_count == 1 &&
303 $conflicts{unstow
}{pkg12
}[0]
304 =~ m!existing target is neither a link nor a directory: man12/man1/file12\.1!
305 => 'unstow pkg12 for third time'
309 # unstow a simple tree minimally when cwd isn't target
312 $stow = new_Stow
(dir
=> "$TEST_DIR/stow", target
=> "$TEST_DIR/target");
314 make_path
("$TEST_DIR/stow/pkg13/bin13");
315 make_file
("$TEST_DIR/stow/pkg13/bin13/file13");
316 make_link
("$TEST_DIR/target/bin13", '../stow/pkg13/bin13');
318 $stow->plan_unstow('pkg13');
319 $stow->process_tasks();
321 $stow->get_conflict_count == 0 &&
322 -f
"$TEST_DIR/stow/pkg13/bin13/file13" && ! -e
"$TEST_DIR/target/bin13"
323 => 'unstow a simple tree'
327 # unstow a simple tree minimally with absolute stow dir when cwd isn't
330 $stow = new_Stow
(dir
=> canon_path
("$TEST_DIR/stow"),
331 target
=> "$TEST_DIR/target");
333 make_path
("$TEST_DIR/stow/pkg14/bin14");
334 make_file
("$TEST_DIR/stow/pkg14/bin14/file14");
335 make_link
("$TEST_DIR/target/bin14", '../stow/pkg14/bin14');
337 $stow->plan_unstow('pkg14');
338 $stow->process_tasks();
340 $stow->get_conflict_count == 0 &&
341 -f
"$TEST_DIR/stow/pkg14/bin14/file14" && ! -e
"$TEST_DIR/target/bin14"
342 => 'unstow a simple tree with absolute stow dir'
346 # unstow a simple tree minimally with absolute stow AND target dirs
347 # when cwd isn't target
349 $stow = new_Stow
(dir
=> canon_path
("$TEST_DIR/stow"),
350 target
=> canon_path
("$TEST_DIR/target"));
352 make_path
("$TEST_DIR/stow/pkg15/bin15");
353 make_file
("$TEST_DIR/stow/pkg15/bin15/file15");
354 make_link
("$TEST_DIR/target/bin15", '../stow/pkg15/bin15');
356 $stow->plan_unstow('pkg15');
357 $stow->process_tasks();
359 $stow->get_conflict_count == 0 &&
360 -f
"$TEST_DIR/stow/pkg15/bin15/file15" && ! -e
"$TEST_DIR/target/bin15"
361 => 'unstow a simple tree with absolute stow and target dirs'
365 # unstow a tree with no-folding enabled -
366 # no refolding should take place
368 cd
("$TEST_DIR/target");
370 sub create_and_stow_pkg
{
373 my $stow_pkg = "../stow/$id-$pkg";
374 make_path
($stow_pkg);
375 make_file
("$stow_pkg/$id-file-$pkg");
377 # create a shallow hierarchy specific to this package and stow
379 make_path
("$stow_pkg/$id-$pkg-only-folded");
380 make_file
("$stow_pkg/$id-$pkg-only-folded/file-$pkg");
381 make_link
("$id-$pkg-only-folded", "$stow_pkg/$id-$pkg-only-folded");
383 # create a deeper hierarchy specific to this package and stow
385 make_path
("$stow_pkg/$id-$pkg-only-folded2/subdir");
386 make_file
("$stow_pkg/$id-$pkg-only-folded2/subdir/file-$pkg");
387 make_link
("$id-$pkg-only-folded2",
388 "$stow_pkg/$id-$pkg-only-folded2");
390 # create a shallow hierarchy specific to this package and stow
392 make_path
("$stow_pkg/$id-$pkg-only-unfolded");
393 make_file
("$stow_pkg/$id-$pkg-only-unfolded/file-$pkg");
394 make_path
("$id-$pkg-only-unfolded");
395 make_link
("$id-$pkg-only-unfolded/file-$pkg",
396 "../$stow_pkg/$id-$pkg-only-unfolded/file-$pkg");
398 # create a deeper hierarchy specific to this package and stow
400 make_path
("$stow_pkg/$id-$pkg-only-unfolded2/subdir");
401 make_file
("$stow_pkg/$id-$pkg-only-unfolded2/subdir/file-$pkg");
402 make_path
("$id-$pkg-only-unfolded2/subdir");
403 make_link
("$id-$pkg-only-unfolded2/subdir/file-$pkg",
404 "../../$stow_pkg/$id-$pkg-only-unfolded2/subdir/file-$pkg");
406 # create a shallow shared hierarchy which this package uses, and stow
407 # its contents without folding
408 make_path
("$stow_pkg/$id-shared");
409 make_file
("$stow_pkg/$id-shared/file-$pkg");
410 make_path
("$id-shared");
411 make_link
("$id-shared/file-$pkg",
412 "../$stow_pkg/$id-shared/file-$pkg");
414 # create a deeper shared hierarchy which this package uses, and stow
415 # its contents without folding
416 make_path
("$stow_pkg/$id-shared2/subdir");
417 make_file
("$stow_pkg/$id-shared2/file-$pkg");
418 make_file
("$stow_pkg/$id-shared2/subdir/file-$pkg");
419 make_path
("$id-shared2/subdir");
420 make_link
("$id-shared2/file-$pkg",
421 "../$stow_pkg/$id-shared2/file-$pkg");
422 make_link
("$id-shared2/subdir/file-$pkg",
423 "../../$stow_pkg/$id-shared2/subdir/file-$pkg");
426 foreach my $pkg (qw{a b
}) {
427 create_and_stow_pkg
('no-folding', $pkg);
430 $stow = new_Stow
('no-folding' => 1);
431 $stow->plan_unstow('no-folding-b');
432 is_deeply
([ $stow->get_conflicts ], [] => 'no conflicts with --no-folding');
434 #warn Dumper($stow->get_tasks);
436 $stow->process_tasks();
438 is_nonexistent_path
('no-folding-b-only-folded');
439 is_nonexistent_path
('no-folding-b-only-folded2');
440 is_nonexistent_path
('no-folding-b-only-unfolded/file-b');
441 is_nonexistent_path
('no-folding-b-only-unfolded2/subdir/file-b');
442 is_dir_not_symlink
('no-folding-shared');
443 is_dir_not_symlink
('no-folding-shared2');
444 is_dir_not_symlink
('no-folding-shared2/subdir');
449 # Test cleaning up subdirs with --paranoid option