Completed
Pull Request — 2.1 (#12704)
by Robert
08:59
created

BaseFileHelper::matchBasename()   B

Complexity

Conditions 6
Paths 8

Size

Total Lines 21
Code Lines 12

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 13
CRAP Score 6.0131

Importance

Changes 0
Metric Value
dl 0
loc 21
rs 8.7624
c 0
b 0
f 0
ccs 13
cts 14
cp 0.9286
cc 6
eloc 12
nc 8
nop 4
crap 6.0131
1
<?php
2
/**
3
 * @link http://www.yiiframework.com/
4
 * @copyright Copyright (c) 2008 Yii Software LLC
5
 * @license http://www.yiiframework.com/license/
6
 */
7
8
namespace yii\helpers;
9
10
use Yii;
11
use yii\base\ErrorException;
12
use yii\base\InvalidConfigException;
13
use yii\base\InvalidParamException;
14
15
/**
16
 * BaseFileHelper provides concrete implementation for [[FileHelper]].
17
 *
18
 * Do not use BaseFileHelper. Use [[FileHelper]] instead.
19
 *
20
 * @author Qiang Xue <[email protected]>
21
 * @author Alex Makarov <[email protected]>
22
 * @since 2.0
23
 */
24
class BaseFileHelper
25
{
26
    const PATTERN_NODIR = 1;
27
    const PATTERN_ENDSWITH = 4;
28
    const PATTERN_MUSTBEDIR = 8;
29
    const PATTERN_NEGATIVE = 16;
30
    const PATTERN_CASE_INSENSITIVE = 32;
31
32
    /**
33
     * @var string the path (or alias) of a PHP file containing MIME type information.
34
     */
35
    public static $mimeMagicFile = '@yii/helpers/mimeTypes.php';
36
37
38
    /**
39
     * Normalizes a file/directory path.
40
     * The normalization does the following work:
41
     *
42
     * - Convert all directory separators into `DIRECTORY_SEPARATOR` (e.g. "\a/b\c" becomes "/a/b/c")
43
     * - Remove trailing directory separators (e.g. "/a/b/c/" becomes "/a/b/c")
44
     * - Turn multiple consecutive slashes into a single one (e.g. "/a///b/c" becomes "/a/b/c")
45
     * - Remove ".." and "." based on their meanings (e.g. "/a/./b/../c" becomes "/a/c")
46
     *
47
     * @param string $path the file/directory path to be normalized
48
     * @param string $ds the directory separator to be used in the normalized result. Defaults to `DIRECTORY_SEPARATOR`.
49
     * @return string the normalized file/directory path
50
     */
51 14
    public static function normalizePath($path, $ds = DIRECTORY_SEPARATOR)
52
    {
53 14
        $path = rtrim(strtr($path, '/\\', $ds . $ds), $ds);
54 14
        if (strpos($ds . $path, "{$ds}.") === false && strpos($path, "{$ds}{$ds}") === false) {
55 13
            return $path;
56
        }
57
        // the path may contain ".", ".." or double slashes, need to clean them up
58 1
        $parts = [];
59 1
        foreach (explode($ds, $path) as $part) {
60 1
            if ($part === '..' && !empty($parts) && end($parts) !== '..') {
61 1
                array_pop($parts);
62 1
            } elseif ($part === '.' || $part === '' && !empty($parts)) {
63 1
                continue;
64
            } else {
65 1
                $parts[] = $part;
66
            }
67 1
        }
68 1
        $path = implode($ds, $parts);
69 1
        return $path === '' ? '.' : $path;
70
    }
71
72
    /**
73
     * Returns the localized version of a specified file.
74
     *
75
     * The searching is based on the specified language code. In particular,
76
     * a file with the same name will be looked for under the subdirectory
77
     * whose name is the same as the language code. For example, given the file "path/to/view.php"
78
     * and language code "zh-CN", the localized file will be looked for as
79
     * "path/to/zh-CN/view.php". If the file is not found, it will try a fallback with just a language code that is
80
     * "zh" i.e. "path/to/zh/view.php". If it is not found as well the original file will be returned.
81
     *
82
     * If the target and the source language codes are the same,
83
     * the original file will be returned.
84
     *
85
     * @param string $file the original file
86
     * @param string $language the target language that the file should be localized to.
87
     * If not set, the value of [[\yii\base\Application::language]] will be used.
88
     * @param string $sourceLanguage the language that the original file is in.
89
     * If not set, the value of [[\yii\base\Application::sourceLanguage]] will be used.
90
     * @return string the matching localized file, or the original file if the localized version is not found.
91
     * If the target and the source language codes are the same, the original file will be returned.
92
     */
93 44
    public static function localize($file, $language = null, $sourceLanguage = null)
94
    {
95 44
        if ($language === null) {
96 43
            $language = Yii::$app->language;
97 43
        }
98 44
        if ($sourceLanguage === null) {
99 43
            $sourceLanguage = Yii::$app->sourceLanguage;
100 43
        }
101 44
        if ($language === $sourceLanguage) {
102 44
            return $file;
103
        }
104 1
        $desiredFile = dirname($file) . DIRECTORY_SEPARATOR . $language . DIRECTORY_SEPARATOR . basename($file);
105 1
        if (is_file($desiredFile)) {
106 1
            return $desiredFile;
107
        } else {
108
            $language = substr($language, 0, 2);
109
            if ($language === $sourceLanguage) {
110
                return $file;
111
            }
112
            $desiredFile = dirname($file) . DIRECTORY_SEPARATOR . $language . DIRECTORY_SEPARATOR . basename($file);
113
114
            return is_file($desiredFile) ? $desiredFile : $file;
115
        }
116
    }
117
118
    /**
119
     * Determines the MIME type of the specified file.
120
     * This method will first try to determine the MIME type based on
121
     * [finfo_open](http://php.net/manual/en/function.finfo-open.php). If the `fileinfo` extension is not installed,
122
     * it will fall back to [[getMimeTypeByExtension()]] when `$checkExtension` is true.
123
     * @param string $file the file name.
124
     * @param string $magicFile name of the optional magic database file (or alias), usually something like `/path/to/magic.mime`.
125
     * This will be passed as the second parameter to [finfo_open()](http://php.net/manual/en/function.finfo-open.php)
126
     * when the `fileinfo` extension is installed. If the MIME type is being determined based via [[getMimeTypeByExtension()]]
127
     * and this is null, it will use the file specified by [[mimeMagicFile]].
128
     * @param boolean $checkExtension whether to use the file extension to determine the MIME type in case
129
     * `finfo_open()` cannot determine it.
130
     * @return string the MIME type (e.g. `text/plain`). Null is returned if the MIME type cannot be determined.
131
     * @throws InvalidConfigException when the `fileinfo` PHP extension is not installed and `$checkExtension` is `false`.
132
     */
133 12
    public static function getMimeType($file, $magicFile = null, $checkExtension = true)
134
    {
135 12
        if ($magicFile !== null) {
136
            $magicFile = Yii::getAlias($magicFile);
137
        }
138 12
        if (!extension_loaded('fileinfo')) {
139
            if ($checkExtension) {
140
                return static::getMimeTypeByExtension($file, $magicFile);
0 ignored issues
show
Bug introduced by
It seems like $magicFile defined by \Yii::getAlias($magicFile) on line 136 can also be of type boolean; however, yii\helpers\BaseFileHelp...etMimeTypeByExtension() does only seem to accept string|null, maybe add an additional type check?

If a method or function can return multiple different values and unless you are sure that you only can receive a single value in this context, we recommend to add an additional type check:

/**
 * @return array|string
 */
function returnsDifferentValues($x) {
    if ($x) {
        return 'foo';
    }

    return array();
}

$x = returnsDifferentValues($y);
if (is_array($x)) {
    // $x is an array.
}

If this a common case that PHP Analyzer should handle natively, please let us know by opening an issue.

Loading history...
141
            } else {
142
                throw new InvalidConfigException('The fileinfo PHP extension is not installed.');
143
            }
144
        }
145 12
        $info = finfo_open(FILEINFO_MIME_TYPE, $magicFile);
146
147 12
        if ($info) {
148 12
            $result = finfo_file($info, $file);
149 12
            finfo_close($info);
150
151 12
            if ($result !== false) {
152 12
                return $result;
153
            }
154
        }
155
156
        return $checkExtension ? static::getMimeTypeByExtension($file, $magicFile) : null;
0 ignored issues
show
Bug introduced by
It seems like $magicFile defined by \Yii::getAlias($magicFile) on line 136 can also be of type boolean; however, yii\helpers\BaseFileHelp...etMimeTypeByExtension() does only seem to accept string|null, maybe add an additional type check?

If a method or function can return multiple different values and unless you are sure that you only can receive a single value in this context, we recommend to add an additional type check:

/**
 * @return array|string
 */
function returnsDifferentValues($x) {
    if ($x) {
        return 'foo';
    }

    return array();
}

$x = returnsDifferentValues($y);
if (is_array($x)) {
    // $x is an array.
}

If this a common case that PHP Analyzer should handle natively, please let us know by opening an issue.

Loading history...
157
    }
158
159
    /**
160
     * Determines the MIME type based on the extension name of the specified file.
161
     * This method will use a local map between extension names and MIME types.
162
     * @param string $file the file name.
163
     * @param string $magicFile the path (or alias) of the file that contains all available MIME type information.
164
     * If this is not set, the file specified by [[mimeMagicFile]] will be used.
165
     * @return string the MIME type. Null is returned if the MIME type cannot be determined.
166
     */
167 8
    public static function getMimeTypeByExtension($file, $magicFile = null)
168
    {
169 8
        $mimeTypes = static::loadMimeTypes($magicFile);
170
171 8
        if (($ext = pathinfo($file, PATHINFO_EXTENSION)) !== '') {
172 8
            $ext = strtolower($ext);
173 8
            if (isset($mimeTypes[$ext])) {
174 8
                return $mimeTypes[$ext];
175
            }
176 1
        }
177
178 1
        return null;
179
    }
180
181
    /**
182
     * Determines the extensions by given MIME type.
183
     * This method will use a local map between extension names and MIME types.
184
     * @param string $mimeType file MIME type.
185
     * @param string $magicFile the path (or alias) of the file that contains all available MIME type information.
186
     * If this is not set, the file specified by [[mimeMagicFile]] will be used.
187
     * @return array the extensions corresponding to the specified MIME type
188
     */
189
    public static function getExtensionsByMimeType($mimeType, $magicFile = null)
190
    {
191
        $mimeTypes = static::loadMimeTypes($magicFile);
192
        return array_keys($mimeTypes, mb_strtolower($mimeType, 'UTF-8'), true);
193
    }
194
195
    private static $_mimeTypes = [];
196
197
    /**
198
     * Loads MIME types from the specified file.
199
     * @param string $magicFile the path (or alias) of the file that contains all available MIME type information.
200
     * If this is not set, the file specified by [[mimeMagicFile]] will be used.
201
     * @return array the mapping from file extensions to MIME types
202
     */
203 8
    protected static function loadMimeTypes($magicFile)
204
    {
205 8
        if ($magicFile === null) {
206 8
            $magicFile = static::$mimeMagicFile;
207 8
        }
208 8
        $magicFile = Yii::getAlias($magicFile);
209 8
        if (!isset(self::$_mimeTypes[$magicFile])) {
210 1
            self::$_mimeTypes[$magicFile] = require($magicFile);
211 1
        }
212 8
        return self::$_mimeTypes[$magicFile];
213
    }
214
215
    /**
216
     * Copies a whole directory as another one.
217
     * The files and sub-directories will also be copied over.
218
     * @param string $src the source directory
219
     * @param string $dst the destination directory
220
     * @param array $options options for directory copy. Valid options are:
221
     *
222
     * - dirMode: integer, the permission to be set for newly copied directories. Defaults to 0775.
223
     * - fileMode:  integer, the permission to be set for newly copied files. Defaults to the current environment setting.
224
     * - filter: callback, a PHP callback that is called for each directory or file.
225
     *   The signature of the callback should be: `function ($path)`, where `$path` refers the full path to be filtered.
226
     *   The callback can return one of the following values:
227
     *
228
     *   * true: the directory or file will be copied (the "only" and "except" options will be ignored)
229
     *   * false: the directory or file will NOT be copied (the "only" and "except" options will be ignored)
230
     *   * null: the "only" and "except" options will determine whether the directory or file should be copied
231
     *
232
     * - only: array, list of patterns that the file paths should match if they want to be copied.
233
     *   A path matches a pattern if it contains the pattern string at its end.
234
     *   For example, '.php' matches all file paths ending with '.php'.
235
     *   Note, the '/' characters in a pattern matches both '/' and '\' in the paths.
236
     *   If a file path matches a pattern in both "only" and "except", it will NOT be copied.
237
     * - except: array, list of patterns that the files or directories should match if they want to be excluded from being copied.
238
     *   A path matches a pattern if it contains the pattern string at its end.
239
     *   Patterns ending with '/' apply to directory paths only, and patterns not ending with '/'
240
     *   apply to file paths only. For example, '/a/b' matches all file paths ending with '/a/b';
241
     *   and '.svn/' matches directory paths ending with '.svn'. Note, the '/' characters in a pattern matches
242
     *   both '/' and '\' in the paths.
243
     * - caseSensitive: boolean, whether patterns specified at "only" or "except" should be case sensitive. Defaults to true.
244
     * - recursive: boolean, whether the files under the subdirectories should also be copied. Defaults to true.
245
     * - beforeCopy: callback, a PHP callback that is called before copying each sub-directory or file.
246
     *   If the callback returns false, the copy operation for the sub-directory or file will be cancelled.
247
     *   The signature of the callback should be: `function ($from, $to)`, where `$from` is the sub-directory or
248
     *   file to be copied from, while `$to` is the copy target.
249
     * - afterCopy: callback, a PHP callback that is called after each sub-directory or file is successfully copied.
250
     *   The signature of the callback should be: `function ($from, $to)`, where `$from` is the sub-directory or
251
     *   file copied from, while `$to` is the copy target.
252
     * @throws \yii\base\InvalidParamException if unable to open directory
253
     */
254 13
    public static function copyDirectory($src, $dst, $options = [])
255
    {
256 13
        $src = static::normalizePath($src);
257 13
        $dst = static::normalizePath($dst);
258
259 13
        if ($src === $dst || strpos($dst, $src . DIRECTORY_SEPARATOR) === 0) {
260 2
            throw new InvalidParamException('Trying to copy a directory to itself or a subdirectory.');
261
        }
262 11
        if (!is_dir($dst)) {
263 9
            static::createDirectory($dst, isset($options['dirMode']) ? $options['dirMode'] : 0775, true);
264 9
        }
265
266 11
        $handle = opendir($src);
267 11
        if ($handle === false) {
268
            throw new InvalidParamException("Unable to open directory: $src");
269
        }
270 11
        if (!isset($options['basePath'])) {
271
            // this should be done only once
272 11
            $options['basePath'] = realpath($src);
273 11
            $options = self::normalizeOptions($options);
274 11
        }
275 11
        while (($file = readdir($handle)) !== false) {
276 11
            if ($file === '.' || $file === '..') {
277 11
                continue;
278
            }
279 9
            $from = $src . DIRECTORY_SEPARATOR . $file;
280 9
            $to = $dst . DIRECTORY_SEPARATOR . $file;
281 9
            if (static::filterPath($from, $options)) {
282 9
                if (isset($options['beforeCopy']) && !call_user_func($options['beforeCopy'], $from, $to)) {
283 2
                    continue;
284
                }
285 7
                if (is_file($from)) {
286 7
                    copy($from, $to);
287 7
                    if (isset($options['fileMode'])) {
288 1
                        @chmod($to, $options['fileMode']);
0 ignored issues
show
Security Best Practice introduced by
It seems like you do not handle an error condition here. This can introduce security issues, and is generally not recommended.

If you suppress an error, we recommend checking for the error condition explicitly:

// For example instead of
@mkdir($dir);

// Better use
if (@mkdir($dir) === false) {
    throw new \RuntimeException('The directory '.$dir.' could not be created.');
}
Loading history...
289 1
                    }
290 7
                } else {
291
                    // recursive copy, defaults to true
292 5
                    if (!isset($options['recursive']) || $options['recursive']) {
293 4
                        static::copyDirectory($from, $to, $options);
294 4
                    }
295
                }
296 7
                if (isset($options['afterCopy'])) {
297
                    call_user_func($options['afterCopy'], $from, $to);
298
                }
299 7
            }
300 7
        }
301 11
        closedir($handle);
302 11
    }
303
304
    /**
305
     * Removes a directory (and all its content) recursively.
306
     *
307
     * @param string $dir the directory to be deleted recursively.
308
     * @param array $options options for directory remove. Valid options are:
309
     *
310
     * - traverseSymlinks: boolean, whether symlinks to the directories should be traversed too.
311
     *   Defaults to `false`, meaning the content of the symlinked directory would not be deleted.
312
     *   Only symlink would be removed in that default case.
313
     *
314
     * @throws ErrorException in case of failure
315
     */
316 86
    public static function removeDirectory($dir, $options = [])
317
    {
318 86
        if (!is_dir($dir)) {
319 8
            return;
320
        }
321 85
        if (isset($options['traverseSymlinks']) && $options['traverseSymlinks'] || !is_link($dir)) {
322 85
            if (!($handle = opendir($dir))) {
323
                return;
324
            }
325 85
            while (($file = readdir($handle)) !== false) {
326 85
                if ($file === '.' || $file === '..') {
327 85
                    continue;
328
                }
329 75
                $path = $dir . DIRECTORY_SEPARATOR . $file;
330 75
                if (is_dir($path)) {
331 45
                    static::removeDirectory($path, $options);
332 45
                } else {
333
                    try {
334 57
                        unlink($path);
335 57
                    } catch (ErrorException $e) {
336
                        if (DIRECTORY_SEPARATOR === '\\') {
337
                            // last resort measure for Windows
338
                            $lines = [];
339
                            exec("DEL /F/Q \"$path\"", $lines, $deleteError);
340
                        } else {
341
                            throw $e;
342
                        }
343
                    }
344
                }
345 75
            }
346 85
            closedir($handle);
347 85
        }
348 85
        if (is_link($dir)) {
349 2
            unlink($dir);
350 2
        } else {
351 85
            rmdir($dir);
352
        }
353 85
    }
354
355
    /**
356
     * Returns the files found under the specified directory and subdirectories.
357
     * @param string $dir the directory under which the files will be looked for.
358
     * @param array $options options for file searching. Valid options are:
359
     *
360
     * - `filter`: callback, a PHP callback that is called for each directory or file.
361
     *   The signature of the callback should be: `function ($path)`, where `$path` refers the full path to be filtered.
362
     *   The callback can return one of the following values:
363
     *
364
     *   * `true`: the directory or file will be returned (the `only` and `except` options will be ignored)
365
     *   * `false`: the directory or file will NOT be returned (the `only` and `except` options will be ignored)
366
     *   * `null`: the `only` and `except` options will determine whether the directory or file should be returned
367
     *
368
     * - `except`: array, list of patterns excluding from the results matching file or directory paths.
369
     *   Patterns ending with slash ('/') apply to directory paths only, and patterns not ending with '/'
370
     *   apply to file paths only. For example, '/a/b' matches all file paths ending with '/a/b';
371
     *   and `.svn/` matches directory paths ending with `.svn`.
372
     *   If the pattern does not contain a slash (`/`), it is treated as a shell glob pattern
373
     *   and checked for a match against the pathname relative to `$dir`.
374
     *   Otherwise, the pattern is treated as a shell glob suitable for consumption by `fnmatch(3)`
375
     *   `with the `FNM_PATHNAME` flag: wildcards in the pattern will not match a `/` in the pathname.
376
     *   For example, `views/*.php` matches `views/index.php` but not `views/controller/index.php`.
377
     *   A leading slash matches the beginning of the pathname. For example, `/*.php` matches `index.php` but not `views/start/index.php`.
378
     *   An optional prefix `!` which negates the pattern; any matching file excluded by a previous pattern will become included again.
379
     *   If a negated pattern matches, this will override lower precedence patterns sources. Put a backslash (`\`) in front of the first `!`
380
     *   for patterns that begin with a literal `!`, for example, `\!important!.txt`.
381
     *   Note, the '/' characters in a pattern matches both '/' and '\' in the paths.
382
     * - `only`: array, list of patterns that the file paths should match if they are to be returned. Directory paths
383
     *   are not checked against them. Same pattern matching rules as in the `except` option are used.
384
     *   If a file path matches a pattern in both `only` and `except`, it will NOT be returned.
385
     * - `caseSensitive`: boolean, whether patterns specified at `only` or `except` should be case sensitive. Defaults to `true`.
386
     * - `recursive`: boolean, whether the files under the subdirectories should also be looked for. Defaults to `true`.
387
     * @return array files found under the directory, in no particular order. Ordering depends on the files system used.
388
     * @throws InvalidParamException if the dir is invalid.
389
     */
390 44
    public static function findFiles($dir, $options = [])
391
    {
392 44
        if (!is_dir($dir)) {
393
            throw new InvalidParamException("The dir argument must be a directory: $dir");
394
        }
395 44
        $dir = rtrim($dir, DIRECTORY_SEPARATOR);
396 44
        if (!isset($options['basePath'])) {
397
            // this should be done only once
398 44
            $options['basePath'] = realpath($dir);
399 44
            $options = self::normalizeOptions($options);
400 44
        }
401 44
        $list = [];
402 44
        $handle = opendir($dir);
403 44
        if ($handle === false) {
404
            throw new InvalidParamException("Unable to open directory: $dir");
405
        }
406 44
        while (($file = readdir($handle)) !== false) {
407 44
            if ($file === '.' || $file === '..') {
408 44
                continue;
409
            }
410 44
            $path = $dir . DIRECTORY_SEPARATOR . $file;
411 44
            if (static::filterPath($path, $options)) {
412 42
                if (is_file($path)) {
413 42
                    $list[] = $path;
414 42
                } elseif (!isset($options['recursive']) || $options['recursive']) {
415 3
                    $list = array_merge($list, static::findFiles($path, $options));
416 3
                }
417 42
            }
418 44
        }
419 44
        closedir($handle);
420
421 44
        return $list;
422
    }
423
424
    /**
425
     * Checks if the given file path satisfies the filtering options.
426
     * @param string $path the path of the file or directory to be checked
427
     * @param array $options the filtering options. See [[findFiles()]] for explanations of
428
     * the supported options.
429
     * @return boolean whether the file or directory satisfies the filtering options.
430
     */
431 51
    public static function filterPath($path, $options)
432
    {
433 51
        if (isset($options['filter'])) {
434 1
            $result = call_user_func($options['filter'], $path);
435 1
            if (is_bool($result)) {
436 1
                return $result;
437
            }
438
        }
439
440 50
        if (empty($options['except']) && empty($options['only'])) {
441 18
            return true;
442
        }
443
444 34
        $path = str_replace('\\', '/', $path);
445
446 34
        if (!empty($options['except'])) {
447 20
            if (($except = self::lastExcludeMatchingFromList($options['basePath'], $path, $options['except'])) !== null) {
448 2
                return $except['flags'] & self::PATTERN_NEGATIVE;
449
            }
450 19
        }
451
452 34
        if (!empty($options['only']) && !is_dir($path)) {
453 33
            if (($except = self::lastExcludeMatchingFromList($options['basePath'], $path, $options['only'])) !== null) {
454
                // don't check PATTERN_NEGATIVE since those entries are not prefixed with !
455 31
                return true;
456
            }
457
458 14
            return false;
459
        }
460
461 2
        return true;
462
    }
463
464
    /**
465
     * Creates a new directory.
466
     *
467
     * This method is similar to the PHP `mkdir()` function except that
468
     * it uses `chmod()` to set the permission of the created directory
469
     * in order to avoid the impact of the `umask` setting.
470
     *
471
     * @param string $path path of the directory to be created.
472
     * @param integer $mode the permission to be set for the created directory.
473
     * @param boolean $recursive whether to create parent directories if they do not exist.
474
     * @return boolean whether the directory is created successfully
475
     * @throws \yii\base\Exception if the directory could not be created (i.e. php error due to parallel changes)
476
     */
477 120
    public static function createDirectory($path, $mode = 0775, $recursive = true)
478
    {
479 120
        if (is_dir($path)) {
480 39
            return true;
481
        }
482 112
        $parentDir = dirname($path);
483
        // recurse if parent dir does not exist and we are not at the root of the file system.
484 112
        if ($recursive && !is_dir($parentDir) && $parentDir !== $path) {
485 2
            static::createDirectory($parentDir, $mode, true);
486 2
        }
487
        try {
488 112
            if (!mkdir($path, $mode)) {
489
                return false;
490
            }
491 112
        } catch (\Exception $e) {
492
            if (!is_dir($path)) {// https://github.com/yiisoft/yii2/issues/9288
493
                throw new \yii\base\Exception("Failed to create directory \"$path\": " . $e->getMessage(), $e->getCode(), $e);
494
            }
495
        }
496
        try {
497 112
            return chmod($path, $mode);
498
        } catch (\Exception $e) {
499
            throw new \yii\base\Exception("Failed to change permissions for directory \"$path\": " . $e->getMessage(), $e->getCode(), $e);
500
        }
501
    }
502
503
    /**
504
     * Performs a simple comparison of file or directory names.
505
     *
506
     * Based on match_basename() from dir.c of git 1.8.5.3 sources.
507
     *
508
     * @param string $baseName file or directory name to compare with the pattern
509
     * @param string $pattern the pattern that $baseName will be compared against
510
     * @param integer|boolean $firstWildcard location of first wildcard character in the $pattern
511
     * @param integer $flags pattern flags
512
     * @return boolean whether the name matches against pattern
513
     */
514 34
    private static function matchBasename($baseName, $pattern, $firstWildcard, $flags)
515
    {
516 34
        if ($firstWildcard === false) {
517 27
            if ($pattern === $baseName) {
518 7
                return true;
519
            }
520 34
        } elseif ($flags & self::PATTERN_ENDSWITH) {
521
            /* "*literal" matching against "fooliteral" */
522 25
            $n = StringHelper::byteLength($pattern);
523 25
            if (StringHelper::byteSubstr($pattern, 1, $n) === StringHelper::byteSubstr($baseName, -$n, $n)) {
524
                return true;
525
            }
526 25
        }
527
528 34
        $fnmatchFlags = 0;
529 34
        if ($flags & self::PATTERN_CASE_INSENSITIVE) {
530 1
            $fnmatchFlags |= FNM_CASEFOLD;
531 1
        }
532
533 34
        return fnmatch($pattern, $baseName, $fnmatchFlags);
534
    }
535
536
    /**
537
     * Compares a path part against a pattern with optional wildcards.
538
     *
539
     * Based on match_pathname() from dir.c of git 1.8.5.3 sources.
540
     *
541
     * @param string $path full path to compare
542
     * @param string $basePath base of path that will not be compared
543
     * @param string $pattern the pattern that path part will be compared against
544
     * @param integer|boolean $firstWildcard location of first wildcard character in the $pattern
545
     * @param integer $flags pattern flags
546
     * @return boolean whether the path part matches against pattern
547
     */
548 19
    private static function matchPathname($path, $basePath, $pattern, $firstWildcard, $flags)
549
    {
550
        // match with FNM_PATHNAME; the pattern has base implicitly in front of it.
551 19
        if (isset($pattern[0]) && $pattern[0] === '/') {
552 19
            $pattern = StringHelper::byteSubstr($pattern, 1, StringHelper::byteLength($pattern));
553 19
            if ($firstWildcard !== false && $firstWildcard !== 0) {
554
                $firstWildcard--;
555
            }
556 19
        }
557
558 19
        $namelen = StringHelper::byteLength($path) - (empty($basePath) ? 0 : StringHelper::byteLength($basePath) + 1);
559 19
        $name = StringHelper::byteSubstr($path, -$namelen, $namelen);
560
561 19
        if ($firstWildcard !== 0) {
562 19
            if ($firstWildcard === false) {
563 19
                $firstWildcard = StringHelper::byteLength($pattern);
564 19
            }
565
            // if the non-wildcard part is longer than the remaining pathname, surely it cannot match.
566 19
            if ($firstWildcard > $namelen) {
567
                return false;
568
            }
569
570 19
            if (strncmp($pattern, $name, $firstWildcard)) {
571 19
                return false;
572
            }
573 1
            $pattern = StringHelper::byteSubstr($pattern, $firstWildcard, StringHelper::byteLength($pattern));
574 1
            $name = StringHelper::byteSubstr($name, $firstWildcard, $namelen);
575
576
            // If the whole pattern did not have a wildcard, then our prefix match is all we need; we do not need to call fnmatch at all.
577 1
            if (empty($pattern) && empty($name)) {
578 1
                return true;
579
            }
580
        }
581
582 1
        $fnmatchFlags = FNM_PATHNAME;
583 1
        if ($flags & self::PATTERN_CASE_INSENSITIVE) {
584
            $fnmatchFlags |= FNM_CASEFOLD;
585
        }
586
587 1
        return fnmatch($pattern, $name, $fnmatchFlags);
588
    }
589
590
    /**
591
     * Scan the given exclude list in reverse to see whether pathname
592
     * should be ignored.  The first match (i.e. the last on the list), if
593
     * any, determines the fate.  Returns the element which
594
     * matched, or null for undecided.
595
     *
596
     * Based on last_exclude_matching_from_list() from dir.c of git 1.8.5.3 sources.
597
     *
598
     * @param string $basePath
599
     * @param string $path
600
     * @param array $excludes list of patterns to match $path against
601
     * @return string null or one of $excludes item as an array with keys: 'pattern', 'flags'
602
     * @throws InvalidParamException if any of the exclude patterns is not a string or an array with keys: pattern, flags, firstWildcard.
603
     */
604 34
    private static function lastExcludeMatchingFromList($basePath, $path, $excludes)
605
    {
606 34
        foreach (array_reverse($excludes) as $exclude) {
607 34
            if (is_string($exclude)) {
608
                $exclude = self::parseExcludePattern($exclude, false);
609
            }
610 34
            if (!isset($exclude['pattern']) || !isset($exclude['flags']) || !isset($exclude['firstWildcard'])) {
611
                throw new InvalidParamException('If exclude/include pattern is an array it must contain the pattern, flags and firstWildcard keys.');
612
            }
613 34
            if ($exclude['flags'] & self::PATTERN_MUSTBEDIR && !is_dir($path)) {
614
                continue;
615
            }
616
617 34
            if ($exclude['flags'] & self::PATTERN_NODIR) {
618 34
                if (self::matchBasename(basename($path), $exclude['pattern'], $exclude['firstWildcard'], $exclude['flags'])) {
619 32
                    return $exclude;
620
                }
621 33
                continue;
622
            }
623
624 19
            if (self::matchPathname($path, $basePath, $exclude['pattern'], $exclude['firstWildcard'], $exclude['flags'])) {
625 1
                return $exclude;
626
            }
627 33
        }
628
629 33
        return null;
630
    }
631
632
    /**
633
     * Processes the pattern, stripping special characters like / and ! from the beginning and settings flags instead.
634
     * @param string $pattern
635
     * @param boolean $caseSensitive
636
     * @throws \yii\base\InvalidParamException
637
     * @return array with keys: (string) pattern, (int) flags, (int|boolean) firstWildcard
638
     */
639 34
    private static function parseExcludePattern($pattern, $caseSensitive)
640
    {
641 34
        if (!is_string($pattern)) {
642
            throw new InvalidParamException('Exclude/include pattern must be a string.');
643
        }
644
645
        $result = [
646 34
            'pattern' => $pattern,
647 34
            'flags' => 0,
648 34
            'firstWildcard' => false,
649 34
        ];
650
651 34
        if (!$caseSensitive) {
652 1
            $result['flags'] |= self::PATTERN_CASE_INSENSITIVE;
653 1
        }
654
655 34
        if (!isset($pattern[0])) {
656
            return $result;
657
        }
658
659 34
        if ($pattern[0] === '!') {
660
            $result['flags'] |= self::PATTERN_NEGATIVE;
661
            $pattern = StringHelper::byteSubstr($pattern, 1, StringHelper::byteLength($pattern));
662
        }
663 34
        if (StringHelper::byteLength($pattern) && StringHelper::byteSubstr($pattern, -1, 1) === '/') {
664
            $pattern = StringHelper::byteSubstr($pattern, 0, -1);
665
            $result['flags'] |= self::PATTERN_MUSTBEDIR;
666
        }
667 34
        if (strpos($pattern, '/') === false) {
668 34
            $result['flags'] |= self::PATTERN_NODIR;
669 34
        }
670 34
        $result['firstWildcard'] = self::firstWildcardInPattern($pattern);
671 34
        if ($pattern[0] === '*' && self::firstWildcardInPattern(StringHelper::byteSubstr($pattern, 1, StringHelper::byteLength($pattern))) === false) {
672 25
            $result['flags'] |= self::PATTERN_ENDSWITH;
673 25
        }
674 34
        $result['pattern'] = $pattern;
675
676 34
        return $result;
677
    }
678
679
    /**
680
     * Searches for the first wildcard character in the pattern.
681
     * @param string $pattern the pattern to search in
682
     * @return integer|boolean position of first wildcard character or false if not found
683
     */
684 34
    private static function firstWildcardInPattern($pattern)
685
    {
686 34
        $wildcards = ['*', '?', '[', '\\'];
687 34
        $wildcardSearch = function ($r, $c) use ($pattern) {
688 34
            $p = strpos($pattern, $c);
689
690 34
            return $r === false ? $p : ($p === false ? $r : min($r, $p));
691 34
        };
692
693 34
        return array_reduce($wildcards, $wildcardSearch, false);
694
    }
695
696
    /**
697
     * @param array $options raw options
698
     * @return array normalized options
699
     */
700 53
    private static function normalizeOptions(array $options)
701
    {
702 53
        if (!array_key_exists('caseSensitive', $options)) {
703 52
            $options['caseSensitive'] = true;
704 52
        }
705 53
        if (isset($options['except'])) {
706 20
            foreach ($options['except'] as $key => $value) {
707 20
                if (is_string($value)) {
708 20
                    $options['except'][$key] = self::parseExcludePattern($value, $options['caseSensitive']);
709 20
                }
710 20
            }
711 20
        }
712 53
        if (isset($options['only'])) {
713 33
            foreach ($options['only'] as $key => $value) {
714 33
                if (is_string($value)) {
715 33
                    $options['only'][$key] = self::parseExcludePattern($value, $options['caseSensitive']);
716 33
                }
717 33
            }
718 33
        }
719 53
        return $options;
720
    }
721
}
722