1 | <?php |
||||
2 | |||||
3 | namespace LeKoala\DevToolkit\Tasks; |
||||
4 | |||||
5 | use FilesystemIterator; |
||||
6 | use SilverStripe\ORM\DB; |
||||
7 | use SilverStripe\Assets\File; |
||||
8 | use RecursiveIteratorIterator; |
||||
9 | use RecursiveDirectoryIterator; |
||||
10 | use SilverStripe\Dev\BuildTask; |
||||
11 | use SilverStripe\ORM\DataObject; |
||||
12 | use SilverStripe\Core\Environment; |
||||
13 | use SilverStripe\Core\Config\Config; |
||||
14 | use SilverStripe\Versioned\Versioned; |
||||
0 ignored issues
–
show
|
|||||
15 | use LeKoala\DevToolkit\BuildTaskTools; |
||||
16 | use SilverStripe\Assets\Flysystem\ProtectedAssetAdapter; |
||||
17 | use SilverStripe\Assets\Folder; |
||||
18 | |||||
19 | /** |
||||
20 | * @author lekoala |
||||
21 | */ |
||||
22 | class DropInvalidFilesTask extends BuildTask |
||||
23 | { |
||||
24 | use BuildTaskTools; |
||||
25 | |||||
26 | protected $title = "Drop Invalid Files"; |
||||
27 | protected $description = 'Drop file objects that are not linked to a proper asset (warning ! experimental)'; |
||||
28 | private static $segment = 'DropInvalidFilesTask'; |
||||
0 ignored issues
–
show
|
|||||
29 | |||||
30 | public function run($request) |
||||
31 | { |
||||
32 | $this->request = $request; |
||||
33 | |||||
34 | $this->addOption("go", "Tick this to proceed", false); |
||||
35 | $this->addOption("remove_files", "Remove db files", false); |
||||
36 | $this->addOption("remove_local", "Remove local files", false); |
||||
37 | |||||
38 | $options = $this->askOptions(); |
||||
39 | |||||
40 | $go = $options['go']; |
||||
41 | $remove_files = $options['remove_files']; |
||||
42 | $remove_local = $options['remove_local']; |
||||
43 | |||||
44 | if (!$go) { |
||||
45 | echo ('Previewing what this task is about to do.'); |
||||
46 | } else { |
||||
47 | echo ("Let's clean this up!"); |
||||
48 | } |
||||
49 | echo ('<hr/>'); |
||||
50 | if ($remove_files) { |
||||
51 | $this->removeFiles($request, $go); |
||||
52 | } |
||||
53 | if ($remove_local) { |
||||
54 | $this->removeLocalFiles($request, $go); |
||||
55 | } |
||||
56 | } |
||||
57 | |||||
58 | protected function removeLocalFiles($request, $go = false) |
||||
0 ignored issues
–
show
The parameter
$request is not used and could be removed.
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
This check looks for parameters that have been defined for a function or method, but which are not used in the method body. ![]() |
|||||
59 | { |
||||
60 | $iter = new RecursiveDirectoryIterator(ASSETS_PATH); |
||||
61 | $iter2 = new RecursiveIteratorIterator($iter); |
||||
62 | |||||
63 | foreach ($iter2 as $file) { |
||||
64 | // Ignore roots and _ |
||||
65 | $startsWithSlash = strpos($file->getName(), '_') === 0; |
||||
66 | $hasVariant = strpos($file->getName(), '__') !== false; |
||||
67 | if ($startsWithSlash || $hasVariant) { |
||||
68 | // $this->message("Ignore " . $file->getPath()); |
||||
69 | continue; |
||||
70 | } |
||||
71 | |||||
72 | // Check for empty dirs |
||||
73 | if ($file->isDir()) { |
||||
74 | // ignores .dot files |
||||
75 | $dirFiles = scandir($file->getPath()); |
||||
76 | $empty = (count($dirFiles) - 2) === 0; |
||||
77 | if ($empty) { |
||||
78 | $this->message($file->getPath() . " is empty"); |
||||
79 | if ($go) { |
||||
80 | rmdir($file->getPath()); |
||||
81 | } |
||||
82 | } |
||||
83 | continue; |
||||
84 | } |
||||
85 | |||||
86 | // Check for files not matching anything in the db |
||||
87 | $thisPath = str_replace(ASSETS_PATH, "", $file->getPath()); |
||||
0 ignored issues
–
show
|
|||||
88 | // $this->message($thisPath); |
||||
89 | // $dbFile = File::get()->filter("FileFilename", $thisPath); |
||||
90 | } |
||||
91 | } |
||||
92 | |||||
93 | protected function removeFiles($request, $go = false) |
||||
0 ignored issues
–
show
The parameter
$request is not used and could be removed.
(
Ignorable by Annotation
)
If this is a false-positive, you can also ignore this issue in your code via the
This check looks for parameters that have been defined for a function or method, but which are not used in the method body. ![]() |
|||||
94 | { |
||||
95 | $conn = DB::get_conn(); |
||||
96 | $schema = DB::get_schema(); |
||||
97 | $dataObjectSchema = DataObject::getSchema(); |
||||
0 ignored issues
–
show
|
|||||
98 | $tableList = $schema->tableList(); |
||||
0 ignored issues
–
show
|
|||||
99 | |||||
100 | $files = File::get(); |
||||
101 | |||||
102 | if ($go) { |
||||
103 | $conn->transactionStart(); |
||||
104 | } |
||||
105 | |||||
106 | $i = 0; |
||||
107 | |||||
108 | /** @var File $file */ |
||||
109 | foreach ($files as $file) { |
||||
110 | if ($file instanceof Folder) { |
||||
111 | continue; |
||||
112 | } |
||||
113 | $path = self::getFullPath($file); |
||||
114 | $hashPath = self::getHashPath($file); |
||||
115 | if (!trim($file->getRelativePath(), '/')) { |
||||
116 | $this->message("#{$file->ID}: path is empty"); |
||||
117 | if ($go) { |
||||
118 | // $file->delete(); |
||||
119 | self::deleteFile($file->ID); |
||||
120 | $i++; |
||||
121 | } |
||||
122 | } |
||||
123 | if (!file_exists($path) && !file_exists($hashPath)) { |
||||
124 | $this->message("#{$file->ID}: $path does not exist"); |
||||
125 | if ($go) { |
||||
126 | // $file->delete(); |
||||
127 | self::deleteFile($file->ID); |
||||
128 | $i++; |
||||
129 | } |
||||
130 | } else { |
||||
131 | $this->message("#{$file->ID}: $path is valid", "success"); |
||||
132 | } |
||||
133 | if ($go && $i % 100 == 0) { |
||||
134 | $conn->transactionEnd(); |
||||
135 | $conn->transactionStart(); |
||||
136 | } |
||||
137 | } |
||||
138 | |||||
139 | if ($go) { |
||||
140 | $conn->transactionEnd(); |
||||
141 | } |
||||
142 | } |
||||
143 | |||||
144 | /** |
||||
145 | * ORM is just too slow for this |
||||
146 | * |
||||
147 | * @param int $ID |
||||
148 | * @return void |
||||
149 | */ |
||||
150 | public static function deleteFile($ID) |
||||
151 | { |
||||
152 | DB::prepared_query("DELETE FROM File WHERE ID = ?", [$ID]); |
||||
153 | DB::prepared_query("DELETE FROM File_Live WHERE ID = ?", [$ID]); |
||||
154 | DB::prepared_query("DELETE FROM File_Versions WHERE RecordID = ?", [$ID]); |
||||
155 | DB::prepared_query("DELETE FROM File_ViewerGroups WHERE FileID = ?", [$ID]); |
||||
156 | DB::prepared_query("DELETE FROM File_EditorGroups WHERE FileID = ?", [$ID]); |
||||
157 | } |
||||
158 | |||||
159 | public static function getFullPath(File $file) |
||||
160 | { |
||||
161 | return ASSETS_PATH . '/' . $file->getRelativePath(); |
||||
162 | } |
||||
163 | |||||
164 | public static function getHashPath(File $file) |
||||
165 | { |
||||
166 | $path = $file->getRelativePath(); |
||||
167 | $parts = explode('/', $path); |
||||
168 | $name = array_pop($parts); |
||||
169 | $folder = implode("/", $parts); |
||||
170 | |||||
171 | $full = ASSETS_PATH . '/' . $folder . '/' . substr($file->getHash(), 0, 10) . '/' . $name; |
||||
172 | $full = str_replace('//', '/', $full); |
||||
173 | return $full; |
||||
174 | } |
||||
175 | |||||
176 | public function getProtectedFullPath(File $file) |
||||
177 | { |
||||
178 | return self::getBaseProtectedPath() . '/' . $file->getRelativePath(); |
||||
179 | } |
||||
180 | |||||
181 | public static function getBaseProtectedPath() |
||||
182 | { |
||||
183 | // Use environment defined path or default location is under assets |
||||
184 | if ($path = Environment::getEnv('SS_PROTECTED_ASSETS_PATH')) { |
||||
185 | return $path; |
||||
186 | } |
||||
187 | |||||
188 | // Default location |
||||
189 | return ASSETS_PATH . '/' . Config::inst()->get(ProtectedAssetAdapter::class, 'secure_folder'); |
||||
190 | } |
||||
191 | } |
||||
192 |
The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g.
excluded_paths: ["lib/*"]
, you can move it to the dependency path list as follows:For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths