artabro/wire/core/WireFileTools.php

2017 lines
75 KiB
PHP
Raw Permalink Normal View History

2024-08-27 11:35:37 +02:00
<?php namespace ProcessWire;
/**
* ProcessWire File Tools ($files API variable)
*
* #pw-summary Helpers for working with files and directories.
* #pw-var-files
*
* ProcessWire 3.x, Copyright 2022 by Ryan Cramer
* https://processwire.com
*
* @method bool include($filename, array $vars = array(), array $options = array())
*
*/
class WireFileTools extends Wire {
/**
* Active file data (as used by getCSV for example)
*
* @var array
*
*/
protected $data = array();
/**
* Destruct
*
*/
public function __destruct() {
foreach($this->data as $value) {
if(isset($value['fp'])) fclose($value['fp']);
}
}
/**
* Create a directory that is writable to ProcessWire and uses the defined $config chmod settings
*
* Unlike PHP's `mkdir()` function, this function manages the read/write mode consistent with ProcessWire's
* setting `$config->chmodDir`, and it can create directories recursively. Meaning, if you want to create directory /a/b/c/
* and directory /a/ doesn't yet exist, this method will take care of creating /a/, /a/b/, and /a/b/c/.
*
* The `$recursive` and `$chmod` arguments may optionally be swapped (since 3.0.34).
*
* ~~~~~
* // Create a new directory in ProcessWire's cache dir
* if($files->mkdir($config->paths->cache . 'foo-bar/')) {
* // directory created: /site/assets/cache/foo-bar/
* }
* ~~~~~
*
* #pw-group-manipulation
*
* @param string $path Directory you want to create
* @param bool|string $recursive If set to true, all directories will be created as needed to reach the end.
* @param string|null|bool $chmod Optional mode to set directory to (default: $config->chmodDir), format must be a string i.e. "0755"
* If omitted, then ProcessWire's `$config->chmodDir` setting is used instead.
* @return bool True on success, false on failure
*
*/
public function mkdir($path, $recursive = false, $chmod = null) {
if(!strlen("$path")) return false;
if(is_string($recursive) && strlen($recursive) > 2) {
// chmod argument specified as $recursive argument or arguments swapped
$_chmod = $recursive;
$recursive = is_bool($chmod) ? $chmod : false;
$chmod = $_chmod;
}
if(!is_dir($path)) {
if($recursive) {
$parentPath = substr($path, 0, strrpos(rtrim($path, '/'), '/'));
if(!is_dir($parentPath) && !$this->mkdir($parentPath, true, $chmod)) return false;
}
if(!@mkdir($path)) {
return $this->filesError(__FUNCTION__, "Unable to mkdir $path");
}
}
$this->chmod($path, false, $chmod);
return true;
}
/**
* Remove a directory and optionally everything within it (recursively)
*
* Unlike PHP's `rmdir()` function, this method provides a recursive option, which can be enabled by specifying true
* for the `$recursive` argument. You should be careful with this option, as it can easily wipe out an entire
* directory tree in a flash.
*
* Note that the $options argument was added in 3.0.118.
*
* ~~~~~
* // Remove directory /site/assets/cache/foo-bar/ and everything in it
* $files->rmdir($config->paths->cache . 'foo-bar/', true);
*
* // Remove directory after ensuring $pathname is somewhere within /site/assets/
* $files->rmdir($pathname, true, [ 'limitPath' => $config->paths->assets ]);
* ~~~~~
*
* #pw-group-manipulation
*
* @param string $path Path/directory you want to remove
* @param bool $recursive If set to true, all files and directories in $path will be recursively removed as well (default=false).
* @param array|bool|string $options Optional settings to adjust behavior or (bool|string) for limitPath option:
* - `limitPath` (string|bool|array): Must be somewhere within given path, boolean true for site assets, or false to disable (default=false).
* - `throw` (bool): Throw verbose WireException (rather than return false) when potentially consequential fail (default=false).
* @return bool True on success, false on failure
*
*/
public function rmdir($path, $recursive = false, $options = array()) {
$defaults = array(
'limitPath' => false,
'throw' => false,
);
if(!is_array($options)) $options = array('limitPath' => $options);
$options = array_merge($defaults, $options);
// if there's nothing to remove, exit now
if(!is_dir($path)) return false;
// verify that path is allowed for this operation
if(!$this->allowPath($path, $options['limitPath'], $options['throw'])) return false;
// handle recursive rmdir
if($recursive === true) {
$files = @scandir($path);
if(is_array($files)) foreach($files as $file) {
if($file == '.' || $file == '..' || strpos($file, '..') !== false) continue;
$pathname = rtrim($path, '/') . '/' . $file;
if(is_dir($pathname)) {
$this->rmdir($pathname, true, $options);
} else {
$this->unlink($pathname, $options['limitPath'], $options['throw']);
}
}
}
if(@rmdir($path)) {
return true;
} else {
return $this->filesError(__FUNCTION__, "Unable to rmdir: $path", $options);
}
}
/**
* Change the read/write mode of a file or directory, consistent with ProcessWire's configuration settings
*
* Unless a specific mode is provided via the `$chmod` argument, this method uses the `$config->chmodDir`
* and `$config->chmodFile` settings in /site/config.php.
*
* This method also provides the option of going recursive, adjusting the read/write mode for an entire
* file/directory tree at once.
*
* The `$recursive` or `$chmod` arguments may be optionally swapped in order (since 3.0.34).
*
* ~~~~~
* // Update the mode of /site/assets/cache/foo-bar/ recursively
* $files->chmod($config->paths->cache . 'foo-bar/', true);
* ~~~~~
*
* #pw-group-manipulation
*
* @param string $path Path or file that you want to adjust mode for (may be a path/directory or a filename).
* @param bool|string $recursive If set to true, all files and directories in $path will be recursively set as well (default=false).
* @param string|null|bool $chmod If you want to set the mode to something other than ProcessWire's chmodFile/chmodDir settings,
* you may override it by specifying it here. Ignored otherwise. Format should be a string, like "0755".
* @return bool Returns true if all changes were successful, or false if at least one chmod failed.
* @throws WireException when it receives incorrect chmod format
*
*/
public function chmod($path, $recursive = false, $chmod = null) {
if(is_string($recursive) && strlen($recursive) > 2) {
// chmod argument specified as $recursive argument or arguments swapped
$_chmod = $recursive;
$recursive = is_bool($chmod) ? $chmod : false;
$chmod = $_chmod;
}
if(is_null($chmod)) {
// default: pull values from PW config
$chmodFile = $this->wire()->config->chmodFile;
$chmodDir = $this->wire()->config->chmodDir;
} else {
// optional, manually specified string
if(!is_string($chmod)) {
$this->filesException(__FUNCTION__, "chmod must be specified as a string like '0755'");
}
$chmodFile = $chmod;
$chmodDir = $chmod;
}
$numFails = 0;
if(is_dir($path)) {
// $path is a directory
if($chmodDir) if(!@chmod($path, octdec($chmodDir))) $numFails++;
// change mode of files in directory, if recursive
if($recursive) foreach(new \DirectoryIterator($path) as $file) {
if($file->isDot()) continue;
$mod = $file->isDir() ? $chmodDir : $chmodFile;
if($mod) if(!@chmod($file->getPathname(), octdec($mod))) $numFails++;
if($file->isDir()) {
if(!$this->chmod($file->getPathname(), true, $chmod)) $numFails++;
}
}
} else {
// $path is a file
$mod = $chmodFile;
if($mod) if(!@chmod($path, octdec($mod))) $numFails++;
}
return $numFails == 0;
}
/**
* Copy all files recursively from one directory ($src) to another directory ($dst)
*
* Unlike PHP's `copy()` function, this method performs a recursive copy by default,
* ensuring that all files and directories in the source ($src) directory are duplicated
* in the destination ($dst) directory.
*
* This method can also be used to copy single files. If a file is specified for $src, and
* only a path is specified for $dst, then the original filename will be retained in $dst.
*
* ~~~~~
* // Copy everything from /site/assets/cache/foo/ to /site/assets/cache/bar/
* $copyFrom = $config->paths->cache . "foo/";
* $copyTo = $config->paths->cache . "bar/";
* $files->copy($copyFrom, $copyTo);
* ~~~~~
*
* #pw-group-manipulation
*
* @param string $src Path to copy files from, or filename to copy.
* @param string $dst Path (or filename) to copy file(s) to. Directory is created if it doesn't already exist.
* @param bool|array $options Array of options:
* - `recursive` (bool): Whether to copy directories within recursively. (default=true)
* - `allowEmptyDirs` (boolean): Copy directories even if they are empty? (default=true)
* - `limitPath` (bool|string|array): Limit copy to within path given here, or true for site assets path.
* The limitPath option requires core 3.0.118+. (default=false).
* - `hidden` (bool): Also copy hidden files/directories within given $src directory? (applies only if $src is dir)
* The hidden option requires core 3.0.180+. (default=true)
* - If a boolean is specified for $options, it is assumed to be the `recursive` option.
* @return bool True on success, false on failure.
* @throws WireException if `limitPath` option is used and either $src or $dst is not allowed
*
*/
public function copy($src, $dst, $options = array()) {
$defaults = array(
'recursive' => true,
'hidden' => true,
'allowEmptyDirs' => true,
'limitPath' => false,
);
if(is_bool($options)) $options = array('recursive' => $options);
$options = array_merge($defaults, $options);
if($options['limitPath'] !== false) {
$this->allowPath($src, $options['limitPath'], true);
$this->allowPath($dst, $options['limitPath'], true);
}
if(!is_dir($src)) {
// just copy a file
if(!file_exists($src)) return false;
if(is_dir($dst)) {
// if only a directory was specified for $dst, then keep same filename but in new dir
$dir = rtrim($dst, '/');
$dst .= '/' . basename($src);
} else {
$dir = dirname($dst);
}
if(!is_dir($dir)) $this->mkdir($dir);
if(!copy($src, $dst)) return false;
$this->chmod($dst);
return true;
}
if(substr($src, -1) != '/') $src .= '/';
if(substr($dst, -1) != '/') $dst .= '/';
$dir = opendir($src);
if(!$dir) return false;
if(!$options['allowEmptyDirs']) {
$isEmpty = true;
while(false !== ($file = readdir($dir))) {
if($file == '.' || $file == '..') continue;
if(!$options['hidden'] && strpos(basename($file), '.') === 0) continue;
$isEmpty = false;
break;
}
if($isEmpty) return true;
}
if(!$this->mkdir($dst)) return false;
while(false !== ($file = readdir($dir))) {
if($file == '.' || $file == '..') continue;
if(!$options['hidden'] && strpos(basename($file), '.') === 0) continue;
$isDir = is_dir($src . $file);
if($options['recursive'] && $isDir) {
$this->copy($src . $file, $dst . $file, $options);
} else if($isDir) {
// skip it, because not recursive
} else {
copy($src . $file, $dst . $file);
$this->chmod($dst . $file);
}
}
closedir($dir);
return true;
}
/**
* Unlink/delete file with additional protections relative to PHP unlink()
*
* - This method requires a full pathname to a file to unlink and does not
* accept any kind of relative path traversal.
*
* - This method will be limited to unlink files only in /site/assets/ if you
* specify `true` for the `$limitPath` option (recommended).
*
* #pw-group-manipulation
*
* @param string $filename
* @param string|bool $limitPath Limit only to files within some starting path? (default=false)
* - Boolean true to limit unlink operations to somewhere within /site/assets/ (only known always writable path).
* - Boolean false to disable to security feature. (default)
* - An alternative path (string) that represents the starting path (full disk path) to limit deletions to.
* - An array with multiple of the above string option.
* @param bool $throw Throw exception on error?
* @return bool True on success, false on fail
* @throws WireException If file is not allowed to be removed or unlink fails
* @since 3.0.118
*
*/
public function unlink($filename, $limitPath = false, $throw = false) {
if(!$this->allowPath($filename, $limitPath, $throw)) {
// path not allowed
return false;
}
if(!is_file($filename) && !is_link($filename)) {
// only files or links (that exist) can be deleted
return $this->filesError(__FUNCTION__, "Given filename is not a file or link: $filename");
}
if(@unlink($filename)) {
return true;
} else {
return $this->filesError(__FUNCTION__, "Unable to unlink file: $filename", $throw);
}
}
/**
* Rename a file or directory and update permissions
*
* Note that this method will fail if pathname given by $newName argument already exists.
*
* #pw-group-manipulation
*
* @param string $oldName Old pathname, must be full disk path.
* @param string $newName New pathname, must be full disk path OR can be basename to assume same path as $oldName.
* @param array|bool|string $options Options array to modify behavior or substitute `limitPath` (bool or string) option here.
* - `limitPath` (bool|string|array): Limit renames to within this path, or boolean TRUE for site/assets, or FALSE to disable (default=false).
* - `throw` (bool): Throw WireException with verbose details on error? (default=false)
* - `chmod` (bool): Adjust permissions to be consistent with $config after rename? (default=true)
* - `copy` (bool): Use copy-then-delete method rather than a file system rename. (default=false) 3.0.178+
* - `retry` (bool): Retry with 'copy' method if regular rename files, applies only if copy option is false. (default=true) 3.0.178+
* - If given a bool or string for $options the `limitPath` option is assumed.
* @return bool True on success, false on fail (or WireException if throw option specified).
* @throws WireException If error occurs and $throw argument was true.
* @since 3.0.118
*
*/
public function rename($oldName, $newName, $options = array()) {
$defaults = array(
'limitPath' => false,
'throw' => false,
'chmod' => true,
'copy' => false,
'retry' => true,
);
if(!is_array($options)) $options = array('limitPath' => $options);
$options = array_merge($defaults, $options);
// if only basename was specified for the newName then use path from oldName
if(basename($newName) === $newName) {
$newName = dirname($oldName) . '/' . $newName;
}
try {
$this->allowPath($oldName, $options['limitPath'], true);
} catch(\Exception $e) {
return $this->filesError(__FUNCTION__, '$oldName path invalid: ' . $e->getMessage(), $options);
}
try {
$this->allowPath($newName, $options['limitPath'], true);
} catch(\Exception $e) {
return $this->filesError(__FUNCTION__, 'Rename $newName path invalid: ' . $e->getMessage(), $options);
}
if(!file_exists($oldName)) {
return $this->filesError(__FUNCTION__, 'Given pathname ($oldName) that does not exist: ' . $oldName, $options);
}
if(file_exists($newName)) {
return $this->filesError(__FUNCTION__, 'Rename to pathname ($newName) that already exists: ' . $newName, $options);
}
if($options['copy']) {
// will use recursive copy method only
$success = false;
} else if($options['retry']) {
// consider any error/warnings from rename a non-event since we can retry
$success = @rename($oldName, $newName);
} else {
// use real rename only
$success = rename($oldName, $newName);
}
if(!$success && ($options['retry'] || $options['copy'])) {
$opt = array(
'limitPath' => $options['limitPath'],
'throw' => $options['throw'],
);
if($this->copy($oldName, $newName, $opt)) {
$success = true;
if(is_dir($oldName)) {
if(!$this->rmdir($oldName, true, $opt)) {
$this->filesError(__FUNCTION__, 'Unable to rmdir source ($oldName): ' . $oldName);
}
} else {
if(!$this->unlink($oldName, $opt['limitPath'], $opt['throw'])) {
$this->filesError(__FUNCTION__, 'Unable to unlink source ($oldName): ' . $oldName);
}
}
}
}
if($success) {
if($options['chmod']) $this->chmod($newName);
} else {
$this->filesError(__FUNCTION__, "Failed: $oldName => $newName", $options);
}
return $success;
}
/**
* Rename by first copying files to destination and then deleting source files
*
* The operation is considered successful so long as the source files were able to be copied to the destination.
* If source files cannot be deleted afterwards, the warning is logged, plus a warning notice is also shown when in debug mode.
*
* #pw-group-manipulation
*
* @param string $oldName Old pathname, must be full disk path.
* @param string $newName New pathname, must be full disk path OR can be basename to assume same path as $oldName.
* @param array $options See options for rename() method
* @return bool
* @throws WireException
* @since 3.0.178
*
*/
public function renameCopy($oldName, $newName, $options = array()) {
$options['copy'] = true;
return $this->rename($oldName, $newName, $options);
}
/**
* Does the given file/link/dir exist?
*
* Thie method accepts an `$options` argument that can be specified as an array
* or a string (space or comma separated). The examples here demonstrate usage as
* a string since it is the simplest for readability.
*
* - This function may return false for symlinks pointing to non-existing
* files, unless you specify `link` as the `type`.
* - Specifying `false` for the `readable` or `writable` argument disables the
* option from being used, it doesnt perform a NOT condition.
* - The `writable` option may also be written as `writeable`, if preferred.
*
* ~~~~~
* // 1. check if exists
* $exists = $files->exists('/path/file.ext');
*
* // 2. check if exists and is readable (or writable)
* $exists = $files->exists('/path/file.ext', 'readable');
* $exists = $files->exists('/path/file.ext', 'writable');
*
* // 3. check if exists and is file, link or dir
* $exists = $files->exists('/path/file.ext', 'file');
* $exists = $files->exists('/path/file.ext', 'link');
* $exists = $files->exists('/path/file.ext', 'dir');
*
* // 4. check if exists and is writable file or dir
* $exists = $files->exists('/path/file.ext', 'writable file');
* $exists = $files->exists('/path/dir/', 'writable dir');
*
* // 5. check if exists and is readable and writable file
* $exists = $files->exists('/path/file.ext', 'readable writable file');
* ~~~~~
*
* #pw-group-retrieval
*
* @param string $filename
* @param array|string $options Can be specified as array or string:
* - `type` (string): Verify it is of type: 'file', 'link', 'dir' (default='')
* - `readable` (bool): Verify it is readable? (default=false)
* - `writable` (bool): Also verify the file is writable? (default=false)
* - `writeable` (bool): Alias of writable (default=false)
* - When specified as string, you can use any combination of the words:
* `readable, writable, file, link, dir` (separated by space or comma).
* @return bool
* @throws WireException if given invalid or unrecognized $options
* @since 3.0.180
*
*
*/
public function exists($filename, $options = '') {
$defaults = array(
'type' => '',
'readable' => false,
'writable' => false,
'writeable' => false, // alias of writable
);
if($options === '') {
$options = $defaults;
} else if(is_array($options)) {
$options = array_merge($defaults, $options);
if(!empty($options['type'])) $options['type'] = strtolower(trim($options['type']));
} else if(is_string($options)) {
$types = array('file', 'link', 'dir');
if(strpos($options, ',') !== false) $options = str_replace(',', ' ', $options);
foreach(explode(' ', $options) as $option) {
$option = strtolower(trim($option));
if(empty($option)) continue;
if(isset($defaults[$option])) {
// readable, writable
$defaults[$option] = true;
} else if(in_array($option, $types, true)) {
// file, dir, link
if(empty($defaults['type'])) $defaults['type'] = $option;
} else {
throw new WireException("Unrecognized option: $option");
}
}
$options = $defaults;
} else {
throw new WireException('Invalid $options argument');
}
if($options['readable'] && !is_readable($filename)) {
$exists = false;
} else if(($options['writable'] || $options['writeable']) && !is_writable($filename)) {
$exists = false;
} else if($options['type'] === '') {
$exists = $options['readable'] ? true : file_exists($filename);
} else if($options['type'] === 'file') {
$exists = is_file($filename);
} else if($options['type'] === 'link') {
$exists = is_link($filename);
} else if($options['type'] === 'dir') {
$exists = is_dir($filename);
} else {
throw new WireException("Unrecognized 'type' option: $options[type]");
}
return $exists;
}
/**
* Get size of file or directory (in bytes)
*
* @param string $path File or directory path
* @param array|bool $options Options array, or boolean true for getString option:
* - `getString` (bool): Get string that summarizes bytes, kB, MB, etc.? (default=false)
* @return int|string
* @since 3.0.214
*
*/
public function size($path, $options = array()) {
if(is_bool($options)) $options = array('getString' => $options);
$size = 0;
$path = realpath($path);
if(!is_string($path) || $path === '' || !file_exists($path)) return 0;
if(is_dir($path)) {
$dir = new \RecursiveDirectoryIterator($path, \FilesystemIterator::SKIP_DOTS);
foreach(new \RecursiveIteratorIterator($dir) as $file) {
try {
$size += $file->getSize();
} catch(\Exception $e) {
// ok
}
}
} else {
$size = (int) filesize($path);
}
return empty($options['getString']) ? $size : wireBytesStr($size);
}
/**
* Allow path or filename to to be used for file manipulation actions?
*
* Given path must be a full path (no relative references). If given a $limitPath, it must be a
* directory that already exists.
*
* Note that this method does not indicate whether or not the given pathname exists, only that it is allowed.
* As a result this can be used for checking a path before creating something in it too.
*
* #pw-internal
*
* @param string $pathname File or directory name to check
* @param bool|string|array $limitPath Any one of the following (default=false):
* - Full disk path (string) that $pathname must be within (whether directly or in subdirectory of).
* - Array of the above.
* - Boolean false to disable (default).
* - Boolean true for site assets path, which is the only known always-writable path in PW.
* @param bool $throw Throw verbose exceptions on error? (default=false).
* @return bool True if given pathname allowed, false if not.
* @throws WireException when $throw argument is true and function would otherwise return false.
* @since 3.0.118
*
*/
public function allowPath($pathname, $limitPath = false, $throw = false) {
if(is_array($limitPath)) {
// run allowPath() for each of the specified limitPaths
$allow = false;
foreach($limitPath as $dir) {
if(!is_string($dir) || empty($dir)) continue;
$allow = $this->allowPath($pathname, $dir, false);
if($allow) break; // found one that is allowed
}
if(!$allow) {
$this->filesError(__FUNCTION__, "Given pathname is not within any of the paths allowed by limitPath", $throw);
}
return $allow;
} else if($limitPath === true) {
// default limitPath
$limitPath = $this->wire()->config->paths->assets;
} else if($limitPath === false) {
// no limitPath in use
} else if(empty($limitPath) || !is_string($limitPath)) {
// invalid limitPath argument (wrong type or path does not exist)
return $this->filesError(__FUNCTION__, "Invalid type for limitPath argument", $throw);
} else if(!is_dir($limitPath)) {
return $this->filesError(__FUNCTION__, "$limitPath (limitPath) does not exist", $throw);
}
if($limitPath !== false) try {
// if limitPath can't pass allowPath then neither can $pathname
$this->allowPath($limitPath, false, true);
} catch(\Exception $e) {
return $this->filesError(__FUNCTION__, "Validating limitPath reported: " . $e->getMessage(), $throw, $e);
}
if(DIRECTORY_SEPARATOR != '/') {
$pathname = str_replace(DIRECTORY_SEPARATOR, '/', $pathname);
if(is_string($limitPath)) $limitPath = str_replace(DIRECTORY_SEPARATOR, '/', $limitPath);
$testname = $pathname;
if(strpos($pathname, ':')) list(,$testname) = explode(':', $pathname, 2); // reduce to no drive letter, if present
} else {
$testname = $pathname;
}
if(!strlen(trim($testname, '/.')) || substr_count($testname, '/') < 2) {
// do not allow paths that consist of nothing but slashes and/or dots
// and do not allow paths off root or lacking absolute path reference
return $this->filesError(__FUNCTION__, "pathname not allowed: $pathname", $throw);
}
if(strpos($pathname, '..') !== false) {
// not allowed to traverse anywhere
return $this->filesError(__FUNCTION__, 'pathname may not traverse “../”', $throw);
}
if(strpos($pathname, '.') === 0 || empty($pathname)) {
return $this->filesError(__FUNCTION__, 'pathname may not begin with “.”', $throw);
}
$pos = strpos($pathname, '//');
if($pos !== false && $pos !== strpos($this->wire()->config->paths->assets, '//')) {
// URLs or accidental extra slashes not allowed, unless they also appear in a known safe system path
return $this->filesError(__FUNCTION__, 'pathname may not contain double slash “//”', $throw);
}
if($limitPath !== false && strpos($pathname, $limitPath) !== 0) {
// disallow paths that do not begin with limitPath (i.e. /path/to/public_html/site/assets/)
return $this->filesError(__FUNCTION__, "Given pathname is not within $limitPath (limitPath)", $throw);
}
return true;
}
/**
* Return a new temporary directory/path ready to use for files
*
* - The temporary directory will be automatically removed at the end of the request.
* - Temporary directories are not http accessible.
* - If you call this with the same non-empty `$name` argument more than once in the
* same request, the same `WireTempDir` instance will be returned.
*
* #pw-advanced
*
* ~~~~~
* $tempDir = $files->tempDir();
* $path = $tempDir->get();
* file_put_contents($path . 'some-file.txt', 'Hello world');
* ~~~~~
*
* @param Object|string $name Any one of the following: (default='')
* - Omit this argument for auto-generated name, 3.0.178+
* - Name/word that you specify using fieldName format, i.e. [_a-zA-Z0-9].
* - Object instance that needs the temp dir.
* @param array|int $options Deprecated argument. Call `WireTempDir` methods if you need more options.
* @return WireTempDir Returns a WireTempDir instance. If you typecast return value to a string,
* it is the temp dir path (with trailing slash).
* @see WireTempDir
*
*/
public function tempDir($name = '', $options = array()) {
static $tempDirs = array();
if($name && isset($tempDirs[$name])) return $tempDirs[$name];
if(is_int($options)) $options = array('maxAge' => $options);
$basePath = isset($options['basePath']) ? $options['basePath'] : '';
$tempDir = new WireTempDir();
$this->wire($tempDir);
if(isset($options['remove']) && $options['remove'] === false) $tempDir->setRemove(false);
$tempDir->init($name, $basePath);
if(isset($options['maxAge'])) $tempDir->setMaxAge($options['maxAge']);
if($name) $tempDirs[$name] = $tempDir;
return $tempDir;
}
/**
* Find all files in the given $path recursively, and return a flat array of all found filenames
*
* #pw-group-retrieval
*
* @param string $path Path to start from (required).
* @param array $options Options to affect what is returned (optional):
* - `recursive` (int|bool): How many levels of subdirectories this method should descend into beyond the 1 given.
* Specify 1 to remain at the one directory level given, or 2+ to descend into subdirectories. (default=10)
* In 3.0.180+ you may also specify true for no limit, or false to disable descending into any subdirectories.
* - `extensions` (array|string): Only include files having these extensions, or omit to include all (default=[]).
* In 3.0.180+ the extensions argument may also be a string (space or comma separated).
* - `excludeDirNames` (array): Do not descend into directories having these names (default=[]).
* - `excludeHidden` (bool): Exclude hidden files? (default=false).
* - `allowDirs` (bool): Allow directories in returned files (except for '.' and '..')? Note that returned
* directories have a trailing slash. (default=false) 3.0.180+
* - `returnRelative` (bool): Make returned array have filenames relative to given start $path? (default=false)
* @return array Flat array of filenames
* @since 3.0.96
*
*/
public function find($path, array $options = array()) {
$defaults = array(
'recursive' => 10,
'extensions' => array(),
'excludeExtensions' => array(),
'excludeDirNames' => array(),
'excludeHidden' => false,
'allowDirs' => false,
'returnRelative' => false,
);
$path = $this->unixDirName($path);
if(!is_dir($path) || !is_readable($path)) return array();
$options = array_merge($defaults, $options);
if(empty($options['_level'])) {
// this is a non-recursive call
$options['_startPath'] = $path;
$options['_level'] = 0;
if(!is_array($options['extensions'])) {
if($options['extensions']) {
$options['extensions'] = preg_replace('/[,;\.\s]+/', ' ', $options['extensions']);
$options['extensions'] = explode(' ', $options['extensions']);
} else {
$options['extensions'] = array();
}
}
foreach($options['extensions'] as $k => $v) {
$options['extensions'][$k] = strtolower(trim($v));
}
}
$options['_level']++;
if($options['recursive'] && $options['recursive'] !== true) {
if($options['_level'] > $options['recursive']) return array();
}
$dirs = array();
$files = array();
foreach(new \DirectoryIterator($path) as $file) {
if($file->isDot()) continue;
$basename = $file->getBasename();
$ext = strtolower($file->getExtension());
if($file->isDir()) {
$dir = $this->unixDirName($file->getPathname());
if($options['allowDirs']) {
if($options['returnRelative'] && strpos($dir, $options['_startPath']) === 0) {
$dir = substr($dir, strlen($options['_startPath']));
}
$files[$dir] = $dir;
}
if($options['recursive'] === false || $options['recursive'] < 1) continue;
if(!in_array($basename, $options['excludeDirNames'])) $dirs[$dir] = $file->getPathname();
continue;
}
if($options['excludeHidden'] && strpos($basename, '.') === 0) continue;
if(!empty($options['extensions']) && !in_array($ext, $options['extensions'])) continue;
if(!empty($options['excludeExtensions']) && in_array($ext, $options['excludeExtensions'])) continue;
$filename = $this->unixFileName($file->getPathname());
if($options['returnRelative'] && strpos($filename, $options['_startPath']) === 0) {
$filename = substr($filename, strlen($options['_startPath']));
}
$files[] = $filename;
}
foreach($dirs as $key => $dir) {
$_files = $this->find($dir, $options);
if(count($_files)) {
foreach($_files as $name) {
$files[] = $name;
}
} else {
// no files found in directory
if($options['allowDirs'] && count($options['extensions']) && isset($files[$key])) {
// remove directory if it didn't match any files having requested extension
unset($files[$key]);
}
}
}
$options['_level']--;
if(!$options['_level']) sort($files);
return $files;
}
/**
* Unzips the given ZIP file to the destination directory
*
* ~~~~~
* // Unzip a file
* $zip = $config->paths->cache . "my-file.zip";
* $dst = $config->paths->cache . "my-files/";
* $items = $files->unzip($zip, $dst);
* if(count($items)) {
* // $items is an array of filenames that were unzipped into $dst
* }
* ~~~~~
*
* #pw-group-archives
*
* @param string $file ZIP file to extract
* @param string $dst Directory where files should be unzipped into. Directory is created if it doesn't exist.
* @return array Returns an array of filenames (excluding $dst) that were unzipped.
* @throws WireException All error conditions result in WireException being thrown.
* @see WireFileTools::zip()
*
*/
public function unzip($file, $dst) {
$dst = rtrim($dst, '/' . DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;
if(!class_exists('\ZipArchive')) $this->filesException(__FUNCTION__, "PHP's ZipArchive class does not exist");
if(!is_file($file)) $this->filesException(__FUNCTION__, "ZIP file does not exist");
if(!is_dir($dst)) $this->mkdir($dst, true);
$names = array();
$chmodFile = $this->wire()->config->chmodFile;
$chmodDir = $this->wire()->config->chmodDir;
$zip = new \ZipArchive();
$res = $zip->open($file);
if($res !== true) $this->filesException(__FUNCTION__, "Unable to open ZIP file, error code: $res");
for($i = 0; $i < $zip->numFiles; $i++) {
$name = $zip->getNameIndex($i);
if(strpos($name, '..') !== false) continue;
if($zip->extractTo($dst, $name)) {
$names[$i] = $name;
$filename = $dst . ltrim($name, '/');
if(is_dir($filename)) {
if($chmodDir) chmod($filename, octdec($chmodDir));
} else if(is_file($filename)) {
if($chmodFile) chmod($filename, octdec($chmodFile));
}
}
}
$zip->close();
return $names;
}
/**
* Creates a ZIP file
*
* ~~~~~
* // Create zip of all files in directory $dir to file $zip
* $dir = $config->paths->cache . "my-files/";
* $zip = $config->paths->cache . "my-file.zip";
* $result = $files->zip($zip, $dir);
*
* echo "<h3>These files were added to the ZIP:</h3>";
* foreach($result['files'] as $file) {
* echo "<li>" $sanitizer->entities($file) . "</li>";
* }
*
* if(count($result['errors'])) {
* echo "<h3>There were errors:</h3>";
* foreach($result['errors'] as $error) {
* echo "<li>" . $sanitizer->entities($error) . "</li>";
* }
* }
* ~~~~~
*
* #pw-group-archives
*
* @param string $zipfile Full path and filename to create or update (i.e. /path/to/myfile.zip)
* @param array|string $files Array of files to add (full path and filename), or directory (string) to add.
* If given a directory, it will recursively add everything in that directory.
* @param array $options Associative array of options to modify default behavior:
* - `allowHidden` (boolean or array): allow hidden files? May be boolean, or array of hidden files (basenames) you allow. (default=false)
* Note that if you actually specify a hidden file in your $files argument, then that overrides this.
* - `allowEmptyDirs` (boolean): allow empty directories in the ZIP file? (default=true)
* - `overwrite` (boolean): Replaces ZIP file if already present (rather than adding to it) (default=false)
* - `maxDepth` (int): Max dir depth 0 for no limit (default=0). Specify 1 to stay only in dirs listed in $files.
* - `exclude` (array): Files or directories to exclude
* - `dir` (string): Directory name to prepend to added files in the ZIP
* @return array Returns associative array of:
* - `files` (array): all files that were added
* - `errors` (array): files that failed to add, if any
* @throws WireException Original ZIP file creation error conditions result in WireException being thrown.
* @see WireFileTools::unzip()
*
*/
public function zip($zipfile, $files, array $options = array()) {
static $depth = 0;
$defaults = array(
'allowHidden' => false,
'allowEmptyDirs' => true,
'overwrite' => false,
'maxDepth' => 0,
'exclude' => array(), // files or dirs to exclude
'dir' => '',
'zip' => null, // internal use: holds ZipArchive instance for recursive use
);
$return = array(
'files' => array(),
'errors' => array(),
);
if(!empty($options['zip']) && !empty($options['dir']) && $options['zip'] instanceof \ZipArchive) {
// internal recursive call
$recursive = true;
$zip = $options['zip']; // ZipArchive instance
} else if(is_string($zipfile)) {
if(!class_exists('\ZipArchive')) $this->filesException(__FUNCTION__, "PHP's ZipArchive class does not exist");
$options = array_merge($defaults, $options);
$zippath = dirname($zipfile);
if(!is_dir($zippath)) $this->filesException(__FUNCTION__, "Path for ZIP file ($zippath) does not exist");
if(!is_writable($zippath)) $this->filesException(__FUNCTION__, "Path for ZIP file ($zippath) is not writable");
if(empty($files)) $this->filesException(__FUNCTION__, "Nothing to add to ZIP file $zipfile");
if(is_file($zipfile) && $options['overwrite'] && !$this->unlink($zipfile)) $this->filesException(__FUNCTION__, "Unable to overwrite $zipfile");
if(!is_array($files)) $files = array($files);
if(!is_array($options['exclude'])) $options['exclude'] = array($options['exclude']);
$recursive = false;
$zip = new \ZipArchive();
if($zip->open($zipfile, \ZipArchive::CREATE) !== true) $this->filesException(__FUNCTION__, "Unable to create ZIP: $zipfile");
} else {
$this->filesException(__FUNCTION__, "Invalid zipfile argument");
return array(); // not reachable
}
$dir = strlen($options['dir']) ? rtrim($options['dir'], '/') . '/' : '';
foreach($files as $file) {
$basename = basename($file);
$name = $dir . $basename;
if($basename[0] == '.' && $recursive) {
if(!$options['allowHidden']) continue;
if(is_array($options['allowHidden']) && !in_array($basename, $options['allowHidden'])) continue;
}
if(count($options['exclude'])) {
if(in_array($name, $options['exclude']) || in_array("$name/", $options['exclude'])) continue;
}
if(is_dir($file)) {
if($options['maxDepth'] > 0 && $depth >= $options['maxDepth']) continue;
$_files = array();
foreach(new \DirectoryIterator($file) as $f) {
if($f->isDot()) continue;
if($options['maxDepth'] > 0 && $f->isDir() && ($depth+1) >= $options['maxDepth']) continue;
$_files[] = $f->getPathname();
}
if(count($_files)) {
$zip->addEmptyDir($name);
$options['dir'] = "$name/";
$options['zip'] = $zip;
$depth++;
$_return = $this->zip($zipfile, $_files, $options);
$depth--;
foreach($_return['files'] as $s) $return['files'][] = $s;
foreach($_return['errors'] as $s) $return['errors'][] = $s;
} else if($options['allowEmptyDirs']) {
$zip->addEmptyDir($name);
}
} else if(file_exists($file)) {
if($zip->addFile($file, $name)) {
$return['files'][] = $name;
} else {
$return['errors'][] = $name;
}
}
}
if(!$recursive) $zip->close();
return $return;
}
/**
* Send the contents of the given filename to the current http connection
*
* This function utilizes the `$config->fileContentTypes` to match file extension to content type headers
* and force-download state.
*
* This function throws a `WireException` if the file cant be sent for some reason. Set the `throw` option to
* false if you want it to instead return integer 0 when errors occur.
*
* #pw-group-http
*
* @param string|bool $filename Full path and filename to send or boolean false if provided in `$options[data]`.
* @param array $options Optional options to modify default behavior:
* - `exit` (bool): Halt program execution after file send (default=true).
* - `partial` (bool): Allow use of partial downloads via HTTP_RANGE requests? Since 3.0.131 (default=true)
* - `forceDownload` (bool|null): Whether file should force download, or null to let content-type header decide (default=null).
* - `downloadFilename` (string): Filename you want the download to show on users computer, or omit to use existing (default='').
* - `headers` (array): The $headers argument to this method can also be provided as an option right here (default=[]). Since 3.0.131.
* - `data` (string): String of data to send rather than file, $filename argument must be false (default=''). Since 3.0.132.
* - `limitPath` (string|bool): Prefix disk path $filename must be within, false to disable, true for site/assets (default=false). Since 3.0.169.
* - `throw` (bool): Throw exceptions on error? When false, it will instead return integer 0 on errors (default=true). Since 3.0.169.
* @param array $headers Optional headers that are sent, below are the defaults:
* - `pragma`: public
* - `expires`: 0
* - `cache-control`: must-revalidate, post-check=0, pre-check=0
* - `content-type`: {content-type} (replaced with actual content type)
* - `content-transfer-encoding`: binary
* - `content-length`: {filesize} (replaced with actual filesize)
* - To remove a header completely, make its value NULL.
* - If preferred, the above headers can be specified in `$options[headers]` instead.
* @return int Returns bytes sent, only if `exit` option is false (since 3.0.169)
* @throws WireException
* @see WireHttp::sendFile()
*
*/
public function send($filename, array $options = array(), array $headers = array()) {
$defaults = array('limitPath' => $this->wire()->getStatus() === 32, 'throw' => true);
$options = array_merge($defaults, $options);
if($filename && !$this->allowPath($filename, $options['limitPath'], $options['throw'])) return 0;
$http = new WireHttp();
$this->wire($http);
try {
$result = $http->sendFile($filename, $options, $headers);
} catch(\Exception $e) {
$this->filesError(__FUNCTION__, $e->getMessage(), $options, $e);
$result = 0;
}
return $result;
}
/**
* Create (overwrite or append) a file, put the $contents in it, and adjust permissions
*
* This is the same as PHPs `file_put_contents()` except that its preferable to use this in
* ProcessWire because it adjusts the file permissions configured with `$config->chmodFile`.
*
* #pw-group-manipulation
*
* @param string $filename Filename to write to
* @param string|mixed $contents Contents to write to file
* @param int $flags Flags to modify behavior:
* - `FILE_APPEND` (constant): Append to file if it already exists.
* - `LOCK_EX` (constant): Acquire exclusive lock to file while writing.
* @return int|bool Number of bytes written or boolean false on fail
* @throws WireException if given invalid $filename (since 3.0.118)
* @see WireFileTools::fileGetContents()
*
*/
public function filePutContents($filename, $contents, $flags = 0) {
$this->allowPath($filename, false, true);
$result = file_put_contents($filename, $contents, $flags);
if($result === false) {
$this->filesError(__FUNCTION__, "Unable to write: $filename");
} else {
$this->chmod($filename);
}
return $result;
}
/**
* Get contents of file
*
* This is the same as PHPs `file_get_contents()` except that the arguments are simpler and
* it may be preferable to use this in ProcessWire for future cases where the file system may be
* abstracted from the installation.
*
* #pw-group-retrieval
*
* @param string $filename Full path and filename to read
* @param int $offset The offset where the reading starts on the original stream. Negative offsets count from the end of the stream.
* @param int $maxlen Maximum length of data read. The default is to read until end of file is reached.
* @return bool|string Returns the read data (string) or boolean false on failure.
* @since 3.0.167
* @see WireFileTools::filePutContents()
*
*/
public function fileGetContents($filename, $offset = 0, $maxlen = 0) {
if($offset && $maxlen) {
return file_get_contents($filename, false, null, $offset, $maxlen);
} else if($offset) {
return file_get_contents($filename, false, null, $offset);
} else if($maxlen) {
return file_get_contents($filename, false, null, 0, $maxlen);
} else {
return file_get_contents($filename);
}
}
/**
* Get next row from a CSV file
*
* This simplifies the reading of a CSV file by abstracting file-open, get-header, get-rows and file-close
* operations into a single method call, where all those operations are handled internally. All you have to
* do is keep calling the `$files->getCSV($filename)` method until it returns false. This method will also
* skip over blank rows by default, unlike PHPs fgetcsv() which will return a 1-column row with null value.
*
* This method should always be used in a loop, meaning you must keep calling it until it returns false.
* Otherwise a CSV file may be unintentionally left open. If you can't do that then use getAllCSV() instead.
*
* For the method `$options` argument note that the `length`, `separator`, `enclosure` and `escape` options
* all correspond to the identically named PHP [fgetcsv](https://www.php.net/manual/en/function.fgetcsv.php)
* arguments.
*
* Example foods.csv file (first row is header):
* ~~~~~
* Food,Type,Color
* Apple,Fruit,Red
* Banana,Fruit,Yellow
* Spinach,Vegetable,Green
* ~~~~~
* Example of reading the foods.csv file above:
* ~~~~~
* while($row = $files->getCSV('/path/to/foods.csv')) {
* echo "Food: $row[Food] ";
* echo "Type: $row[Type] ";
* echo "Color: $row[Color] ";
* }
* ~~~~~
*
* #pw-group-CSV
*
* @param string $filename CSV filename to read from
* @param array $options
* - `header` (bool|array): Indicate whether CSV has header and how it should be used (default=true):
* True to treat first line as header and return rows as associative arrays indexed by the header values.
* False to indicate there is no header and/or to indicate it should return regular non-associative PHP arrays for rows.
* Array to use it as the header and return rows as associative arrays indexed by your values.
* - `length` (int): Optional. When specified, must be greater than the longest line (in characters) to be found in the CSV file
* (allowing for trailing line-end characters). Otherwise the line is split in chunks of length characters, unless the split
* would occur inside an enclosure. Omitting this parameter (or setting it to 0, or null in PHP 8.0.0 or later) the maximum
* line length is not limited, which is slightly slower. (default=0)
* - `separator` (string): The field separator/delimiter, one single-byte character only. (default=',')
* - `enclosure` (string): The field enclosure character, one single-byte character only. (default='"')
* - `escape` (string): The escape character, at most one single-byte character. An empty string ("") disables the proprietary
* escape mechanism. (default="\\")
* - `blanks` (bool): Allow blank rows? (default=false)
* - `convert` (bool): Convert digit-only strings to integers? (default=false)
* @return array|false Returns array for next row or boolean false when there are no more rows.
* @see https://www.php.net/manual/en/function.fgetcsv.php
* @see getAllCSV()
* @since 3.0.197
*
*/
public function getCSV($filename, array $options = array()) {
$defaults = array(
'header' => true, // or array
'length' => 0,
'separator' => ',',
'enclosure' => '"',
'escape' => "\\",
'convert' => false,
'blanks' => false,
);
$options = array_merge($defaults, $options);
$dataKey = "csv:$filename";
$header = false;
$row = false;
if(isset($this->data[$dataKey])) {
// file is open
$fp = $this->data[$dataKey]['fp'];
$header = $this->data[$dataKey]['header'];
$row = $this->data[$dataKey]['nextRow'];
if($row === false) {
// EOF, close file and return false
fclose($fp);
unset($this->data[$dataKey]);
} else {
$this->data[$dataKey]['nextRow'] = $this->fgetcsv($fp, $options);
}
} else if(($fp = fopen($filename, "r")) !== false) {
// open new file
if($options['header'] === true) {
// get header row and row after it
$header = $this->fgetcsv($fp, $options);
if($header !== false) {
$row = $this->fgetcsv($fp, $options);
foreach($header as $key => $value) {
$header[$key] = trim($value);
}
}
} else {
// get row only
$header = $options['header'];
$row = $this->fgetcsv($fp, $options);
}
if($row === false) {
// file has no rows
fclose($fp);
} else {
// store for next call
$this->data[$dataKey] = array(
'fp' => $fp,
'header' => $header,
'nextRow' => $this->fgetcsv($fp, $options)
);
}
}
if($row === false) return false;
if(empty($options['blanks']) && (empty($row) || (count($row) === 1 && $row[0] === null))) {
// per fgetcsv() does, a blank line in CSV file returns as array with single null field
// rather than accepting that behavior, we just move on to the next non-blank row
return $this->getCSV($filename, $options);
}
if(is_array($header)) {
// index row by header
$a = array();
foreach($header as $key => $name) {
$a[$name] = isset($row[$key]) ? $row[$key] : '';
}
$row = $a;
}
if($options['convert']) {
// convert digit-only strings to integers
foreach($row as $key => $value) {
if(ctype_digit($value)) $row[$key] = (int) $value;
}
}
return $row;
}
/**
* Get all rows from a CSV file
*
* This simplifies the reading of a CSV file by abstracting file-open, get-header, get-rows and file-close
* operations into a single method call, where all those operations are handled internally. All you have to
* do is call the `$files->getAllCSV($filename)` method once and it will return an array of arrays (one per row).
* This method will also skip over blank rows by default, unlike PHPs fgetcsv() which will return a 1-column row
* with null value.
*
* This method is limited by available memory, so when working with potentially large files, you should use the
* `$files->getCSV()` method instead, which reads the CSV file row-by-row rather than all at once.
*
* Note for the method `$options` argument that the `length`, `separator`, `enclosure` and `escape` options
* all correspond to the identically named PHP [fgetcsv](https://www.php.net/manual/en/function.fgetcsv.php)
* arguments.
*
* Example foods.csv file (first row is header):
* ~~~~~
* Food,Type,Color
* Apple,Fruit,Red
* Banana,Fruit,Yellow
* Spinach,Vegetable,Green
* ~~~~~
* Example of reading the foods.csv file above:
* ~~~~~
* $rows = $files->getAllCSV('/path/to/foods.csv');
* foreach($rows as $row) {
* echo "Food: $row[Food] ";
* echo "Type: $row[Type] ";
* echo "Color: $row[Color] ";
* }
* ~~~~~
*
* #pw-group-CSV
*
* @param string $filename CSV filename to read from
* @param array $options
* - `header` (bool|array): Indicate whether CSV has header and how it should be used (default=true):
* True to treat first line as header and return rows as associative arrays indexed by the header values.
* False to indicate there is no header and/or to indicate it should return regular non-associative PHP arrays for rows.
* Array to use it as the header and return rows as associative arrays indexed by your values.
* - `length` (int): Optional. When specified, must be greater than the longest line (in characters) to be found in the CSV file
* (allowing for trailing line-end characters). Otherwise the line is split in chunks of length characters, unless the split
* would occur inside an enclosure. Omitting this parameter (or setting it to 0, or null in PHP 8.0.0 or later) the maximum
* line length is not limited, which is slightly slower. (default=0)
* - `separator` (string): The field separator/delimiter, one single-byte character only. (default=',')
* - `enclosure` (string): The field enclosure character, one single-byte character only. (default='"')
* - `escape` (string): The escape character, at most one single-byte character. An empty string ("") disables the proprietary
* escape mechanism. (default="\\")
* - `convert` (bool): Convert digit-only strings to integers? (default=false)
* - `blanks` (bool): Allow blank rows? (default=false)
* @return array
* @see https://www.php.net/manual/en/function.fgetcsv.php
* @see getCSV()
* @since 3.0.197
*
*/
public function getAllCSV($filename, array $options = array()) {
$rows = array();
while(false !== ($row = $this->getCSV($filename, $options))) {
$rows[] = $row;
}
return $rows;
}
/**
* PHPs fgetcsv function in an internal options method
*
* #pw-internal
*
* @param $fp
* @param $options
* @return array|false
* @since 3.0.197
*
*/
protected function fgetcsv($fp, $options) {
$defaults = array(
'length' => 0,
'separator' => ',',
'enclosure' => '"',
'escape' => "\\",
);
$options = array_merge($defaults, $options);
return fgetcsv($fp, $options['length'], $options['separator'], $options['enclosure'], $options['escape']);
}
/**
* Given a filename, render it as a ProcessWire template file
*
* This is a shortcut to using the TemplateFile class.
*
* File is assumed relative to `/site/templates/` (or a directory within there) unless you specify a full path.
* If you specify a full path, it will accept files in or below any of the following:
*
* - /site/templates/
* - /site/modules/
* - /wire/modules/
*
* Note this function returns the output to you, so that you can send the output wherever you want (delayed output).
* For direct output, use the `$files->include()` function instead.
*
* #pw-group-includes
*
* @param string $filename Assumed relative to /site/templates/ unless you provide a full path name with the filename.
* If you provide a path, it must resolve somewhere in site/templates/, site/modules/ or wire/modules/.
* @param array $vars Optional associative array of variables to send to template file.
* Please note that all template files automatically receive all API variables already (you don't have to provide them).
* @param array $options Associative array of options to modify behavior:
* - `defaultPath` (string): Path where files are assumed to be when only filename or relative filename is specified (default=/site/templates/)
* - `autoExtension` (string): Extension to assume when no ext in filename, make blank for no auto assumption (default=php)
* - `allowedPaths` (array): Array of paths that are allowed (default is templates, core modules and site modules)
* - `allowDotDot` (bool): Allow use of ".." in paths? (default=false)
* - `throwExceptions` (bool): Throw exceptions when fatal error occurs? (default=true)
* - `cache` (int|string|Page): Specify non-zero value to cache rendered result for this many seconds, or see the `WireCache::renderFile()`
* method `$expire` argument for more options you can specify here. (default=0, no cache) *Note: this option added in 3.0.130*
* @return string|bool Rendered template file or boolean false on fatal error (and throwExceptions disabled)
* @throws WireException if template file doesn't exist
* @see WireFileTools::include()
*
*/
public function render($filename, array $vars = array(), array $options = array()) {
$paths = $this->wire()->config->paths;
$defaults = array(
'defaultPath' => $paths->templates,
'autoExtension' => 'php',
'allowedPaths' => array(
$paths->templates,
$paths->adminTemplates,
$paths->modules,
$paths->siteModules,
$paths->cache
),
'allowDotDot' => false,
'throwExceptions' => true,
'cache' => 0,
);
$options = array_merge($defaults, $options);
$filename = $this->unixFileName($filename);
// add .php extension if filename doesn't already have an extension
if($options['autoExtension'] && !strrpos(basename($filename), '.')) {
$filename .= "." . $options['autoExtension'];
}
if(!$options['allowDotDot'] && strpos($filename, '..')) {
// make path relative to /site/templates/ if filename is not an absolute path
$error = 'Filename may not have ".."';
if($options['throwExceptions']) $this->filesException(__FUNCTION__, $error);
$this->error($error);
return false;
}
if($options['defaultPath'] && strpos($filename, './') === 0) {
$filename = rtrim($options['defaultPath'], '/') . '/' . substr($filename, 2);
} else if($options['defaultPath'] && strpos($filename, '/') !== 0 && strpos($filename, ':') !== 1) {
// filename is relative to defaultPath (typically /site/templates/)
$filename = rtrim($options['defaultPath'], '/') . '/' . $filename;
} else if(strpos($filename, '/') !== false) {
// filename is absolute, make sure it's in a location we consider safe
$allowed = false;
foreach($options['allowedPaths'] as $path) {
if(strpos($filename, $path) === 0) {
$allowed = true;
break;
}
}
if(!$allowed) {
$error = "Filename $filename is not in an allowed path." ;
$error .= ' Paths: ' . implode("\n", $options['allowedPaths']);
if($options['throwExceptions']) $this->filesException(__FUNCTION__, $error);
$this->error($error);
return false;
}
}
if($options['cache']) {
$cache = $this->wire()->cache;
$o = $options;
unset($o['cache']);
$o['vars'] = $vars;
return $cache->renderFile($filename, $options['cache'], $o);
}
// render file and return output
$t = $this->wire(new TemplateFile()); /** @var TemplateFile $t */
$t->setThrowExceptions($options['throwExceptions']);
$t->setFilename($filename);
foreach($vars as $key => $value) {
$t->data($key, $value);
}
return $t->render();
}
/**
* Include a PHP file passing it all API variables and optionally your own specified variables
*
* This is the same as PHPs `include()` function except for the following:
*
* - It receives all API variables and optionally your custom variables
* - If your filename is not absolute, it doesnt look in PHPs include path, only in the current dir.
* - It only allows including files that are part of the PW installation: templates, core modules or site modules
* - It will assume a .php” extension if filename has no extension.
*
* Note this function produces direct output. To retrieve output as a return value, use the
* `$files->render()` function instead.
*
* #pw-group-includes
*
* @param string $filename Filename to include
* @param array $vars Optional variables you want to hand to the include (associative array)
* @param array $options Array of options to modify behavior:
* - `func` (string): Function to use: include, include_once, require or require_once (default=include)
* - `autoExtension` (string): Extension to assume when no ext in filename, make blank for no auto assumption (default=php)
* - `allowedPaths` (array): Array of start paths include files are allowed from. Note current dir is always allowed.
* @return bool Always returns true
* @throws WireException if file doesnt exist or is not allowed
*
*/
public function ___include($filename, array $vars = array(), array $options = array()) {
$paths = $this->wire()->config->paths;
$defaults = array(
'func' => 'include',
'autoExtension' => 'php',
'allowedPaths' => array(
$paths->templates,
$paths->adminTemplates,
$paths->modules,
$paths->siteModules,
$paths->cache
)
);
// need to use different name than $options
$options = array_merge($defaults, $options);
$filename = trim($filename);
// add .php extension if filename doesn't already have an extension
if($options['autoExtension'] && !strrpos(basename($filename), '.')) {
$filename .= '.' . $options['autoExtension'];
}
if(strpos($filename, '..') !== false) {
// if backtrack/relative components, convert to real path
$_filename = $filename;
$filename = realpath($filename);
if($filename === false) $this->filesException(__FUNCTION__, "File does not exist: $_filename");
}
$filename = $this->unixFileName($filename);
if(strpos($filename, '//') !== false) {
$this->filesException(__FUNCTION__, "File is not allowed (double-slash): $filename");
}
if(strpos($filename, './') !== 0) {
// file does not specify "current directory"
$slashPos = strpos($filename, '/');
// If no absolute path specified, ensure it only looks in current directory
if($slashPos !== 0 && strpos($filename, ':/') === false) $filename = "./$filename";
}
if(strpos($filename, '/') === 0 || strpos($filename, ':/') !== false) {
// absolute path, make sure it's part of PW's installation
$allowed = false;
foreach($options['allowedPaths'] as $path) {
if($this->fileInPath($filename, $path)) $allowed = true;
if($allowed) break;
}
if(!$allowed) $this->filesException(__FUNCTION__, "File is not in an allowed path: $filename");
}
if(!file_exists($filename)) $this->filesException(__FUNCTION__, "File does not exist: $filename");
// remember options[func] and $filename outside this method because of the extract() which can overwrite
$this->includeFunc = $options['func'];
$this->includeFile = $filename;
// extract all API vars
$fuel = array_merge($this->wire()->fuel->getArray(), $vars);
extract($fuel);
// include the file
TemplateFile::pushRenderStack($this->includeFile);
if($this->includeFunc === 'require') {
require($this->includeFile);
} else if($this->includeFunc === 'require_once') {
require_once($this->includeFile);
} else if($this->includeFunc === 'include_once') {
include_once($this->includeFile);
} else {
include($this->includeFile);
}
TemplateFile::popRenderStack();
return true;
}
protected $includeFunc = '';
protected $includeFile = '';
/**
* Same as include() method except that file will not be executed if it as previously been included
*
* See the `WireFileTools::include()` method for details, arguments and options.
*
* #pw-group-includes
*
* @param string $filename
* @param array $vars
* @param array $options
* @return bool
* @see WireFileTools::include()
* @since 3.0.96
*
*/
public function includeOnce($filename, array $vars = array(), array $options = array()) {
$options['func'] = 'include_once';
return $this->include($filename, $vars, $options);
}
/**
* Get the namespace used in the given .php or .module file
*
* #pw-advanced
*
* @param string $file File name or file data (if file data, specify true for 2nd argument)
* @param bool $fileIsContents Specify true if the given $file is actually the contents of the file, rather than file name.
* @return string Actual found namespace or "\" (root namespace) if none found
*
*/
public function getNamespace($file, $fileIsContents = false) {
$namespace = "\\"; // root namespace, if no namespace found
if($fileIsContents) {
$data = trim($file);
} else {
$data = file_get_contents($file);
if($data === false) return $namespace;
$data = trim($data);
}
// if there's no "namespace" keyword in the file, it's not declaring one
$namespacePos = strpos($data, 'namespace');
if($namespacePos === false) return $namespace;
// quick optimization for common ProcessWire namespace usage
if(strpos($data, '<' . '?php namespace ProcessWire;') === 0) return 'ProcessWire';
// if file doesn't start with an opening PHP tag, then it's not going to have a namespace declaration
$phpOpen = strpos($data, '<' . '?');
if($phpOpen !== 0) {
// file does not begin with opening php tag
// note: this fails for PHP files executable on their own (like shell scripts)
return $namespace;
}
// get everything that appears before "namespace" keyword
$head = substr($data, 0, $namespacePos);
$headPrev = $head;
// declare(...); is the one statement allowed to appear before namespace in PHP files
if(strpos($head, 'declare')) {
$head = preg_replace('/declare[ ]*\(.+?\)[ ]*;\s*/s', '', $head);
}
// single line comment(s) appear before namespace
if(strpos($head, '//') !== false) {
$head = preg_replace('!//.*!', '', $head);
}
// single or multi-line comments before namespace
if(strpos($head, '/' . '*') !== false) {
$head = preg_replace('!/\*.*\*/!s', '', $head);
}
// replace cleaned up head in data
if($head !== $headPrev) {
$data = str_replace($headPrev, $head, $data);
}
$namespacePos = strpos($data, 'namespace'); // get fresh position
if($namespacePos === false) return $namespace; // was likely in a comment
$test = substr($data, 0, $namespacePos-1);
$test = trim(str_replace(array('<' . '?php', '<' . '?', "\n", "\r", "\t", " "), "", $test));
if(!strlen($test)) {
// namespace declaration is the first thing in the file (other than php tag and whitespace)
$namespacePos += 10; // skip over "namespace" word
$semiPos = strpos($data, ';');
if($semiPos > $namespacePos) {
$test = substr($data, 0, $semiPos);
$namespace = substr($test, $namespacePos);
return trim($namespace, "; ");
}
}
/* for reference, the above (hopefully faster) replaces this regex
if(preg_match('/^<\?[ph]*[\s\r\n]+namespace\s+([^;]+);/s', $data, $matches)) {
return trim($matches[1]);
}
*/
// remove anything after a closing php tag
$phpEnd = strpos($data, '?' . '>');
if($phpEnd !== false) $data = substr($data, 0, $phpEnd);
// if there's no 'namespace' word present in the data, nothing is declared
if(strpos($data, 'namespace') === false) return $namespace;
// normalize line endings
if(strpos($data, "\r") !== false) $data = str_replace("\r", "\n", $data);
while(preg_match('/(^.*[\s\r\n]+)namespace\s+([_a-zA-Z0-9\\\\]+);\s*$/m', $data, $matches)) {
// $open is everything that comes before the namespace line
$open = $matches[1];
// potential namespace, if our checks succeed
$_namespace = trim($matches[2]);
// $line is everything preceding the 'namespace' declaration, on the same line as the declaration
$lastNewlinePos = strrpos($open, "\n");
if($lastNewlinePos !== false) {
$line = substr($open, $lastNewlinePos);
} else {
$line = $open;
}
// determine if line is commented
$hasComment = strpos($line, '//') !== false;
if(!$hasComment) {
// determine if namespace declaration is in a comment block
$startCommentPos = strrpos($open, '/*');
$closeCommentPos = strrpos($open, '*/');
if($startCommentPos !== false && ((int) $closeCommentPos) < $startCommentPos) $hasComment = true;
}
if(!$hasComment) {
// if we've reached this point, we have found a valid namespace
$namespace = $_namespace;
break;
}
// reduce $data for next preg_match
$data = str_replace($matches[0], '', $data);
}
return $namespace;
}
/**
* Compile the given file using ProcessWires file compiler
*
* #pw-internal
*
* @param string $file File to compile
* @param array $options Optional associative array of the following:
* - `includes` (bool): Also compile files include()'d from the given $file? (default=true)
* - `namespace` (bool): Compile to make compatible with ProcessWire namespace? (default=true)
* - `modules` (bool): Allow FileCompilerModule module's to process the file as well? (default=false)
* - `skipIfNamespace` (bool): Return source $file if it declares a namespace (default=false)
* @return string Full path and filename of compiled file, or returns original `$file` if compilation is not necessary.
* @throws WireException if given invalid $file or other fatal error
*
*/
public function compile($file, array $options = array()) {
static $compiled = array();
if(strpos($file, '/modules/')) {
// for multi-instance support, use the same compiled version
// otherwise, require_once() statements in a file may not work as intended
// applied just to site/modules for the moment, but may need to do site/templates too
$f = str_replace($this->wire()->config->paths->root, '', $file);
if(isset($compiled[$f])) return $compiled[$f];
} else {
$f = '';
}
/** @var FileCompiler $compiler */
$compiler = $this->wire(new FileCompiler(dirname($file), $options));
$compiledFile = $compiler->compile(basename($file));
if($f) $compiled[$f] = $compiledFile;
return $compiledFile;
}
/**
* Compile and include() the given file
*
* #pw-internal
*
* @param string $file File to compile and include
* @param array $options Optional associative array of the following:
* - `includes` (bool): Also compile files include()'d from the given $file? (default=true)
* - `namespace` (bool): Compile to make compatible with ProcessWire namespace? (default=true)
* - `modules` (bool): Allow FileCompilerModule module's to process the file as well? (default=false)
* - `skipIfNamespace` (bool): Return source $file if it declares a namespace (default=false)
* @throws WireException if given invalid $file or other fatal error
*
*/
public function compileInclude($file, array $options = array()) {
$file = $this->compile($file, $options);
TemplateFile::pushRenderStack($file);
include($file);
TemplateFile::popRenderStack();
}
/**
* Compile and include_once() the given file
*
* #pw-internal
*
* @param string $file File to compile and include
* @param array $options Optional associative array of the following:
* - `includes` (bool): Also compile files include()'d from the given $file? (default=true)
* - `namespace` (bool): Compile to make compatible with ProcessWire namespace? (default=true)
* - `modules` (bool): Allow FileCompilerModule module's to process the file as well? (default=false)
* - `skipIfNamespace` (bool): Return source $file if it declares a namespace (default=false)
* @throws WireException if given invalid $file or other fatal error
*
*/
public function compileIncludeOnce($file, array $options = array()) {
$file = $this->compile($file, $options);
TemplateFile::pushRenderStack($file);
include_once($file);
TemplateFile::popRenderStack();
}
/**
* Compile and require() the given file
*
* #pw-internal
*
* @param string $file File to compile and include
* @param array $options Optional associative array of the following:
* - `includes` (bool): Also compile files include()'d from the given $file? (default=true)
* - `namespace` (bool): Compile to make compatible with ProcessWire namespace? (default=true)
* - `modules` (bool): Allow FileCompilerModule module's to process the file as well? (default=false)
* - `skipIfNamespace` (bool): Return source $file if it declares a namespace (default=false)
* @throws WireException if given invalid $file or other fatal error
*
*/
public function compileRequire($file, array $options = array()) {
$file = $this->compile($file, $options);
TemplateFile::pushRenderStack($file);
require($file);
TemplateFile::popRenderStack();
}
/**
* Compile and require_once() the given file
*
* #pw-internal
*
* @param string $file File to compile and include
* @param array $options Optional associative array of the following:
* - `includes` (bool): Also compile files include()'d from the given $file? (default=true)
* - `namespace` (bool): Compile to make compatible with ProcessWire namespace? (default=true)
* - `modules` (bool): Allow FileCompilerModule module's to process the file as well? (default=false)
* - `skipIfNamespace` (bool): Return source $file if it declares a namespace (default=false)
* @throws WireException if given invalid $file or other fatal error
*
*/
public function compileRequireOnce($file, array $options = array()) {
TemplateFile::pushRenderStack($file);
$file = $this->compile($file, $options);
require_once($file);
TemplateFile::popRenderStack();
}
/**
* Convert given directory name to use unix slashes and enforce trailing or no-trailing slash
*
* #pw-group-filenames
*
* @param string $dir Directory name to adust (if it needs it)
* @param bool $trailingSlash True to force trailing slash, false to force no trailing slash (default=true)
* @return string Adjusted directory name
*
*/
public function unixDirName($dir, $trailingSlash = true) {
if(DIRECTORY_SEPARATOR != '/' && strpos($dir, DIRECTORY_SEPARATOR) !== false) {
$dir = str_replace(DIRECTORY_SEPARATOR, '/', $dir);
}
$dir = rtrim($dir, '/');
if($trailingSlash) $dir .= '/';
return $dir;
}
/**
* Convert given file name to use unix slashes (if it isnt already)
*
* #pw-group-filenames
*
* @param string $file File name to adjust (if it needs it)
* @return string Adjusted file name
*
*/
public function unixFileName($file) {
return $this->unixDirName($file, false);
}
/**
* Is given $file name in given $path name? (aka: is $file a subdirectory somewhere within $path)
*
* This is purely for string comparison purposes, it does not check if file/path actually exists.
* Note that if $file and $path are identical, this method returns false.
*
* #pw-group-filenames
*
* @param string $file May be a file or a directory
* @param string $path
* @return bool
*
*/
public function fileInPath($file, $path) {
$file = $this->unixDirName($file); // use of unixDirName rather than unixFileName intentional
$path = $this->unixDirName($path);
if($file === $path || strlen($file) <= strlen($path)) return false;
return strpos($file, $path) === 0;
}
/**
* Get the current path / work directory
*
* This is like PHPs getcwd() function except that is in ProcessWire format as unix path with trailing slash.
*
* #pw-group-filenames
*
* @return string
* @since 3.0.130
*
*/
public function currentPath() {
return $this->unixDirName(getcwd());
}
/**
* Report/log/throw an error
*
* #pw-internal
*
* @param string $method
* @param string $msg
* @param bool|array $throw Throw exception? May be boolean or array with 'throw' index containing boolean.
* @param \Exception|null $e Previous exception, if applicable
* @return bool Always returns boolean false (so it can be used in error return statements)
* @throws WireFilesException
* @since 3.0.178
*
*/
public function filesError($method, $msg, $throw = false, $e = null) {
if(is_array($throw)) $throw = isset($throw['throw']) ? $throw['throw'] : false;
$msg = "$method: $msg";
$this->log($msg, array('name' => 'files-errors'));
if($throw) {
if($e) throw new WireFilesException($msg, $e->getCode(), $e);
throw new WireFilesException($msg);
} else if($this->wire()->config->debug) {
$this->warning($msg, Notice::debug);
}
return false;
}
/**
* Throw a files exception
*
* #pw-internal
*
* @param string $method
* @param string $msg
* @param \Exception|null $e
* @throws WireFilesException
* @since 3.0.178
*
*/
public function filesException($method, $msg, $e = null) {
$this->filesError($method, $msg, true, $e);
}
/**
* Log a message for this class
*
* #pw-internal
*
* @param string $str Text to log, or omit to return the `$log` API variable.
* @param array $options Optional extras to include, see Wire::___log()
* @return WireLog
*
*/
public function ___log($str = '', array $options = array()) {
if(empty($options['name'])) $options['name'] = 'files';
return parent::___log($str, $options);
}
}