Skip to content

Commit df4e6ed

Browse files
committed
Refactor DirectoryCleaner to actually work with nested directories
This fixes Issue #216
1 parent 503d404 commit df4e6ed

File tree

2 files changed

+102
-9
lines changed

2 files changed

+102
-9
lines changed

src/shared/DirectoryCleaner.php

+18-9
Original file line numberDiff line numberDiff line change
@@ -10,20 +10,29 @@ public function process(FileInfo $path) {
1010
DirectoryCleanerException::SecurityLimitation
1111
);
1212
}
13-
if (!file_exists($path)) {
13+
14+
if (!$path->exists()) {
1415
return;
1516
}
16-
$worker = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($path->getPathname()));
17-
foreach($worker as $x) {
18-
if($x->getFilename() == "." || $x->getFilename() == "..") {
19-
continue;
20-
}
21-
if ($x->isDir()) {
22-
$this->process(new FileInfo($x->getPathname()));
17+
18+
$worker = new \RecursiveIteratorIterator(
19+
new \RecursiveDirectoryIterator(
20+
$path->getPathname(),
21+
\FilesystemIterator::SKIP_DOTS
22+
),
23+
\RecursiveIteratorIterator::CHILD_FIRST
24+
);
25+
26+
foreach($worker as $entry) {
27+
if ($entry->isDir() && !$entry->isLink()) {
28+
rmdir($entry->getPathname());
29+
} else {
30+
unlink($entry->getPathname());
2331
}
24-
unlink($x->getPathname());
2532
}
33+
rmdir($path);
2634
}
35+
2736
}
2837

2938
}

tests/Unit/DirectoryCleanerTest.php

+84
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
<?php
2+
/**
3+
* Copyright (c) 2015 Arne Blankerts <[email protected]>
4+
* All rights reserved.
5+
*
6+
* Redistribution and use in source and binary forms, with or without modification,
7+
* are permitted provided that the following conditions are met:
8+
*
9+
* * Redistributions of source code must retain the above copyright notice,
10+
* this list of conditions and the following disclaimer.
11+
*
12+
* * Redistributions in binary form must reproduce the above copyright notice,
13+
* this list of conditions and the following disclaimer in the documentation
14+
* and/or other materials provided with the distribution.
15+
*
16+
* * Neither the name of Arne Blankerts nor the names of contributors
17+
* may be used to endorse or promote products derived from this software
18+
* without specific prior written permission.
19+
*
20+
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21+
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT * NOT LIMITED TO,
22+
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
23+
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER ORCONTRIBUTORS
24+
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
25+
* OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27+
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28+
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29+
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30+
* POSSIBILITY OF SUCH DAMAGE.
31+
*
32+
* @package phpDox
33+
* @author Arne Blankerts <[email protected]>
34+
* @copyright Arne Blankerts <[email protected]>, All rights reserved.
35+
* @license BSD License
36+
*/
37+
namespace TheSeer\phpDox {
38+
39+
class DirectoryCleanerTest extends \PHPUnit_Framework_TestCase {
40+
41+
/**
42+
* @var DirectoryCleaner
43+
*/
44+
private $cleaner;
45+
46+
protected function setUp() {
47+
$this->cleaner = new DirectoryCleaner();
48+
}
49+
50+
/**
51+
* @expectedException \TheSeer\phpDox\DirectoryCleanerException
52+
* @expectedExceptionCode \TheSeer\phpDox\DirectoryCleanerException::SecurityLimitation
53+
*/
54+
public function testTryingToDeleteAShortPathThrowsException() {
55+
$this->cleaner->process(new FileInfo('/tmp'));
56+
}
57+
58+
public function testTryingToDeleteNonExistingDirectoryJustReturns() {
59+
$this->cleaner->process(new FileInfo('/not/existing/directory'));
60+
$this->assertTrue(true);
61+
}
62+
63+
public function testCanDeleteRecursiveDirectoryStructure() {
64+
$base = '/tmp/'. uniqid('dctest-');
65+
$path = $base . '/a/b/c/d/e/f/g/h';
66+
mkdir( $path, 0700, true);
67+
touch( $path . '/test-h.txt' );
68+
touch( $path . '/../test-g.txt' );
69+
touch( $path . '/../../test-f.txt' );
70+
71+
$this->assertTrue(file_exists($path. '/test-h.txt'));
72+
$this->assertTrue(is_dir($path));
73+
74+
$this->cleaner->process(new FileInfo($base));
75+
76+
$this->assertFalse(file_exists($path. '/test-h.txt'), 'File vanished');
77+
$this->assertFalse(is_dir($base), 'Directory vanished');
78+
79+
}
80+
81+
82+
}
83+
84+
}

0 commit comments

Comments
 (0)