forked from extern/egroupware
Vfs no longer extends Vfs\StreamWrapper, as this is the only way to keep static function signatures for Vfs class, also cleans up a bit
This commit is contained in:
parent
3845467b73
commit
112cc725a2
471
api/src/Vfs.php
471
api/src/Vfs.php
@ -67,9 +67,17 @@ use HTTP_WebDAV_Server;
|
|||||||
* Vfs::parse_url($url, $component=-1), Vfs::dirname($url) and Vfs::basename($url) work
|
* Vfs::parse_url($url, $component=-1), Vfs::dirname($url) and Vfs::basename($url) work
|
||||||
* on urls containing utf-8 characters, which get NOT urlencoded in our VFS!
|
* on urls containing utf-8 characters, which get NOT urlencoded in our VFS!
|
||||||
*/
|
*/
|
||||||
class Vfs extends Vfs\StreamWrapper
|
class Vfs
|
||||||
{
|
{
|
||||||
const PREFIX = 'vfs://default';
|
const PREFIX = 'vfs://default';
|
||||||
|
/**
|
||||||
|
* Scheme / protocol used for this stream-wrapper
|
||||||
|
*/
|
||||||
|
const SCHEME = Vfs\StreamWrapper::SCHEME;
|
||||||
|
/**
|
||||||
|
* Mime type of directories, the old vfs used 'Directory', while eg. WebDAV uses 'httpd/unix-directory'
|
||||||
|
*/
|
||||||
|
const DIR_MIME_TYPE = Vfs\StreamWrapper::DIR_MIME_TYPE;
|
||||||
/**
|
/**
|
||||||
* Readable bit, for dirs traversable
|
* Readable bit, for dirs traversable
|
||||||
*/
|
*/
|
||||||
@ -238,7 +246,7 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
*/
|
*/
|
||||||
static function stat($path,$try_create_home=false)
|
static function stat($path,$try_create_home=false)
|
||||||
{
|
{
|
||||||
if ($path[0] != '/')
|
if ($path[0] != '/' && strpos($path, self::PREFIX.'/') !== 0)
|
||||||
{
|
{
|
||||||
throw new Exception\AssertionFailed("File '$path' is not an absolute path!");
|
throw new Exception\AssertionFailed("File '$path' is not an absolute path!");
|
||||||
}
|
}
|
||||||
@ -319,69 +327,7 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
*/
|
*/
|
||||||
static function mount($url=null,$path=null,$check_url=null,$persitent_mount=true,$clear_fstab=false)
|
static function mount($url=null,$path=null,$check_url=null,$persitent_mount=true,$clear_fstab=false)
|
||||||
{
|
{
|
||||||
if (is_null($check_url)) $check_url = strpos($url,'$') === false;
|
return Vfs\StreamWrapper::mount($url, $path, $check_url, $persitent_mount, $clear_fstab);
|
||||||
|
|
||||||
if (!isset($GLOBALS['egw_info']['server']['vfs_fstab'])) // happens eg. in setup
|
|
||||||
{
|
|
||||||
$api_config = Config::read('phpgwapi');
|
|
||||||
if (isset($api_config['vfs_fstab']) && is_array($api_config['vfs_fstab']))
|
|
||||||
{
|
|
||||||
self::$fstab = $api_config['vfs_fstab'];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
self::$fstab = array(
|
|
||||||
'/' => 'sqlfs://$host/',
|
|
||||||
'/apps' => 'links://$host/apps',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
unset($api_config);
|
|
||||||
}
|
|
||||||
if (is_null($url) || is_null($path))
|
|
||||||
{
|
|
||||||
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') returns '.array2string(self::$fstab));
|
|
||||||
return self::$fstab;
|
|
||||||
}
|
|
||||||
if (!self::$is_root)
|
|
||||||
{
|
|
||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') permission denied, you are NOT root!');
|
|
||||||
return false; // only root can mount
|
|
||||||
}
|
|
||||||
if ($clear_fstab)
|
|
||||||
{
|
|
||||||
self::$fstab = array();
|
|
||||||
}
|
|
||||||
if (isset(self::$fstab[$path]) && self::$fstab[$path] === $url)
|
|
||||||
{
|
|
||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') already mounted.');
|
|
||||||
return true; // already mounted
|
|
||||||
}
|
|
||||||
self::load_wrapper(self::parse_url($url,PHP_URL_SCHEME));
|
|
||||||
|
|
||||||
if ($check_url && (!file_exists($url) || opendir($url) === false))
|
|
||||||
{
|
|
||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') url does NOT exist!');
|
|
||||||
return false; // url does not exist
|
|
||||||
}
|
|
||||||
self::$fstab[$path] = $url;
|
|
||||||
|
|
||||||
uksort(self::$fstab, function($a, $b)
|
|
||||||
{
|
|
||||||
return strlen($a) - strlen($b);
|
|
||||||
});
|
|
||||||
|
|
||||||
if ($persitent_mount)
|
|
||||||
{
|
|
||||||
Config::save_value('vfs_fstab',self::$fstab,'phpgwapi');
|
|
||||||
$GLOBALS['egw_info']['server']['vfs_fstab'] = self::$fstab;
|
|
||||||
// invalidate session cache
|
|
||||||
if (method_exists($GLOBALS['egw'],'invalidate_session_cache')) // egw object in setup is limited
|
|
||||||
{
|
|
||||||
$GLOBALS['egw']->invalidate_session_cache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') returns true (successful new mount).');
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -391,27 +337,18 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
*/
|
*/
|
||||||
static function umount($path)
|
static function umount($path)
|
||||||
{
|
{
|
||||||
if (!self::$is_root)
|
return Vfs\StreamWrapper::umount($path);
|
||||||
{
|
}
|
||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($path).','.array2string($path).') permission denied, you are NOT root!');
|
|
||||||
return false; // only root can mount
|
|
||||||
}
|
|
||||||
if (!isset(self::$fstab[$path]) && ($path = array_search($path,self::$fstab)) === false)
|
|
||||||
{
|
|
||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($path).') NOT mounted!');
|
|
||||||
return false; // $path not mounted
|
|
||||||
}
|
|
||||||
unset(self::$fstab[$path]);
|
|
||||||
|
|
||||||
Config::save_value('vfs_fstab',self::$fstab,'phpgwapi');
|
/**
|
||||||
$GLOBALS['egw_info']['server']['vfs_fstab'] = self::$fstab;
|
* Returns mount url of a full url returned by resolve_url
|
||||||
// invalidate session cache
|
*
|
||||||
if (method_exists($GLOBALS['egw'],'invalidate_session_cache')) // egw object in setup is limited
|
* @param string $fullurl full url returned by resolve_url
|
||||||
{
|
* @return string|NULL mount url or null if not found
|
||||||
$GLOBALS['egw']->invalidate_session_cache();
|
*/
|
||||||
}
|
static function mount_url($fullurl)
|
||||||
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($path).') returns true (successful unmount).');
|
{
|
||||||
return true;
|
return Vfs\StreamWrapper::mount_url($fullurl);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -666,12 +603,12 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
|
|
||||||
if ($options['url'])
|
if ($options['url'])
|
||||||
{
|
{
|
||||||
$stat = @lstat($path);
|
$lstat = @lstat($path);
|
||||||
|
$stat = array_slice($lstat,13); // remove numerical indices 0-12
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
$vfs = new Vfs\StreamWrapper();
|
$stat = self::stat($path, STREAM_URL_STAT_LINK);
|
||||||
$stat = $vfs->url_stat($path,STREAM_URL_STAT_LINK);
|
|
||||||
}
|
}
|
||||||
if (!$stat)
|
if (!$stat)
|
||||||
{
|
{
|
||||||
@ -682,7 +619,6 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
{
|
{
|
||||||
return; // wrong type
|
return; // wrong type
|
||||||
}
|
}
|
||||||
$stat = array_slice($stat,13); // remove numerical indices 0-12
|
|
||||||
$stat['path'] = self::parse_url($path,PHP_URL_PATH);
|
$stat['path'] = self::parse_url($path,PHP_URL_PATH);
|
||||||
$stat['name'] = $options['remove'] > 0 ? implode('/',array_slice(explode('/',$stat['path']),$options['remove'])) : self::basename($path);
|
$stat['name'] = $options['remove'] > 0 ? implode('/',array_slice(explode('/',$stat['path']),$options['remove'])) : self::basename($path);
|
||||||
|
|
||||||
@ -1298,6 +1234,59 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
return array_pop($parts);
|
return array_pop($parts);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utf-8 save version of parse_url
|
||||||
|
*
|
||||||
|
* Does caching withing request, to not have to parse urls over and over again.
|
||||||
|
*
|
||||||
|
* @param string $url
|
||||||
|
* @param int $component =-1 PHP_URL_* constants
|
||||||
|
* @return array|string|boolean on success array or string, if $component given, or false on failure
|
||||||
|
*/
|
||||||
|
static function parse_url($url, $component=-1)
|
||||||
|
{
|
||||||
|
static $component2str = array(
|
||||||
|
PHP_URL_SCHEME => 'scheme',
|
||||||
|
PHP_URL_HOST => 'host',
|
||||||
|
PHP_URL_PORT => 'port',
|
||||||
|
PHP_URL_USER => 'user',
|
||||||
|
PHP_URL_PASS => 'pass',
|
||||||
|
PHP_URL_PATH => 'path',
|
||||||
|
PHP_URL_QUERY => 'query',
|
||||||
|
PHP_URL_FRAGMENT => 'fragment',
|
||||||
|
);
|
||||||
|
static $cache = array(); // some caching
|
||||||
|
|
||||||
|
$result =& $cache[$url];
|
||||||
|
|
||||||
|
if (!isset($result))
|
||||||
|
{
|
||||||
|
// Build arrays of values we need to decode before parsing
|
||||||
|
static $entities = array('%21', '%2A', '%27', '%28', '%29', '%3B', '%3A', '%40', '%26', '%3D', '%24', '%2C', '%2F', '%3F', '%23', '%5B', '%5D');
|
||||||
|
static $replacements = array('!', '*', "'", "(", ")", ";", ":", "@", "&", "=", "$", ",", "/", "?", "#", "[", "]");
|
||||||
|
static $str_replace = null;
|
||||||
|
if (!isset($str_replace)) $str_replace = function_exists('mb_str_replace') ? 'mb_str_replace' : 'str_replace';
|
||||||
|
|
||||||
|
// Create encoded URL with special URL characters decoded so it can be parsed
|
||||||
|
// All other characters will be encoded
|
||||||
|
$encodedURL = $str_replace($entities, $replacements, urlencode($url));
|
||||||
|
|
||||||
|
// Parse the encoded URL
|
||||||
|
$result = $encodedParts = parse_url($encodedURL);
|
||||||
|
|
||||||
|
// Now, decode each value of the resulting array
|
||||||
|
if ($encodedParts)
|
||||||
|
{
|
||||||
|
$result = array();
|
||||||
|
foreach ($encodedParts as $key => $value)
|
||||||
|
{
|
||||||
|
$result[$key] = urldecode($str_replace($replacements, $entities, $value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $component >= 0 ? $result[$component2str[$component]] : $result;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the directory / parent of a given path or url(!), return false for '/'!
|
* Get the directory / parent of a given path or url(!), return false for '/'!
|
||||||
*
|
*
|
||||||
@ -2140,6 +2129,310 @@ class Vfs extends Vfs\StreamWrapper
|
|||||||
//error_log(__METHOD__."($file1, $file2) returning ".array2string($read1 === $read2)." (content differs)");
|
//error_log(__METHOD__."($file1, $file2) returning ".array2string($read1 === $read2)." (content differs)");
|
||||||
return $read1 === $read2;
|
return $read1 === $read2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve the given path according to our fstab AND symlinks
|
||||||
|
*
|
||||||
|
* @param string $_path
|
||||||
|
* @param boolean $file_exists =true true if file needs to exists, false if not
|
||||||
|
* @param boolean $resolve_last_symlink =true
|
||||||
|
* @param array|boolean &$stat=null on return: stat of existing file or false for non-existing files
|
||||||
|
* @return string|boolean false if the url cant be resolved, should not happen if fstab has a root entry
|
||||||
|
*/
|
||||||
|
static function resolve_url_symlinks($_path,$file_exists=true,$resolve_last_symlink=true,&$stat=null)
|
||||||
|
{
|
||||||
|
$vfs = new Vfs\StreamWrapper();
|
||||||
|
return $vfs->resolve_url_symlinks($_path, $file_exists, $resolve_last_symlink, $stat);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is called in response to mkdir() calls on URL paths associated with the wrapper.
|
||||||
|
*
|
||||||
|
* It should attempt to create the directory specified by path.
|
||||||
|
* In order for the appropriate error message to be returned, do not define this method if your wrapper does not support creating directories.
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param int $mode
|
||||||
|
* @param int $options Posible values include STREAM_REPORT_ERRORS and STREAM_MKDIR_RECURSIVE
|
||||||
|
* @return boolean TRUE on success or FALSE on failure
|
||||||
|
*/
|
||||||
|
static function mkdir ($path, $mode=0750, $options=0)
|
||||||
|
{
|
||||||
|
return $path[0] == '/' && mkdir(self::PREFIX.$path, $mode, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is called in response to rmdir() calls on URL paths associated with the wrapper.
|
||||||
|
*
|
||||||
|
* It should attempt to remove the directory specified by path.
|
||||||
|
* In order for the appropriate error message to be returned, do not define this method if your wrapper does not support removing directories.
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param int $options Possible values include STREAM_REPORT_ERRORS.
|
||||||
|
* @return boolean TRUE on success or FALSE on failure.
|
||||||
|
*/
|
||||||
|
static function rmdir ( $path, $options=0 )
|
||||||
|
{
|
||||||
|
return $path[0] == '/' && rmdir(self::PREFIX.$path, $options);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is called in response to unlink() calls on URL paths associated with the wrapper.
|
||||||
|
*
|
||||||
|
* It should attempt to delete the item specified by path.
|
||||||
|
* In order for the appropriate error message to be returned, do not define this method if your wrapper does not support unlinking!
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @return boolean TRUE on success or FALSE on failure
|
||||||
|
*/
|
||||||
|
static function unlink ( $path )
|
||||||
|
{
|
||||||
|
return $path[0] == '/' && unlink(self::PREFIX.$path);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allow to call methods of the underlying stream wrapper: touch, chmod, chgrp, chown, ...
|
||||||
|
*
|
||||||
|
* We cant use a magic __call() method, as it does not work for static methods!
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @param array $params first param has to be the path, otherwise we can not determine the correct wrapper
|
||||||
|
* @param boolean $fail_silent =false should only false be returned if function is not supported by the backend,
|
||||||
|
* or should an E_USER_WARNING error be triggered (default)
|
||||||
|
* @param int $path_param_key =0 key in params containing the path, default 0
|
||||||
|
* @return mixed return value of backend or false if function does not exist on backend
|
||||||
|
*/
|
||||||
|
static protected function _call_on_backend($name,$params,$fail_silent=false,$path_param_key=0)
|
||||||
|
{
|
||||||
|
$pathes = $params[$path_param_key];
|
||||||
|
|
||||||
|
$scheme2urls = array();
|
||||||
|
foreach(is_array($pathes) ? $pathes : array($pathes) as $path)
|
||||||
|
{
|
||||||
|
if (!($url = self::resolve_url_symlinks($path,false,false)))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$k=(string)self::parse_url($url,PHP_URL_SCHEME);
|
||||||
|
if (!(is_array($scheme2urls[$k]))) $scheme2urls[$k] = array();
|
||||||
|
$scheme2urls[$k][$path] = $url;
|
||||||
|
}
|
||||||
|
$ret = array();
|
||||||
|
foreach($scheme2urls as $scheme => $urls)
|
||||||
|
{
|
||||||
|
if ($scheme)
|
||||||
|
{
|
||||||
|
if (!class_exists($class = self::scheme2class($scheme)) || !method_exists($class,$name))
|
||||||
|
{
|
||||||
|
if (!$fail_silent) trigger_error("Can't $name for scheme $scheme!\n",E_USER_WARNING);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!is_array($pathes))
|
||||||
|
{
|
||||||
|
$params[$path_param_key] = $url;
|
||||||
|
|
||||||
|
return call_user_func_array(array($class,$name),$params);
|
||||||
|
}
|
||||||
|
$params[$path_param_key] = $urls;
|
||||||
|
if (!is_array($r = call_user_func_array(array($class,$name),$params)))
|
||||||
|
{
|
||||||
|
return $r;
|
||||||
|
}
|
||||||
|
// we need to re-translate the urls to pathes, as they can eg. contain symlinks
|
||||||
|
foreach($urls as $path => $url)
|
||||||
|
{
|
||||||
|
if (isset($r[$url]) || isset($r[$url=self::parse_url($url,PHP_URL_PATH)]))
|
||||||
|
{
|
||||||
|
$ret[$path] = $r[$url];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// call the filesystem specific function (dont allow to use arrays!)
|
||||||
|
elseif(!function_exists($name) || is_array($pathes))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$time = null;
|
||||||
|
return $name($url,$time);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param int $time =null modification time (unix timestamp), default null = current time
|
||||||
|
* @param int $atime =null access time (unix timestamp), default null = current time, not implemented in the vfs!
|
||||||
|
* @return boolean true on success, false otherwise
|
||||||
|
*/
|
||||||
|
static function touch($path,$time=null,$atime=null)
|
||||||
|
{
|
||||||
|
return self::_call_on_backend('touch',array($path,$time,$atime));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* Requires owner or root rights!
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param string $mode mode string see Vfs::mode2int
|
||||||
|
* @return boolean true on success, false otherwise
|
||||||
|
*/
|
||||||
|
static function chmod($path,$mode)
|
||||||
|
{
|
||||||
|
return self::_call_on_backend('chmod',array($path,$mode));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* Requires root rights!
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param int $owner numeric user id
|
||||||
|
* @return boolean true on success, false otherwise
|
||||||
|
*/
|
||||||
|
static function chown($path,$owner)
|
||||||
|
{
|
||||||
|
return self::_call_on_backend('chown',array($path,$owner));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* Requires owner or root rights!
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param int $group numeric group id
|
||||||
|
* @return boolean true on success, false otherwise
|
||||||
|
*/
|
||||||
|
static function chgrp($path,$group)
|
||||||
|
{
|
||||||
|
return self::_call_on_backend('chgrp',array($path,$group));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the target of a symbolic link
|
||||||
|
*
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @return string|boolean link-data or false if no link
|
||||||
|
*/
|
||||||
|
static function readlink($path)
|
||||||
|
{
|
||||||
|
$ret = self::_call_on_backend('readlink',array($path),true); // true = fail silent, if backend does not support readlink
|
||||||
|
//error_log(__METHOD__."('$path') returning ".array2string($ret).' '.function_backtrace());
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a symbolic link
|
||||||
|
*
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* @param string $target target of the link
|
||||||
|
* @param string $link path of the link to create
|
||||||
|
* @return boolean true on success, false on error
|
||||||
|
*/
|
||||||
|
static function symlink($target,$link)
|
||||||
|
{
|
||||||
|
if (($ret = self::_call_on_backend('symlink',array($target,$link),false,1))) // 1=path is in $link!
|
||||||
|
{
|
||||||
|
self::symlinkCache_remove($link);
|
||||||
|
}
|
||||||
|
return $ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
||||||
|
*
|
||||||
|
* The methods use the following ways to get the mime type (in that order)
|
||||||
|
* - directories (is_dir()) --> self::DIR_MIME_TYPE
|
||||||
|
* - stream implemented by class defining the STAT_RETURN_MIME_TYPE constant --> use mime-type returned by url_stat
|
||||||
|
* - for regular filesystem use mime_content_type function if available
|
||||||
|
* - use eGW's mime-magic class
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @param boolean $recheck =false true = do a new check, false = rely on stored mime type (if existing)
|
||||||
|
* @return string mime-type (self::DIR_MIME_TYPE for directories)
|
||||||
|
*/
|
||||||
|
static function mime_content_type($path,$recheck=false)
|
||||||
|
{
|
||||||
|
if (!($url = self::resolve_url_symlinks($path)))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (($scheme = self::parse_url($url,PHP_URL_SCHEME)) && !$recheck)
|
||||||
|
{
|
||||||
|
// check it it's an eGW stream wrapper returning mime-type via url_stat
|
||||||
|
// we need to first check if the constant is defined, as we get a fatal error in php5.3 otherwise
|
||||||
|
if (class_exists($class = self::scheme2class($scheme)) &&
|
||||||
|
defined($class.'::STAT_RETURN_MIME_TYPE') &&
|
||||||
|
($mime_attr = constant($class.'::STAT_RETURN_MIME_TYPE')))
|
||||||
|
{
|
||||||
|
$inst = new $class;
|
||||||
|
$stat = $inst->url_stat(self::parse_url($url,PHP_URL_PATH),0);
|
||||||
|
if ($stat && $stat[$mime_attr])
|
||||||
|
{
|
||||||
|
$mime = $stat[$mime_attr];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!$mime && is_dir($url))
|
||||||
|
{
|
||||||
|
$mime = self::DIR_MIME_TYPE;
|
||||||
|
}
|
||||||
|
// if we operate on the regular filesystem and the mime_content_type function is available --> use it
|
||||||
|
if (!$mime && !$scheme && function_exists('mime_content_type'))
|
||||||
|
{
|
||||||
|
$mime = mime_content_type($path);
|
||||||
|
}
|
||||||
|
// using EGw's own mime magic (currently only checking the extension!)
|
||||||
|
if (!$mime)
|
||||||
|
{
|
||||||
|
$mime = MimeMagic::filename2mime(self::parse_url($url,PHP_URL_PATH));
|
||||||
|
}
|
||||||
|
//error_log(__METHOD__."($path,$recheck) mime=$mime");
|
||||||
|
return $mime;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the class-name for a scheme
|
||||||
|
*
|
||||||
|
* A scheme is not allowed to contain an underscore, but allows a dot and a class names only allow or need underscores, but no dots
|
||||||
|
* --> we replace dots in scheme with underscored to get the class-name
|
||||||
|
*
|
||||||
|
* @param string $scheme eg. vfs
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
static function scheme2class($scheme)
|
||||||
|
{
|
||||||
|
return Vfs\StreamWrapper::scheme2class($scheme);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears our internal stat and symlink cache
|
||||||
|
*
|
||||||
|
* Normaly not necessary, as it is automatically cleared/updated, UNLESS Vfs::$user changes!
|
||||||
|
*
|
||||||
|
* We have to clear the symlink cache before AND after calling the backend,
|
||||||
|
* because auf traversal rights may be different when Vfs::$user changes!
|
||||||
|
*
|
||||||
|
* @param string $path ='/' path of backend, whos cache to clear
|
||||||
|
*/
|
||||||
|
static function clearstatcache($path='/')
|
||||||
|
{
|
||||||
|
//error_log(__METHOD__."('$path')");
|
||||||
|
Vfs\StreamWrapper::clearstatcache($path);
|
||||||
|
self::_call_on_backend('clearstatcache', array($path), true, 0);
|
||||||
|
Vfs\StreamWrapper::clearstatcache($path);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Vfs::init_static();
|
Vfs::init_static();
|
||||||
|
@ -172,9 +172,9 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
$url = $stat['url'];
|
$url = $stat['url'];
|
||||||
}
|
}
|
||||||
// if the url resolves to a symlink to the vfs, resolve this vfs:// url direct
|
// if the url resolves to a symlink to the vfs, resolve this vfs:// url direct
|
||||||
if ($url && self::parse_url($url,PHP_URL_SCHEME) == self::SCHEME)
|
if ($url && Vfs::parse_url($url,PHP_URL_SCHEME) == self::SCHEME)
|
||||||
{
|
{
|
||||||
$url = self::resolve_url(self::parse_url($url,PHP_URL_PATH));
|
$url = self::resolve_url(Vfs::parse_url($url,PHP_URL_PATH));
|
||||||
}
|
}
|
||||||
if (self::LOG_LEVEL > 1) error_log(__METHOD__."($path,file_exists=$file_exists,resolve_last_symlink=$resolve_last_symlink) = '$url'$log");
|
if (self::LOG_LEVEL > 1) error_log(__METHOD__."($path,file_exists=$file_exists,resolve_last_symlink=$resolve_last_symlink) = '$url'$log");
|
||||||
return $url;
|
return $url;
|
||||||
@ -223,7 +223,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
'home' => str_replace(array('\\\\','\\'),array('','/'),$GLOBALS['egw_info']['user']['homedirectory']),
|
'home' => str_replace(array('\\\\','\\'),array('','/'),$GLOBALS['egw_info']['user']['homedirectory']),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
$parts = array_merge(self::parse_url($path),$defaults);
|
$parts = array_merge(Vfs::parse_url($path),$defaults);
|
||||||
if (!$parts['host']) $parts['host'] = 'default'; // otherwise we get an invalid url (scheme:///path/to/something)!
|
if (!$parts['host']) $parts['host'] = 'default'; // otherwise we get an invalid url (scheme:///path/to/something)!
|
||||||
|
|
||||||
if (!empty($parts['scheme']) && $parts['scheme'] != self::SCHEME)
|
if (!empty($parts['scheme']) && $parts['scheme'] != self::SCHEME)
|
||||||
@ -237,7 +237,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
if ($mounted == '/' || $mounted == $parts['path'] || $mounted.'/' == substr($parts['path'],0,strlen($mounted)+1))
|
if ($mounted == '/' || $mounted == $parts['path'] || $mounted.'/' == substr($parts['path'],0,strlen($mounted)+1))
|
||||||
{
|
{
|
||||||
$scheme = self::parse_url($url,PHP_URL_SCHEME);
|
$scheme = Vfs::parse_url($url,PHP_URL_SCHEME);
|
||||||
if (is_null(self::$wrappers) || !in_array($scheme,self::$wrappers))
|
if (is_null(self::$wrappers) || !in_array($scheme,self::$wrappers))
|
||||||
{
|
{
|
||||||
self::load_wrapper($scheme);
|
self::load_wrapper($scheme);
|
||||||
@ -330,7 +330,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
$this->opened_stream_mode = $mode;
|
$this->opened_stream_mode = $mode;
|
||||||
$this->opened_stream_path = $path[0] == '/' ? $path : self::parse_url($path, PHP_URL_PATH);
|
$this->opened_stream_path = $path[0] == '/' ? $path : Vfs::parse_url($path, PHP_URL_PATH);
|
||||||
$this->opened_stream_url = $url;
|
$this->opened_stream_url = $url;
|
||||||
$this->opened_stream_is_new = !$stat;
|
$this->opened_stream_is_new = !$stat;
|
||||||
|
|
||||||
@ -509,7 +509,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
Api\Hooks::process(array(
|
Api\Hooks::process(array(
|
||||||
'location' => 'vfs_unlink',
|
'location' => 'vfs_unlink',
|
||||||
'path' => $path[0] == '/' ? $path : self::parse_url($path, PHP_URL_PATH),
|
'path' => $path[0] == '/' ? $path : Vfs::parse_url($path, PHP_URL_PATH),
|
||||||
'url' => $url,
|
'url' => $url,
|
||||||
'stat' => $stat,
|
'stat' => $stat,
|
||||||
),'',true);
|
),'',true);
|
||||||
@ -537,7 +537,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// if file is moved from one filesystem / wrapper to an other --> copy it (rename fails cross wrappers)
|
// if file is moved from one filesystem / wrapper to an other --> copy it (rename fails cross wrappers)
|
||||||
if (self::parse_url($url_from,PHP_URL_SCHEME) == self::parse_url($url_to,PHP_URL_SCHEME))
|
if (Vfs::parse_url($url_from,PHP_URL_SCHEME) == Vfs::parse_url($url_to,PHP_URL_SCHEME))
|
||||||
{
|
{
|
||||||
self::symlinkCache_remove($path_from);
|
self::symlinkCache_remove($path_from);
|
||||||
$ret = rename($url_from,$url_to);
|
$ret = rename($url_from,$url_to);
|
||||||
@ -562,8 +562,8 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
Api\Hooks::process(array(
|
Api\Hooks::process(array(
|
||||||
'location' => 'vfs_rename',
|
'location' => 'vfs_rename',
|
||||||
'from' => $path_from[0] == '/' ? $path_from : self::parse_url($path_from, PHP_URL_PATH),
|
'from' => $path_from[0] == '/' ? $path_from : Vfs::parse_url($path_from, PHP_URL_PATH),
|
||||||
'to' => $path_to[0] == '/' ? $path_to : self::parse_url($path_to, PHP_URL_PATH),
|
'to' => $path_to[0] == '/' ? $path_to : Vfs::parse_url($path_to, PHP_URL_PATH),
|
||||||
'url_from' => $url_from,
|
'url_from' => $url_from,
|
||||||
'url_to' => $url_to,
|
'url_to' => $url_to,
|
||||||
),'',true);
|
),'',true);
|
||||||
@ -595,7 +595,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
Api\Hooks::process(array(
|
Api\Hooks::process(array(
|
||||||
'location' => 'vfs_mkdir',
|
'location' => 'vfs_mkdir',
|
||||||
'path' => $path[0] == '/' ? $path : self::parse_url($path, PHP_URL_PATH),
|
'path' => $path[0] == '/' ? $path : Vfs::parse_url($path, PHP_URL_PATH),
|
||||||
'url' => $url,
|
'url' => $url,
|
||||||
),'',true);
|
),'',true);
|
||||||
}
|
}
|
||||||
@ -633,7 +633,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
Api\Hooks::process(array(
|
Api\Hooks::process(array(
|
||||||
'location' => 'vfs_rmdir',
|
'location' => 'vfs_rmdir',
|
||||||
'path' => $path[0] == '/' ? $path : self::parse_url($path, PHP_URL_PATH),
|
'path' => $path[0] == '/' ? $path : Vfs::parse_url($path, PHP_URL_PATH),
|
||||||
'url' => $url,
|
'url' => $url,
|
||||||
'stat' => $stat,
|
'stat' => $stat,
|
||||||
),'',true);
|
),'',true);
|
||||||
@ -641,220 +641,6 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
return $ok;
|
return $ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Allow to call methods of the underlying stream wrapper: touch, chmod, chgrp, chown, ...
|
|
||||||
*
|
|
||||||
* We cant use a magic __call() method, as it does not work for static methods!
|
|
||||||
*
|
|
||||||
* @param string $name
|
|
||||||
* @param array $params first param has to be the path, otherwise we can not determine the correct wrapper
|
|
||||||
* @param boolean $fail_silent =false should only false be returned if function is not supported by the backend,
|
|
||||||
* or should an E_USER_WARNING error be triggered (default)
|
|
||||||
* @param int $path_param_key =0 key in params containing the path, default 0
|
|
||||||
* @return mixed return value of backend or false if function does not exist on backend
|
|
||||||
*/
|
|
||||||
static protected function _call_on_backend($name,$params,$fail_silent=false,$path_param_key=0)
|
|
||||||
{
|
|
||||||
$pathes = $params[$path_param_key];
|
|
||||||
|
|
||||||
$scheme2urls = array();
|
|
||||||
$vfs = new Vfs\StreamWrapper();
|
|
||||||
foreach(is_array($pathes) ? $pathes : array($pathes) as $path)
|
|
||||||
{
|
|
||||||
if (!($url = $vfs->resolve_url_symlinks($path,false,false)))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
$k=(string)self::parse_url($url,PHP_URL_SCHEME);
|
|
||||||
if (!(is_array($scheme2urls[$k]))) $scheme2urls[$k] = array();
|
|
||||||
$scheme2urls[$k][$path] = $url;
|
|
||||||
}
|
|
||||||
$ret = array();
|
|
||||||
foreach($scheme2urls as $scheme => $urls)
|
|
||||||
{
|
|
||||||
if ($scheme)
|
|
||||||
{
|
|
||||||
if (!class_exists($class = self::scheme2class($scheme)) || !method_exists($class,$name))
|
|
||||||
{
|
|
||||||
if (!$fail_silent) trigger_error("Can't $name for scheme $scheme!\n",E_USER_WARNING);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (!is_array($pathes))
|
|
||||||
{
|
|
||||||
$params[$path_param_key] = $url;
|
|
||||||
|
|
||||||
return call_user_func_array(array($class,$name),$params);
|
|
||||||
}
|
|
||||||
$params[$path_param_key] = $urls;
|
|
||||||
if (!is_array($r = call_user_func_array(array($class,$name),$params)))
|
|
||||||
{
|
|
||||||
return $r;
|
|
||||||
}
|
|
||||||
// we need to re-translate the urls to pathes, as they can eg. contain symlinks
|
|
||||||
foreach($urls as $path => $url)
|
|
||||||
{
|
|
||||||
if (isset($r[$url]) || isset($r[$url=self::parse_url($url,PHP_URL_PATH)]))
|
|
||||||
{
|
|
||||||
$ret[$path] = $r[$url];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// call the filesystem specific function (dont allow to use arrays!)
|
|
||||||
elseif(!function_exists($name) || is_array($pathes))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$time = null;
|
|
||||||
return $name($url,$time);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @param int $time =null modification time (unix timestamp), default null = current time
|
|
||||||
* @param int $atime =null access time (unix timestamp), default null = current time, not implemented in the vfs!
|
|
||||||
* @return boolean true on success, false otherwise
|
|
||||||
*/
|
|
||||||
static function touch($path,$time=null,$atime=null)
|
|
||||||
{
|
|
||||||
return self::_call_on_backend('touch',array($path,$time,$atime));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* Requires owner or root rights!
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @param string $mode mode string see Vfs::mode2int
|
|
||||||
* @return boolean true on success, false otherwise
|
|
||||||
*/
|
|
||||||
static function chmod($path,$mode)
|
|
||||||
{
|
|
||||||
return self::_call_on_backend('chmod',array($path,$mode));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* Requires root rights!
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @param int $owner numeric user id
|
|
||||||
* @return boolean true on success, false otherwise
|
|
||||||
*/
|
|
||||||
static function chown($path,$owner)
|
|
||||||
{
|
|
||||||
return self::_call_on_backend('chown',array($path,$owner));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* Requires owner or root rights!
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @param int $group numeric group id
|
|
||||||
* @return boolean true on success, false otherwise
|
|
||||||
*/
|
|
||||||
static function chgrp($path,$group)
|
|
||||||
{
|
|
||||||
return self::_call_on_backend('chgrp',array($path,$group));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the target of a symbolic link
|
|
||||||
*
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @return string|boolean link-data or false if no link
|
|
||||||
*/
|
|
||||||
static function readlink($path)
|
|
||||||
{
|
|
||||||
$ret = self::_call_on_backend('readlink',array($path),true); // true = fail silent, if backend does not support readlink
|
|
||||||
//error_log(__METHOD__."('$path') returning ".array2string($ret).' '.function_backtrace());
|
|
||||||
return $ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a symbolic link
|
|
||||||
*
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* @param string $target target of the link
|
|
||||||
* @param string $link path of the link to create
|
|
||||||
* @return boolean true on success, false on error
|
|
||||||
*/
|
|
||||||
static function symlink($target,$link)
|
|
||||||
{
|
|
||||||
if (($ret = self::_call_on_backend('symlink',array($target,$link),false,1))) // 1=path is in $link!
|
|
||||||
{
|
|
||||||
self::symlinkCache_remove($link);
|
|
||||||
}
|
|
||||||
return $ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This is not (yet) a stream-wrapper function, but it's necessary and can be used static
|
|
||||||
*
|
|
||||||
* The methods use the following ways to get the mime type (in that order)
|
|
||||||
* - directories (is_dir()) --> self::DIR_MIME_TYPE
|
|
||||||
* - stream implemented by class defining the STAT_RETURN_MIME_TYPE constant --> use mime-type returned by url_stat
|
|
||||||
* - for regular filesystem use mime_content_type function if available
|
|
||||||
* - use eGW's mime-magic class
|
|
||||||
*
|
|
||||||
* @param string $path
|
|
||||||
* @param boolean $recheck =false true = do a new check, false = rely on stored mime type (if existing)
|
|
||||||
* @return string mime-type (self::DIR_MIME_TYPE for directories)
|
|
||||||
*/
|
|
||||||
static function mime_content_type($path,$recheck=false)
|
|
||||||
{
|
|
||||||
$vfs = new self();
|
|
||||||
if (!($url = $vfs->resolve_url_symlinks($path)))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (($scheme = self::parse_url($url,PHP_URL_SCHEME)) && !$recheck)
|
|
||||||
{
|
|
||||||
// check it it's an eGW stream wrapper returning mime-type via url_stat
|
|
||||||
// we need to first check if the constant is defined, as we get a fatal error in php5.3 otherwise
|
|
||||||
if (class_exists($class = self::scheme2class($scheme)) &&
|
|
||||||
defined($class.'::STAT_RETURN_MIME_TYPE') &&
|
|
||||||
($mime_attr = constant($class.'::STAT_RETURN_MIME_TYPE')))
|
|
||||||
{
|
|
||||||
$inst = new $class;
|
|
||||||
$stat = $inst->url_stat(self::parse_url($url,PHP_URL_PATH),0);
|
|
||||||
if ($stat && $stat[$mime_attr])
|
|
||||||
{
|
|
||||||
$mime = $stat[$mime_attr];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!$mime && is_dir($url))
|
|
||||||
{
|
|
||||||
$mime = self::DIR_MIME_TYPE;
|
|
||||||
}
|
|
||||||
// if we operate on the regular filesystem and the mime_content_type function is available --> use it
|
|
||||||
if (!$mime && !$scheme && function_exists('mime_content_type'))
|
|
||||||
{
|
|
||||||
$mime = mime_content_type($path);
|
|
||||||
}
|
|
||||||
// using EGw's own mime magic (currently only checking the extension!)
|
|
||||||
if (!$mime)
|
|
||||||
{
|
|
||||||
$mime = Api\MimeMagic::filename2mime(self::parse_url($url,PHP_URL_PATH));
|
|
||||||
}
|
|
||||||
//error_log(__METHOD__."($path,$recheck) mime=$mime");
|
|
||||||
return $mime;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method is called immediately when your stream object is created for examining directory contents with opendir().
|
* This method is called immediately when your stream object is created for examining directory contents with opendir().
|
||||||
*
|
*
|
||||||
@ -880,7 +666,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
}
|
}
|
||||||
$this->opened_dir_writable = Vfs::check_access($this->opened_dir_url,Vfs::WRITABLE);
|
$this->opened_dir_writable = Vfs::check_access($this->opened_dir_url,Vfs::WRITABLE);
|
||||||
// check our fstab if we need to add some of the mountpoints
|
// check our fstab if we need to add some of the mountpoints
|
||||||
$basepath = self::parse_url($path,PHP_URL_PATH);
|
$basepath = Vfs::parse_url($path,PHP_URL_PATH);
|
||||||
foreach(array_keys(self::$fstab) as $mounted)
|
foreach(array_keys(self::$fstab) as $mounted)
|
||||||
{
|
{
|
||||||
if (((Vfs::dirname($mounted) == $basepath || Vfs::dirname($mounted).'/' == $basepath) && $mounted != '/') &&
|
if (((Vfs::dirname($mounted) == $basepath || Vfs::dirname($mounted).'/' == $basepath) && $mounted != '/') &&
|
||||||
@ -949,11 +735,11 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
if (self::LOG_LEVEL > 0) error_log(__METHOD__."('$path',$flags) maximum symlink depth exceeded, might be a circular symlink!");
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__."('$path',$flags) maximum symlink depth exceeded, might be a circular symlink!");
|
||||||
$stat = false;
|
$stat = false;
|
||||||
}
|
}
|
||||||
elseif (($lpath = self::readlink($url)))
|
elseif (($lpath = Vfs::readlink($url)))
|
||||||
{
|
{
|
||||||
if ($lpath[0] != '/') // concat relative path
|
if ($lpath[0] != '/') // concat relative path
|
||||||
{
|
{
|
||||||
$lpath = Vfs::concat(self::parse_url($path,PHP_URL_PATH),'../'.$lpath);
|
$lpath = Vfs::concat(Vfs::parse_url($path,PHP_URL_PATH),'../'.$lpath);
|
||||||
}
|
}
|
||||||
$u_query = parse_url($url,PHP_URL_QUERY);
|
$u_query = parse_url($url,PHP_URL_QUERY);
|
||||||
$url = Vfs::PREFIX.$lpath;
|
$url = Vfs::PREFIX.$lpath;
|
||||||
@ -991,7 +777,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
throw $e;
|
throw $e;
|
||||||
}
|
}
|
||||||
// check if a failed url_stat was for a home dir, in that case silently create it
|
// check if a failed url_stat was for a home dir, in that case silently create it
|
||||||
if (!$stat && $try_create_home && Vfs::dirname(self::parse_url($path,PHP_URL_PATH)) == '/home' &&
|
if (!$stat && $try_create_home && Vfs::dirname(Vfs::parse_url($path,PHP_URL_PATH)) == '/home' &&
|
||||||
($id = $GLOBALS['egw']->accounts->name2id(basename($path))) &&
|
($id = $GLOBALS['egw']->accounts->name2id(basename($path))) &&
|
||||||
$GLOBALS['egw']->accounts->id2name($id) == basename($path)) // make sure path has the right case!
|
$GLOBALS['egw']->accounts->id2name($id) == basename($path)) // make sure path has the right case!
|
||||||
{
|
{
|
||||||
@ -1060,13 +846,13 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
{
|
{
|
||||||
if (($stat = $this->url_stat($url,0,false,false)))
|
if (($stat = $this->url_stat($url,0,false,false)))
|
||||||
{
|
{
|
||||||
if (is_link($url) && ($lpath = self::readlink($url)))
|
if (is_link($url) && ($lpath = Vfs::readlink($url)))
|
||||||
{
|
{
|
||||||
if (self::LOG_LEVEL > 1) $log = "rel_path='$rel_path', url='$url': lpath='$lpath'";
|
if (self::LOG_LEVEL > 1) $log = "rel_path='$rel_path', url='$url': lpath='$lpath'";
|
||||||
|
|
||||||
if ($lpath[0] != '/')
|
if ($lpath[0] != '/')
|
||||||
{
|
{
|
||||||
$lpath = Vfs::concat(self::parse_url($url,PHP_URL_PATH),'../'.$lpath);
|
$lpath = Vfs::concat(Vfs::parse_url($url,PHP_URL_PATH),'../'.$lpath);
|
||||||
}
|
}
|
||||||
//self::symlinkCache_add($path,Vfs::PREFIX.$lpath);
|
//self::symlinkCache_add($path,Vfs::PREFIX.$lpath);
|
||||||
$url = Vfs::PREFIX.Vfs::concat($lpath,$rel_path);
|
$url = Vfs::PREFIX.Vfs::concat($lpath,$rel_path);
|
||||||
@ -1101,7 +887,7 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
|
|
||||||
if (isset(self::$symlink_cache[$path])) return; // nothing to do
|
if (isset(self::$symlink_cache[$path])) return; // nothing to do
|
||||||
|
|
||||||
if ($target[0] != '/') $target = self::parse_url($target,PHP_URL_PATH);
|
if ($target[0] != '/') $target = Vfs::parse_url($target,PHP_URL_PATH);
|
||||||
|
|
||||||
self::$symlink_cache[$path] = $target;
|
self::$symlink_cache[$path] = $target;
|
||||||
|
|
||||||
@ -1165,17 +951,9 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
* Clears our internal stat and symlink cache
|
* Clears our internal stat and symlink cache
|
||||||
*
|
*
|
||||||
* Normaly not necessary, as it is automatically cleared/updated, UNLESS Vfs::$user changes!
|
* Normaly not necessary, as it is automatically cleared/updated, UNLESS Vfs::$user changes!
|
||||||
*
|
|
||||||
* We have to clear the symlink cache before AND after calling the backend,
|
|
||||||
* because auf traversal rights may be different when Vfs::$user changes!
|
|
||||||
*
|
|
||||||
* @param string $path ='/' path of backend, whos cache to clear
|
|
||||||
*/
|
*/
|
||||||
static function clearstatcache($path='/')
|
static function clearstatcache()
|
||||||
{
|
{
|
||||||
//error_log(__METHOD__."('$path')");
|
|
||||||
self::$symlink_cache = self::$resolve_url_cache = array();
|
|
||||||
self::_call_on_backend('clearstatcache', array($path), true, 0);
|
|
||||||
self::$symlink_cache = self::$resolve_url_cache = array();
|
self::$symlink_cache = self::$resolve_url_cache = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1313,59 +1091,6 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Utf-8 save version of parse_url
|
|
||||||
*
|
|
||||||
* Does caching withing request, to not have to parse urls over and over again.
|
|
||||||
*
|
|
||||||
* @param string $url
|
|
||||||
* @param int $component =-1 PHP_URL_* constants
|
|
||||||
* @return array|string|boolean on success array or string, if $component given, or false on failure
|
|
||||||
*/
|
|
||||||
static function parse_url($url, $component=-1)
|
|
||||||
{
|
|
||||||
static $component2str = array(
|
|
||||||
PHP_URL_SCHEME => 'scheme',
|
|
||||||
PHP_URL_HOST => 'host',
|
|
||||||
PHP_URL_PORT => 'port',
|
|
||||||
PHP_URL_USER => 'user',
|
|
||||||
PHP_URL_PASS => 'pass',
|
|
||||||
PHP_URL_PATH => 'path',
|
|
||||||
PHP_URL_QUERY => 'query',
|
|
||||||
PHP_URL_FRAGMENT => 'fragment',
|
|
||||||
);
|
|
||||||
static $cache = array(); // some caching
|
|
||||||
|
|
||||||
$result =& $cache[$url];
|
|
||||||
|
|
||||||
if (!isset($result))
|
|
||||||
{
|
|
||||||
// Build arrays of values we need to decode before parsing
|
|
||||||
static $entities = array('%21', '%2A', '%27', '%28', '%29', '%3B', '%3A', '%40', '%26', '%3D', '%24', '%2C', '%2F', '%3F', '%23', '%5B', '%5D');
|
|
||||||
static $replacements = array('!', '*', "'", "(", ")", ";", ":", "@", "&", "=", "$", ",", "/", "?", "#", "[", "]");
|
|
||||||
static $str_replace = null;
|
|
||||||
if (!isset($str_replace)) $str_replace = function_exists('mb_str_replace') ? 'mb_str_replace' : 'str_replace';
|
|
||||||
|
|
||||||
// Create encoded URL with special URL characters decoded so it can be parsed
|
|
||||||
// All other characters will be encoded
|
|
||||||
$encodedURL = $str_replace($entities, $replacements, urlencode($url));
|
|
||||||
|
|
||||||
// Parse the encoded URL
|
|
||||||
$result = $encodedParts = parse_url($encodedURL);
|
|
||||||
|
|
||||||
// Now, decode each value of the resulting array
|
|
||||||
if ($encodedParts)
|
|
||||||
{
|
|
||||||
$result = array();
|
|
||||||
foreach ($encodedParts as $key => $value)
|
|
||||||
{
|
|
||||||
$result[$key] = urldecode($str_replace($replacements, $entities, $value));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $component >= 0 ? $result[$component2str[$component]] : $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Getting the path from an url (or path) AND removing trailing slashes
|
* Getting the path from an url (or path) AND removing trailing slashes
|
||||||
*
|
*
|
||||||
@ -1375,14 +1100,14 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
*/
|
*/
|
||||||
static protected function get_path($path,$only_remove_scheme=self::SCHEME)
|
static protected function get_path($path,$only_remove_scheme=self::SCHEME)
|
||||||
{
|
{
|
||||||
if ($path[0] != '/' && (!$only_remove_scheme || self::parse_url($path, PHP_URL_SCHEME) == $only_remove_scheme))
|
if ($path[0] != '/' && (!$only_remove_scheme || Vfs::parse_url($path, PHP_URL_SCHEME) == $only_remove_scheme))
|
||||||
{
|
{
|
||||||
$path = self::parse_url($path, PHP_URL_PATH);
|
$path = Vfs::parse_url($path, PHP_URL_PATH);
|
||||||
}
|
}
|
||||||
// remove trailing slashes eg. added by WebDAV, but do NOT remove / from "sqlfs://default/"!
|
// remove trailing slashes eg. added by WebDAV, but do NOT remove / from "sqlfs://default/"!
|
||||||
if ($path != '/')
|
if ($path != '/')
|
||||||
{
|
{
|
||||||
while (mb_substr($path, -1) == '/' && $path != '/' && ($path[0] == '/' || self::parse_url($path, PHP_URL_PATH) != '/'))
|
while (mb_substr($path, -1) == '/' && $path != '/' && ($path[0] == '/' || Vfs::parse_url($path, PHP_URL_PATH) != '/'))
|
||||||
{
|
{
|
||||||
$path = mb_substr($path,0,-1);
|
$path = mb_substr($path,0,-1);
|
||||||
}
|
}
|
||||||
@ -1408,6 +1133,116 @@ class StreamWrapper implements StreamWrapperIface
|
|||||||
return $ret;
|
return $ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mounts $url under $path in the vfs, called without parameter it returns the fstab
|
||||||
|
*
|
||||||
|
* The fstab is stored in the eGW configuration and used for all eGW users.
|
||||||
|
*
|
||||||
|
* @param string $url =null url of the filesystem to mount, eg. oldvfs://default/
|
||||||
|
* @param string $path =null path to mount the filesystem in the vfs, eg. /
|
||||||
|
* @param boolean $check_url =null check if url is an existing directory, before mounting it
|
||||||
|
* default null only checks if url does not contain a $ as used in $user or $pass
|
||||||
|
* @param boolean $persitent_mount =true create a persitent mount, or only a temprary for current request
|
||||||
|
* @param boolean $clear_fstab =false true clear current fstab, false (default) only add given mount
|
||||||
|
* @return array|boolean array with fstab, if called without parameter or true on successful mount
|
||||||
|
*/
|
||||||
|
static function mount($url=null,$path=null,$check_url=null,$persitent_mount=true,$clear_fstab=false)
|
||||||
|
{
|
||||||
|
if (is_null($check_url)) $check_url = strpos($url,'$') === false;
|
||||||
|
|
||||||
|
if (!isset($GLOBALS['egw_info']['server']['vfs_fstab'])) // happens eg. in setup
|
||||||
|
{
|
||||||
|
$api_config = Config::read('phpgwapi');
|
||||||
|
if (isset($api_config['vfs_fstab']) && is_array($api_config['vfs_fstab']))
|
||||||
|
{
|
||||||
|
self::$fstab = $api_config['vfs_fstab'];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
self::$fstab = array(
|
||||||
|
'/' => 'sqlfs://$host/',
|
||||||
|
'/apps' => 'links://$host/apps',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
unset($api_config);
|
||||||
|
}
|
||||||
|
if (is_null($url) || is_null($path))
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') returns '.array2string(self::$fstab));
|
||||||
|
return self::$fstab;
|
||||||
|
}
|
||||||
|
if (!self::$is_root)
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') permission denied, you are NOT root!');
|
||||||
|
return false; // only root can mount
|
||||||
|
}
|
||||||
|
if ($clear_fstab)
|
||||||
|
{
|
||||||
|
self::$fstab = array();
|
||||||
|
}
|
||||||
|
if (isset(self::$fstab[$path]) && self::$fstab[$path] === $url)
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') already mounted.');
|
||||||
|
return true; // already mounted
|
||||||
|
}
|
||||||
|
self::load_wrapper(Vfs::parse_url($url,PHP_URL_SCHEME));
|
||||||
|
|
||||||
|
if ($check_url && (!file_exists($url) || opendir($url) === false))
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') url does NOT exist!');
|
||||||
|
return false; // url does not exist
|
||||||
|
}
|
||||||
|
self::$fstab[$path] = $url;
|
||||||
|
|
||||||
|
uksort(self::$fstab, function($a, $b)
|
||||||
|
{
|
||||||
|
return strlen($a) - strlen($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
if ($persitent_mount)
|
||||||
|
{
|
||||||
|
Config::save_value('vfs_fstab',self::$fstab,'phpgwapi');
|
||||||
|
$GLOBALS['egw_info']['server']['vfs_fstab'] = self::$fstab;
|
||||||
|
// invalidate session cache
|
||||||
|
if (method_exists($GLOBALS['egw'],'invalidate_session_cache')) // egw object in setup is limited
|
||||||
|
{
|
||||||
|
$GLOBALS['egw']->invalidate_session_cache();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($url).','.array2string($path).') returns true (successful new mount).');
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unmounts a filesystem part of the vfs
|
||||||
|
*
|
||||||
|
* @param string $path url or path of the filesystem to unmount
|
||||||
|
*/
|
||||||
|
static function umount($path)
|
||||||
|
{
|
||||||
|
if (!self::$is_root)
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($path).','.array2string($path).') permission denied, you are NOT root!');
|
||||||
|
return false; // only root can mount
|
||||||
|
}
|
||||||
|
if (!isset(self::$fstab[$path]) && ($path = array_search($path,self::$fstab)) === false)
|
||||||
|
{
|
||||||
|
if (self::LOG_LEVEL > 0) error_log(__METHOD__.'('.array2string($path).') NOT mounted!');
|
||||||
|
return false; // $path not mounted
|
||||||
|
}
|
||||||
|
unset(self::$fstab[$path]);
|
||||||
|
|
||||||
|
Config::save_value('vfs_fstab',self::$fstab,'phpgwapi');
|
||||||
|
$GLOBALS['egw_info']['server']['vfs_fstab'] = self::$fstab;
|
||||||
|
// invalidate session cache
|
||||||
|
if (method_exists($GLOBALS['egw'],'invalidate_session_cache')) // egw object in setup is limited
|
||||||
|
{
|
||||||
|
$GLOBALS['egw']->invalidate_session_cache();
|
||||||
|
}
|
||||||
|
if (self::LOG_LEVEL > 1) error_log(__METHOD__.'('.array2string($path).') returns true (successful unmount).');
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Init our static properties and register this wrapper
|
* Init our static properties and register this wrapper
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user