Total Complexity | 188 |
Total Lines | 749 |
Duplicated Lines | 0 % |
Changes | 0 |
Complex classes like Filesystem often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use Filesystem, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
23 | class Filesystem |
||
24 | { |
||
25 | private static ?string $lastError = null; |
||
26 | |||
27 | /** |
||
28 | * Copies a file. |
||
29 | * |
||
30 | * If the target file is older than the origin file, it's always overwritten. |
||
31 | * If the target file is newer, it is overwritten only when the |
||
32 | * $overwriteNewerFiles option is set to true. |
||
33 | * |
||
34 | * @throws FileNotFoundException When originFile doesn't exist |
||
35 | * @throws IOException When copy fails |
||
36 | */ |
||
37 | public function copy(string $originFile, string $targetFile, bool $overwriteNewerFiles = false): void |
||
38 | { |
||
39 | $originIsLocal = stream_is_local($originFile) || 0 === stripos($originFile, 'file://'); |
||
40 | if ($originIsLocal && !is_file($originFile)) { |
||
41 | throw new FileNotFoundException(\sprintf('Failed to copy "%s" because file does not exist.', $originFile), 0, null, $originFile); |
||
42 | } |
||
43 | |||
44 | $this->mkdir(\dirname($targetFile)); |
||
45 | |||
46 | $doCopy = true; |
||
47 | if (!$overwriteNewerFiles && !parse_url($originFile, \PHP_URL_HOST) && is_file($targetFile)) { |
||
48 | $doCopy = filemtime($originFile) > filemtime($targetFile); |
||
49 | } |
||
50 | |||
51 | if ($doCopy) { |
||
52 | // https://bugs.php.net/64634 |
||
53 | if (!$source = self::box('fopen', $originFile, 'r')) { |
||
54 | throw new IOException(\sprintf('Failed to copy "%s" to "%s" because source file could not be opened for reading: ', $originFile, $targetFile).self::$lastError, 0, null, $originFile); |
||
55 | } |
||
56 | |||
57 | // Stream context created to allow files overwrite when using FTP stream wrapper - disabled by default |
||
58 | if (!$target = self::box('fopen', $targetFile, 'w', false, stream_context_create(['ftp' => ['overwrite' => true]]))) { |
||
59 | throw new IOException(\sprintf('Failed to copy "%s" to "%s" because target file could not be opened for writing: ', $originFile, $targetFile).self::$lastError, 0, null, $originFile); |
||
60 | } |
||
61 | |||
62 | $bytesCopied = stream_copy_to_stream($source, $target); |
||
63 | fclose($source); |
||
64 | fclose($target); |
||
65 | unset($source, $target); |
||
66 | |||
67 | if (!is_file($targetFile)) { |
||
68 | throw new IOException(\sprintf('Failed to copy "%s" to "%s".', $originFile, $targetFile), 0, null, $originFile); |
||
69 | } |
||
70 | |||
71 | if ($originIsLocal) { |
||
72 | // Like `cp`, preserve executable permission bits |
||
73 | self::box('chmod', $targetFile, fileperms($targetFile) | (fileperms($originFile) & 0111)); |
||
74 | |||
75 | // Like `cp`, preserve the file modification time |
||
76 | self::box('touch', $targetFile, filemtime($originFile)); |
||
77 | |||
78 | if ($bytesCopied !== $bytesOrigin = filesize($originFile)) { |
||
79 | throw new IOException(\sprintf('Failed to copy the whole content of "%s" to "%s" (%g of %g bytes copied).', $originFile, $targetFile, $bytesCopied, $bytesOrigin), 0, null, $originFile); |
||
80 | } |
||
81 | } |
||
82 | } |
||
83 | } |
||
84 | |||
85 | /** |
||
86 | * Creates a directory recursively. |
||
87 | * |
||
88 | * @throws IOException On any directory creation failure |
||
89 | */ |
||
90 | public function mkdir(string|iterable $dirs, int $mode = 0777): void |
||
91 | { |
||
92 | foreach ($this->toIterable($dirs) as $dir) { |
||
93 | if (is_dir($dir)) { |
||
|
|||
94 | continue; |
||
95 | } |
||
96 | |||
97 | if (!self::box('mkdir', $dir, $mode, true) && !is_dir($dir)) { |
||
98 | throw new IOException(\sprintf('Failed to create "%s": ', $dir).self::$lastError, 0, null, $dir); |
||
99 | } |
||
100 | } |
||
101 | } |
||
102 | |||
103 | /** |
||
104 | * Checks the existence of files or directories. |
||
105 | */ |
||
106 | public function exists(string|iterable $files): bool |
||
107 | { |
||
108 | $maxPathLength = \PHP_MAXPATHLEN - 2; |
||
109 | |||
110 | foreach ($this->toIterable($files) as $file) { |
||
111 | if (\strlen($file) > $maxPathLength) { |
||
112 | throw new IOException(\sprintf('Could not check if file exist because path length exceeds %d characters.', $maxPathLength), 0, null, $file); |
||
113 | } |
||
114 | |||
115 | if (!file_exists($file)) { |
||
116 | return false; |
||
117 | } |
||
118 | } |
||
119 | |||
120 | return true; |
||
121 | } |
||
122 | |||
123 | /** |
||
124 | * Sets access and modification time of file. |
||
125 | * |
||
126 | * @param int|null $time The touch time as a Unix timestamp, if not supplied the current system time is used |
||
127 | * @param int|null $atime The access time as a Unix timestamp, if not supplied the current system time is used |
||
128 | * |
||
129 | * @throws IOException When touch fails |
||
130 | */ |
||
131 | public function touch(string|iterable $files, ?int $time = null, ?int $atime = null): void |
||
132 | { |
||
133 | foreach ($this->toIterable($files) as $file) { |
||
134 | if (!($time ? self::box('touch', $file, $time, $atime) : self::box('touch', $file))) { |
||
135 | throw new IOException(\sprintf('Failed to touch "%s": ', $file).self::$lastError, 0, null, $file); |
||
136 | } |
||
137 | } |
||
138 | } |
||
139 | |||
140 | /** |
||
141 | * Removes files or directories. |
||
142 | * |
||
143 | * @throws IOException When removal fails |
||
144 | */ |
||
145 | public function remove(string|iterable $files): void |
||
146 | { |
||
147 | if ($files instanceof \Traversable) { |
||
148 | $files = iterator_to_array($files, false); |
||
149 | } elseif (!\is_array($files)) { |
||
150 | $files = [$files]; |
||
151 | } |
||
152 | |||
153 | self::doRemove($files, false); |
||
154 | } |
||
155 | |||
156 | private static function doRemove(array $files, bool $isRecursive): void |
||
157 | { |
||
158 | $files = array_reverse($files); |
||
159 | foreach ($files as $file) { |
||
160 | if (is_link($file)) { |
||
161 | // See https://bugs.php.net/52176 |
||
162 | if (!(self::box('unlink', $file) || '\\' !== \DIRECTORY_SEPARATOR || self::box('rmdir', $file)) && file_exists($file)) { |
||
163 | throw new IOException(\sprintf('Failed to remove symlink "%s": ', $file).self::$lastError); |
||
164 | } |
||
165 | } elseif (is_dir($file)) { |
||
166 | if (!$isRecursive) { |
||
167 | $tmpName = \dirname(realpath($file)).'/.!'.strrev(strtr(base64_encode(random_bytes(2)), '/=', '-!')); |
||
168 | |||
169 | if (file_exists($tmpName)) { |
||
170 | try { |
||
171 | self::doRemove([$tmpName], true); |
||
172 | } catch (IOException) { |
||
173 | } |
||
174 | } |
||
175 | |||
176 | if (!file_exists($tmpName) && self::box('rename', $file, $tmpName)) { |
||
177 | $origFile = $file; |
||
178 | $file = $tmpName; |
||
179 | } else { |
||
180 | $origFile = null; |
||
181 | } |
||
182 | } |
||
183 | |||
184 | $filesystemIterator = new \FilesystemIterator($file, \FilesystemIterator::CURRENT_AS_PATHNAME | \FilesystemIterator::SKIP_DOTS); |
||
185 | self::doRemove(iterator_to_array($filesystemIterator, true), true); |
||
186 | |||
187 | if (!self::box('rmdir', $file) && file_exists($file) && !$isRecursive) { |
||
188 | $lastError = self::$lastError; |
||
189 | |||
190 | if (null !== $origFile && self::box('rename', $file, $origFile)) { |
||
191 | $file = $origFile; |
||
192 | } |
||
193 | |||
194 | throw new IOException(\sprintf('Failed to remove directory "%s": ', $file).$lastError); |
||
195 | } |
||
196 | } elseif (!self::box('unlink', $file) && ((self::$lastError && str_contains(self::$lastError, 'Permission denied')) || file_exists($file))) { |
||
197 | throw new IOException(\sprintf('Failed to remove file "%s": ', $file).self::$lastError); |
||
198 | } |
||
199 | } |
||
200 | } |
||
201 | |||
202 | /** |
||
203 | * Change mode for an array of files or directories. |
||
204 | * |
||
205 | * @param int $mode The new mode (octal) |
||
206 | * @param int $umask The mode mask (octal) |
||
207 | * @param bool $recursive Whether change the mod recursively or not |
||
208 | * |
||
209 | * @throws IOException When the change fails |
||
210 | */ |
||
211 | public function chmod(string|iterable $files, int $mode, int $umask = 0000, bool $recursive = false): void |
||
212 | { |
||
213 | foreach ($this->toIterable($files) as $file) { |
||
214 | if (!self::box('chmod', $file, $mode & ~$umask)) { |
||
215 | throw new IOException(\sprintf('Failed to chmod file "%s": ', $file).self::$lastError, 0, null, $file); |
||
216 | } |
||
217 | if ($recursive && is_dir($file) && !is_link($file)) { |
||
218 | $this->chmod(new \FilesystemIterator($file), $mode, $umask, true); |
||
219 | } |
||
220 | } |
||
221 | } |
||
222 | |||
223 | /** |
||
224 | * Change the owner of an array of files or directories. |
||
225 | * |
||
226 | * This method always throws on Windows, as the underlying PHP function is not supported. |
||
227 | * |
||
228 | * @see https://www.php.net/chown |
||
229 | * |
||
230 | * @param string|int $user A user name or number |
||
231 | * @param bool $recursive Whether change the owner recursively or not |
||
232 | * |
||
233 | * @throws IOException When the change fails |
||
234 | */ |
||
235 | public function chown(string|iterable $files, string|int $user, bool $recursive = false): void |
||
236 | { |
||
237 | foreach ($this->toIterable($files) as $file) { |
||
238 | if ($recursive && is_dir($file) && !is_link($file)) { |
||
239 | $this->chown(new \FilesystemIterator($file), $user, true); |
||
240 | } |
||
241 | if (is_link($file) && \function_exists('lchown')) { |
||
242 | if (!self::box('lchown', $file, $user)) { |
||
243 | throw new IOException(\sprintf('Failed to chown file "%s": ', $file).self::$lastError, 0, null, $file); |
||
244 | } |
||
245 | } else { |
||
246 | if (!self::box('chown', $file, $user)) { |
||
247 | throw new IOException(\sprintf('Failed to chown file "%s": ', $file).self::$lastError, 0, null, $file); |
||
248 | } |
||
249 | } |
||
250 | } |
||
251 | } |
||
252 | |||
253 | /** |
||
254 | * Change the group of an array of files or directories. |
||
255 | * |
||
256 | * This method always throws on Windows, as the underlying PHP function is not supported. |
||
257 | * |
||
258 | * @see https://www.php.net/chgrp |
||
259 | * |
||
260 | * @param string|int $group A group name or number |
||
261 | * @param bool $recursive Whether change the group recursively or not |
||
262 | * |
||
263 | * @throws IOException When the change fails |
||
264 | */ |
||
265 | public function chgrp(string|iterable $files, string|int $group, bool $recursive = false): void |
||
266 | { |
||
267 | foreach ($this->toIterable($files) as $file) { |
||
268 | if ($recursive && is_dir($file) && !is_link($file)) { |
||
269 | $this->chgrp(new \FilesystemIterator($file), $group, true); |
||
270 | } |
||
271 | if (is_link($file) && \function_exists('lchgrp')) { |
||
272 | if (!self::box('lchgrp', $file, $group)) { |
||
273 | throw new IOException(\sprintf('Failed to chgrp file "%s": ', $file).self::$lastError, 0, null, $file); |
||
274 | } |
||
275 | } else { |
||
276 | if (!self::box('chgrp', $file, $group)) { |
||
277 | throw new IOException(\sprintf('Failed to chgrp file "%s": ', $file).self::$lastError, 0, null, $file); |
||
278 | } |
||
279 | } |
||
280 | } |
||
281 | } |
||
282 | |||
283 | /** |
||
284 | * Renames a file or a directory. |
||
285 | * |
||
286 | * @throws IOException When target file or directory already exists |
||
287 | * @throws IOException When origin cannot be renamed |
||
288 | */ |
||
289 | public function rename(string $origin, string $target, bool $overwrite = false): void |
||
290 | { |
||
291 | // we check that target does not exist |
||
292 | if (!$overwrite && $this->isReadable($target)) { |
||
293 | throw new IOException(\sprintf('Cannot rename because the target "%s" already exists.', $target), 0, null, $target); |
||
294 | } |
||
295 | |||
296 | if (!self::box('rename', $origin, $target)) { |
||
297 | if (is_dir($origin)) { |
||
298 | // See https://bugs.php.net/54097 & https://php.net/rename#113943 |
||
299 | $this->mirror($origin, $target, null, ['override' => $overwrite, 'delete' => $overwrite]); |
||
300 | $this->remove($origin); |
||
301 | |||
302 | return; |
||
303 | } |
||
304 | throw new IOException(\sprintf('Cannot rename "%s" to "%s": ', $origin, $target).self::$lastError, 0, null, $target); |
||
305 | } |
||
306 | } |
||
307 | |||
308 | /** |
||
309 | * Tells whether a file exists and is readable. |
||
310 | * |
||
311 | * @throws IOException When windows path is longer than 258 characters |
||
312 | */ |
||
313 | private function isReadable(string $filename): bool |
||
314 | { |
||
315 | $maxPathLength = \PHP_MAXPATHLEN - 2; |
||
316 | |||
317 | if (\strlen($filename) > $maxPathLength) { |
||
318 | throw new IOException(\sprintf('Could not check if file is readable because path length exceeds %d characters.', $maxPathLength), 0, null, $filename); |
||
319 | } |
||
320 | |||
321 | return is_readable($filename); |
||
322 | } |
||
323 | |||
324 | /** |
||
325 | * Creates a symbolic link or copy a directory. |
||
326 | * |
||
327 | * @throws IOException When symlink fails |
||
328 | */ |
||
329 | public function symlink(string $originDir, string $targetDir, bool $copyOnWindows = false): void |
||
330 | { |
||
331 | self::assertFunctionExists('symlink'); |
||
332 | |||
333 | if ('\\' === \DIRECTORY_SEPARATOR) { |
||
334 | $originDir = strtr($originDir, '/', '\\'); |
||
335 | $targetDir = strtr($targetDir, '/', '\\'); |
||
336 | |||
337 | if ($copyOnWindows) { |
||
338 | $this->mirror($originDir, $targetDir); |
||
339 | |||
340 | return; |
||
341 | } |
||
342 | } |
||
343 | |||
344 | $this->mkdir(\dirname($targetDir)); |
||
345 | |||
346 | if (is_link($targetDir)) { |
||
347 | if (readlink($targetDir) === $originDir) { |
||
348 | return; |
||
349 | } |
||
350 | $this->remove($targetDir); |
||
351 | } |
||
352 | |||
353 | if (!self::box('symlink', $originDir, $targetDir)) { |
||
354 | $this->linkException($originDir, $targetDir, 'symbolic'); |
||
355 | } |
||
356 | } |
||
357 | |||
358 | /** |
||
359 | * Creates a hard link, or several hard links to a file. |
||
360 | * |
||
361 | * @param string|string[] $targetFiles The target file(s) |
||
362 | * |
||
363 | * @throws FileNotFoundException When original file is missing or not a file |
||
364 | * @throws IOException When link fails, including if link already exists |
||
365 | */ |
||
366 | public function hardlink(string $originFile, string|iterable $targetFiles): void |
||
367 | { |
||
368 | self::assertFunctionExists('link'); |
||
369 | |||
370 | if (!$this->exists($originFile)) { |
||
371 | throw new FileNotFoundException(null, 0, null, $originFile); |
||
372 | } |
||
373 | |||
374 | if (!is_file($originFile)) { |
||
375 | throw new FileNotFoundException(\sprintf('Origin file "%s" is not a file.', $originFile)); |
||
376 | } |
||
377 | |||
378 | foreach ($this->toIterable($targetFiles) as $targetFile) { |
||
379 | if (is_file($targetFile)) { |
||
380 | if (fileinode($originFile) === fileinode($targetFile)) { |
||
381 | continue; |
||
382 | } |
||
383 | $this->remove($targetFile); |
||
384 | } |
||
385 | |||
386 | if (!self::box('link', $originFile, $targetFile)) { |
||
387 | $this->linkException($originFile, $targetFile, 'hard'); |
||
388 | } |
||
389 | } |
||
390 | } |
||
391 | |||
392 | /** |
||
393 | * @param string $linkType Name of the link type, typically 'symbolic' or 'hard' |
||
394 | */ |
||
395 | private function linkException(string $origin, string $target, string $linkType): never |
||
396 | { |
||
397 | if (self::$lastError) { |
||
398 | if ('\\' === \DIRECTORY_SEPARATOR && str_contains(self::$lastError, 'error code(1314)')) { |
||
399 | throw new IOException(\sprintf('Unable to create "%s" link due to error code 1314: \'A required privilege is not held by the client\'. Do you have the required Administrator-rights?', $linkType), 0, null, $target); |
||
400 | } |
||
401 | } |
||
402 | throw new IOException(\sprintf('Failed to create "%s" link from "%s" to "%s": ', $linkType, $origin, $target).self::$lastError, 0, null, $target); |
||
403 | } |
||
404 | |||
405 | /** |
||
406 | * Resolves links in paths. |
||
407 | * |
||
408 | * With $canonicalize = false (default) |
||
409 | * - if $path does not exist or is not a link, returns null |
||
410 | * - if $path is a link, returns the next direct target of the link without considering the existence of the target |
||
411 | * |
||
412 | * With $canonicalize = true |
||
413 | * - if $path does not exist, returns null |
||
414 | * - if $path exists, returns its absolute fully resolved final version |
||
415 | */ |
||
416 | public function readlink(string $path, bool $canonicalize = false): ?string |
||
417 | { |
||
418 | if (!$canonicalize && !is_link($path)) { |
||
419 | return null; |
||
420 | } |
||
421 | |||
422 | if ($canonicalize) { |
||
423 | if (!$this->exists($path)) { |
||
424 | return null; |
||
425 | } |
||
426 | |||
427 | return realpath($path); |
||
428 | } |
||
429 | |||
430 | return readlink($path); |
||
431 | } |
||
432 | |||
433 | /** |
||
434 | * Given an existing path, convert it to a path relative to a given starting path. |
||
435 | */ |
||
436 | public function makePathRelative(string $endPath, string $startPath): string |
||
437 | { |
||
438 | if (!$this->isAbsolutePath($startPath)) { |
||
439 | throw new InvalidArgumentException(\sprintf('The start path "%s" is not absolute.', $startPath)); |
||
440 | } |
||
441 | |||
442 | if (!$this->isAbsolutePath($endPath)) { |
||
443 | throw new InvalidArgumentException(\sprintf('The end path "%s" is not absolute.', $endPath)); |
||
444 | } |
||
445 | |||
446 | // Normalize separators on Windows |
||
447 | if ('\\' === \DIRECTORY_SEPARATOR) { |
||
448 | $endPath = str_replace('\\', '/', $endPath); |
||
449 | $startPath = str_replace('\\', '/', $startPath); |
||
450 | } |
||
451 | |||
452 | $splitDriveLetter = fn ($path) => (\strlen($path) > 2 && ':' === $path[1] && '/' === $path[2] && ctype_alpha($path[0])) |
||
453 | ? [substr($path, 2), strtoupper($path[0])] |
||
454 | : [$path, null]; |
||
455 | |||
456 | $splitPath = function ($path) { |
||
457 | $result = []; |
||
458 | |||
459 | foreach (explode('/', trim($path, '/')) as $segment) { |
||
460 | if ('..' === $segment) { |
||
461 | array_pop($result); |
||
462 | } elseif ('.' !== $segment && '' !== $segment) { |
||
463 | $result[] = $segment; |
||
464 | } |
||
465 | } |
||
466 | |||
467 | return $result; |
||
468 | }; |
||
469 | |||
470 | [$endPath, $endDriveLetter] = $splitDriveLetter($endPath); |
||
471 | [$startPath, $startDriveLetter] = $splitDriveLetter($startPath); |
||
472 | |||
473 | $startPathArr = $splitPath($startPath); |
||
474 | $endPathArr = $splitPath($endPath); |
||
475 | |||
476 | if ($endDriveLetter && $startDriveLetter && $endDriveLetter != $startDriveLetter) { |
||
477 | // End path is on another drive, so no relative path exists |
||
478 | return $endDriveLetter.':/'.($endPathArr ? implode('/', $endPathArr).'/' : ''); |
||
479 | } |
||
480 | |||
481 | // Find for which directory the common path stops |
||
482 | $index = 0; |
||
483 | while (isset($startPathArr[$index]) && isset($endPathArr[$index]) && $startPathArr[$index] === $endPathArr[$index]) { |
||
484 | ++$index; |
||
485 | } |
||
486 | |||
487 | // Determine how deep the start path is relative to the common path (ie, "web/bundles" = 2 levels) |
||
488 | if (1 === \count($startPathArr) && '' === $startPathArr[0]) { |
||
489 | $depth = 0; |
||
490 | } else { |
||
491 | $depth = \count($startPathArr) - $index; |
||
492 | } |
||
493 | |||
494 | // Repeated "../" for each level need to reach the common path |
||
495 | $traverser = str_repeat('../', $depth); |
||
496 | |||
497 | $endPathRemainder = implode('/', \array_slice($endPathArr, $index)); |
||
498 | |||
499 | // Construct $endPath from traversing to the common path, then to the remaining $endPath |
||
500 | $relativePath = $traverser.('' !== $endPathRemainder ? $endPathRemainder.'/' : ''); |
||
501 | |||
502 | return '' === $relativePath ? './' : $relativePath; |
||
503 | } |
||
504 | |||
505 | /** |
||
506 | * Mirrors a directory to another. |
||
507 | * |
||
508 | * Copies files and directories from the origin directory into the target directory. By default: |
||
509 | * |
||
510 | * - existing files in the target directory will be overwritten, except if they are newer (see the `override` option) |
||
511 | * - files in the target directory that do not exist in the source directory will not be deleted (see the `delete` option) |
||
512 | * |
||
513 | * @param \Traversable|null $iterator Iterator that filters which files and directories to copy, if null a recursive iterator is created |
||
514 | * @param array $options An array of boolean options |
||
515 | * Valid options are: |
||
516 | * - $options['override'] If true, target files newer than origin files are overwritten (see copy(), defaults to false) |
||
517 | * - $options['copy_on_windows'] Whether to copy files instead of links on Windows (see symlink(), defaults to false) |
||
518 | * - $options['delete'] Whether to delete files that are not in the source directory (defaults to false) |
||
519 | * |
||
520 | * @throws IOException When file type is unknown |
||
521 | */ |
||
522 | public function mirror(string $originDir, string $targetDir, ?\Traversable $iterator = null, array $options = []): void |
||
523 | { |
||
524 | $targetDir = rtrim($targetDir, '/\\'); |
||
525 | $originDir = rtrim($originDir, '/\\'); |
||
526 | $originDirLen = \strlen($originDir); |
||
527 | |||
528 | if (!$this->exists($originDir)) { |
||
529 | throw new IOException(\sprintf('The origin directory specified "%s" was not found.', $originDir), 0, null, $originDir); |
||
530 | } |
||
531 | |||
532 | // Iterate in destination folder to remove obsolete entries |
||
533 | if ($this->exists($targetDir) && isset($options['delete']) && $options['delete']) { |
||
534 | $deleteIterator = $iterator; |
||
535 | if (null === $deleteIterator) { |
||
536 | $flags = \FilesystemIterator::SKIP_DOTS; |
||
537 | $deleteIterator = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($targetDir, $flags), \RecursiveIteratorIterator::CHILD_FIRST); |
||
538 | } |
||
539 | $targetDirLen = \strlen($targetDir); |
||
540 | foreach ($deleteIterator as $file) { |
||
541 | $origin = $originDir.substr($file->getPathname(), $targetDirLen); |
||
542 | if (!$this->exists($origin)) { |
||
543 | $this->remove($file); |
||
544 | } |
||
545 | } |
||
546 | } |
||
547 | |||
548 | $copyOnWindows = $options['copy_on_windows'] ?? false; |
||
549 | |||
550 | if (null === $iterator) { |
||
551 | $flags = $copyOnWindows ? \FilesystemIterator::SKIP_DOTS | \FilesystemIterator::FOLLOW_SYMLINKS : \FilesystemIterator::SKIP_DOTS; |
||
552 | $iterator = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($originDir, $flags), \RecursiveIteratorIterator::SELF_FIRST); |
||
553 | } |
||
554 | |||
555 | $this->mkdir($targetDir); |
||
556 | $filesCreatedWhileMirroring = []; |
||
557 | |||
558 | foreach ($iterator as $file) { |
||
559 | if ($file->getPathname() === $targetDir || $file->getRealPath() === $targetDir || isset($filesCreatedWhileMirroring[$file->getRealPath()])) { |
||
560 | continue; |
||
561 | } |
||
562 | |||
563 | $target = $targetDir.substr($file->getPathname(), $originDirLen); |
||
564 | $filesCreatedWhileMirroring[$target] = true; |
||
565 | |||
566 | if (!$copyOnWindows && is_link($file)) { |
||
567 | $this->symlink($file->getLinkTarget(), $target); |
||
568 | } elseif (is_dir($file)) { |
||
569 | $this->mkdir($target); |
||
570 | } elseif (is_file($file)) { |
||
571 | $this->copy($file, $target, $options['override'] ?? false); |
||
572 | } else { |
||
573 | throw new IOException(\sprintf('Unable to guess "%s" file type.', $file), 0, null, $file); |
||
574 | } |
||
575 | } |
||
576 | } |
||
577 | |||
578 | /** |
||
579 | * Returns whether the file path is an absolute path. |
||
580 | */ |
||
581 | public function isAbsolutePath(string $file): bool |
||
589 | ); |
||
590 | } |
||
591 | |||
592 | /** |
||
593 | * Creates a temporary file with support for custom stream wrappers. |
||
594 | * |
||
595 | * @param string $prefix The prefix of the generated temporary filename |
||
596 | * Note: Windows uses only the first three characters of prefix |
||
597 | * @param string $suffix The suffix of the generated temporary filename |
||
598 | * |
||
599 | * @return string The new temporary filename (with path), or throw an exception on failure |
||
600 | */ |
||
601 | public function tempnam(string $dir, string $prefix, string $suffix = ''): string |
||
602 | { |
||
603 | [$scheme, $hierarchy] = $this->getSchemeAndHierarchy($dir); |
||
604 | |||
605 | // If no scheme or scheme is "file" or "gs" (Google Cloud) create temp file in local filesystem |
||
606 | if ((null === $scheme || 'file' === $scheme || 'gs' === $scheme) && '' === $suffix) { |
||
607 | // If tempnam failed or no scheme return the filename otherwise prepend the scheme |
||
608 | if ($tmpFile = self::box('tempnam', $hierarchy, $prefix)) { |
||
609 | if (null !== $scheme && 'gs' !== $scheme) { |
||
610 | return $scheme.'://'.$tmpFile; |
||
611 | } |
||
612 | |||
613 | return $tmpFile; |
||
614 | } |
||
615 | |||
616 | throw new IOException('A temporary file could not be created: '.self::$lastError); |
||
617 | } |
||
618 | |||
619 | // Loop until we create a valid temp file or have reached 10 attempts |
||
620 | for ($i = 0; $i < 10; ++$i) { |
||
621 | // Create a unique filename |
||
622 | $tmpFile = $dir.'/'.$prefix.bin2hex(random_bytes(4)).$suffix; |
||
623 | |||
624 | // Use fopen instead of file_exists as some streams do not support stat |
||
625 | // Use mode 'x+' to atomically check existence and create to avoid a TOCTOU vulnerability |
||
626 | if (!$handle = self::box('fopen', $tmpFile, 'x+')) { |
||
627 | continue; |
||
628 | } |
||
629 | |||
630 | // Close the file if it was successfully opened |
||
631 | self::box('fclose', $handle); |
||
632 | |||
633 | return $tmpFile; |
||
634 | } |
||
635 | |||
636 | throw new IOException('A temporary file could not be created: '.self::$lastError); |
||
637 | } |
||
638 | |||
639 | /** |
||
640 | * Atomically dumps content into a file. |
||
641 | * |
||
642 | * @param string|resource $content The data to write into the file |
||
643 | * |
||
644 | * @throws IOException if the file cannot be written to |
||
645 | */ |
||
646 | public function dumpFile(string $filename, $content): void |
||
647 | { |
||
648 | if (\is_array($content)) { |
||
649 | throw new \TypeError(\sprintf('Argument 2 passed to "%s()" must be string or resource, array given.', __METHOD__)); |
||
650 | } |
||
651 | |||
652 | $dir = \dirname($filename); |
||
653 | |||
654 | if (is_link($filename) && $linkTarget = $this->readlink($filename)) { |
||
655 | $this->dumpFile(Path::makeAbsolute($linkTarget, $dir), $content); |
||
656 | |||
657 | return; |
||
658 | } |
||
659 | |||
660 | if (!is_dir($dir)) { |
||
661 | $this->mkdir($dir); |
||
662 | } |
||
663 | |||
664 | // Will create a temp file with 0600 access rights |
||
665 | // when the filesystem supports chmod. |
||
666 | $tmpFile = $this->tempnam($dir, basename($filename)); |
||
667 | |||
668 | try { |
||
669 | if (false === self::box('file_put_contents', $tmpFile, $content)) { |
||
670 | throw new IOException(\sprintf('Failed to write file "%s": ', $filename).self::$lastError, 0, null, $filename); |
||
671 | } |
||
672 | |||
673 | self::box('chmod', $tmpFile, self::box('fileperms', $filename) ?: 0666 & ~umask()); |
||
674 | |||
675 | $this->rename($tmpFile, $filename, true); |
||
676 | } finally { |
||
677 | if (file_exists($tmpFile)) { |
||
678 | if ('\\' === \DIRECTORY_SEPARATOR && !is_writable($tmpFile)) { |
||
679 | self::box('chmod', $tmpFile, self::box('fileperms', $tmpFile) | 0200); |
||
680 | } |
||
681 | |||
682 | self::box('unlink', $tmpFile); |
||
683 | } |
||
684 | } |
||
685 | } |
||
686 | |||
687 | /** |
||
688 | * Appends content to an existing file. |
||
689 | * |
||
690 | * @param string|resource $content The content to append |
||
691 | * @param bool $lock Whether the file should be locked when writing to it |
||
692 | * |
||
693 | * @throws IOException If the file is not writable |
||
694 | */ |
||
695 | public function appendToFile(string $filename, $content, bool $lock = false): void |
||
696 | { |
||
697 | if (\is_array($content)) { |
||
698 | throw new \TypeError(\sprintf('Argument 2 passed to "%s()" must be string or resource, array given.', __METHOD__)); |
||
699 | } |
||
700 | |||
701 | $dir = \dirname($filename); |
||
702 | |||
703 | if (!is_dir($dir)) { |
||
704 | $this->mkdir($dir); |
||
705 | } |
||
706 | |||
707 | if (false === self::box('file_put_contents', $filename, $content, \FILE_APPEND | ($lock ? \LOCK_EX : 0))) { |
||
708 | throw new IOException(\sprintf('Failed to write file "%s": ', $filename).self::$lastError, 0, null, $filename); |
||
709 | } |
||
710 | } |
||
711 | |||
712 | /** |
||
713 | * Returns the content of a file as a string. |
||
714 | * |
||
715 | * @throws IOException If the file cannot be read |
||
716 | */ |
||
717 | public function readFile(string $filename): string |
||
718 | { |
||
719 | if (is_dir($filename)) { |
||
720 | throw new IOException(\sprintf('Failed to read file "%s": File is a directory.', $filename)); |
||
721 | } |
||
722 | |||
723 | $content = self::box('file_get_contents', $filename); |
||
724 | if (false === $content) { |
||
725 | throw new IOException(\sprintf('Failed to read file "%s": ', $filename).self::$lastError, 0, null, $filename); |
||
726 | } |
||
727 | |||
728 | return $content; |
||
729 | } |
||
730 | |||
731 | private function toIterable(string|iterable $files): iterable |
||
732 | { |
||
733 | return is_iterable($files) ? $files : [$files]; |
||
734 | } |
||
735 | |||
736 | /** |
||
737 | * Gets a 2-tuple of scheme (may be null) and hierarchical part of a filename (e.g. file:///tmp -> [file, tmp]). |
||
738 | */ |
||
739 | private function getSchemeAndHierarchy(string $filename): array |
||
740 | { |
||
741 | $components = explode('://', $filename, 2); |
||
742 | |||
743 | return 2 === \count($components) ? [$components[0], $components[1]] : [null, $components[0]]; |
||
744 | } |
||
745 | |||
746 | private static function assertFunctionExists(string $func): void |
||
747 | { |
||
748 | if (!\function_exists($func)) { |
||
749 | throw new IOException(\sprintf('Unable to perform filesystem operation because the "%s()" function has been disabled.', $func)); |
||
750 | } |
||
751 | } |
||
752 | |||
753 | private static function box(string $func, mixed ...$args): mixed |
||
763 | } |
||
764 | } |
||
765 | |||
766 | /** |
||
767 | * @internal |
||
768 | */ |
||
769 | public static function handleError(int $type, string $msg): void |
||
770 | { |
||
771 | self::$lastError = $msg; |
||
772 | } |
||
773 | } |
||
774 |