mirror of
https://github.com/EGroupware/egroupware.git
synced 2024-11-30 11:53:57 +01:00
143 lines
3.7 KiB
PHP
143 lines
3.7 KiB
PHP
<?php
|
|
/**
|
|
* eGroupWare API: VFS - stream wrapper for global variables
|
|
*
|
|
* @link http://www.egroupware.org
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
* @package api
|
|
* @subpackage vfs
|
|
* @version $Id$
|
|
*/
|
|
|
|
/**
|
|
* eGroupWare API: stream wrapper for global variables. It makes variables available as streams.
|
|
*
|
|
* Original from an expample on php.net:
|
|
* @see http://de.php.net/manual/en/function.stream-wrapper-register.php
|
|
*
|
|
* Use as global://varname (please note global://host/varname does NOT work, as parse_url() would return with a leading slash!)
|
|
*
|
|
* Streamwrapper is now mbstring.func_overload save.
|
|
*
|
|
* @todo: allow to use path to access arrays: global://varname/key1/key2 to access $string from $GLOBALS['varname'] = array('key1'=>array('key2'=>$string));
|
|
*/
|
|
class global_stream_wrapper
|
|
{
|
|
private $pos;
|
|
private $stream;
|
|
private $name;
|
|
|
|
public function stream_open($path, $mode, $options, &$opened_path)
|
|
{
|
|
$this->stream = &$GLOBALS[$this->name=parse_url($path,PHP_URL_HOST)];
|
|
$this->pos = 0;
|
|
if (!is_string($this->stream)) return false;
|
|
return true;
|
|
}
|
|
|
|
public function stream_read($count)
|
|
{
|
|
$ret = self::_cut_bytes($this->stream, $this->pos, $count);
|
|
//error_log(__METHOD__."($count) this->pos=$this->pos, self::_bytes(this->stream)=".self::_bytes($this->stream)." self::_bytes(ret)=".self::_bytes($ret));
|
|
$this->pos += self::_bytes($ret);
|
|
return $ret;
|
|
}
|
|
|
|
public function stream_write($data)
|
|
{
|
|
$l=self::_bytes($data);
|
|
$this->stream =
|
|
self::_cut_bytes($this->stream, 0, $this->pos) .
|
|
$data .
|
|
self::_cut_bytes($this->stream, $this->pos += $l);
|
|
return $l;
|
|
}
|
|
|
|
public function stream_tell()
|
|
{
|
|
return $this->pos;
|
|
}
|
|
|
|
public function stream_eof()
|
|
{
|
|
return $this->pos >= self::_bytes($this->stream);
|
|
}
|
|
|
|
public function stream_seek($offset, $whence)
|
|
{
|
|
$l=self::_bytes($this->stream);
|
|
switch ($whence)
|
|
{
|
|
case SEEK_SET: $newPos = $offset; break;
|
|
case SEEK_CUR: $newPos = $this->pos + $offset; break;
|
|
case SEEK_END: $newPos = $l + $offset; break;
|
|
default: return false;
|
|
}
|
|
$ret = ($newPos >=0 && $newPos <=$l);
|
|
if ($ret) $this->pos=$newPos;
|
|
return $ret;
|
|
}
|
|
|
|
public function stream_stat()
|
|
{
|
|
if (!isset($this->stream))
|
|
{
|
|
return false;
|
|
}
|
|
return array(
|
|
'ino' => md5($this->name),
|
|
'name' => $this->name,
|
|
'mode' => 0100000,
|
|
'size' => bytes($this->stream),
|
|
'uid' => 0,
|
|
'gid' => 0,
|
|
'mtime' => 0,
|
|
'ctime' => 0,
|
|
'nlink' => 1,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* are the string functions overloaded by their mbstring variants
|
|
*
|
|
* @var boolean
|
|
*/
|
|
private static $mbstring_func_overload;
|
|
|
|
/**
|
|
* mbstring.func_overload safe strlen
|
|
*
|
|
* @param string &$data
|
|
* @return int
|
|
*/
|
|
private static function _bytes(&$data)
|
|
{
|
|
return self::$mbstring_func_overload ? mb_strlen($data,'ascii') : strlen($data);
|
|
}
|
|
|
|
/**
|
|
* mbstring.func_overload safe substr
|
|
*
|
|
* @param string &$data
|
|
* @param int $offset
|
|
* @param int $len
|
|
* @return string
|
|
*/
|
|
private static function _cut_bytes(&$data,$offset,$len=null)
|
|
{
|
|
return self::$mbstring_func_overload ? mb_substr($data,$offset,$len,'ascii') : substr($data,$offset,$len);
|
|
}
|
|
|
|
/**
|
|
* Init the (static parts) of the stream-wrapper
|
|
*
|
|
*/
|
|
public static function init()
|
|
{
|
|
self::$mbstring_func_overload = @extension_loaded('mbstring') && (ini_get('mbstring.func_overload') & 2);
|
|
|
|
stream_wrapper_register('global', 'global_stream_wrapper');
|
|
}
|
|
}
|
|
global_stream_wrapper::init();
|