|
@@ -0,0 +1,2651 @@
|
|
|
+<?php
|
|
|
+
|
|
|
+/*
|
|
|
+ Copyright (c) 2009-2014 F3::Factory/Bong Cosca, All rights reserved.
|
|
|
+
|
|
|
+ This file is part of the Fat-Free Framework (http://fatfree.sf.net).
|
|
|
+
|
|
|
+ THE SOFTWARE AND DOCUMENTATION ARE PROVIDED "AS IS" WITHOUT WARRANTY OF
|
|
|
+ ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
|
|
+ IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR
|
|
|
+ PURPOSE.
|
|
|
+
|
|
|
+ Please see the license.txt file for more information.
|
|
|
+*/
|
|
|
+
|
|
|
+//! Factory class for single-instance objects
|
|
|
+abstract class Prefab {
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return class instance
|
|
|
+ * @return static
|
|
|
+ **/
|
|
|
+ static function instance() {
|
|
|
+ if (!Registry::exists($class=get_called_class())) {
|
|
|
+ $ref=new Reflectionclass($class);
|
|
|
+ $args=func_get_args();
|
|
|
+ Registry::set($class,
|
|
|
+ $args?$ref->newinstanceargs($args):new $class);
|
|
|
+ }
|
|
|
+ return Registry::get($class);
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! Base structure
|
|
|
+class Base extends Prefab {
|
|
|
+
|
|
|
+ //@{ Framework details
|
|
|
+ const
|
|
|
+ PACKAGE='Fat-Free Framework',
|
|
|
+ VERSION='3.3.0-Dev';
|
|
|
+ //@}
|
|
|
+
|
|
|
+ //@{ HTTP status codes (RFC 2616)
|
|
|
+ const
|
|
|
+ HTTP_100='Continue',
|
|
|
+ HTTP_101='Switching Protocols',
|
|
|
+ HTTP_200='OK',
|
|
|
+ HTTP_201='Created',
|
|
|
+ HTTP_202='Accepted',
|
|
|
+ HTTP_203='Non-Authorative Information',
|
|
|
+ HTTP_204='No Content',
|
|
|
+ HTTP_205='Reset Content',
|
|
|
+ HTTP_206='Partial Content',
|
|
|
+ HTTP_300='Multiple Choices',
|
|
|
+ HTTP_301='Moved Permanently',
|
|
|
+ HTTP_302='Found',
|
|
|
+ HTTP_303='See Other',
|
|
|
+ HTTP_304='Not Modified',
|
|
|
+ HTTP_305='Use Proxy',
|
|
|
+ HTTP_307='Temporary Redirect',
|
|
|
+ HTTP_400='Bad Request',
|
|
|
+ HTTP_401='Unauthorized',
|
|
|
+ HTTP_402='Payment Required',
|
|
|
+ HTTP_403='Forbidden',
|
|
|
+ HTTP_404='Not Found',
|
|
|
+ HTTP_405='Method Not Allowed',
|
|
|
+ HTTP_406='Not Acceptable',
|
|
|
+ HTTP_407='Proxy Authentication Required',
|
|
|
+ HTTP_408='Request Timeout',
|
|
|
+ HTTP_409='Conflict',
|
|
|
+ HTTP_410='Gone',
|
|
|
+ HTTP_411='Length Required',
|
|
|
+ HTTP_412='Precondition Failed',
|
|
|
+ HTTP_413='Request Entity Too Large',
|
|
|
+ HTTP_414='Request-URI Too Long',
|
|
|
+ HTTP_415='Unsupported Media Type',
|
|
|
+ HTTP_416='Requested Range Not Satisfiable',
|
|
|
+ HTTP_417='Expectation Failed',
|
|
|
+ HTTP_500='Internal Server Error',
|
|
|
+ HTTP_501='Not Implemented',
|
|
|
+ HTTP_502='Bad Gateway',
|
|
|
+ HTTP_503='Service Unavailable',
|
|
|
+ HTTP_504='Gateway Timeout',
|
|
|
+ HTTP_505='HTTP Version Not Supported';
|
|
|
+ //@}
|
|
|
+
|
|
|
+ const
|
|
|
+ //! Mapped PHP globals
|
|
|
+ GLOBALS='GET|POST|COOKIE|REQUEST|SESSION|FILES|SERVER|ENV',
|
|
|
+ //! HTTP verbs
|
|
|
+ VERBS='GET|HEAD|POST|PUT|PATCH|DELETE|CONNECT',
|
|
|
+ //! Default directory permissions
|
|
|
+ MODE=0755,
|
|
|
+ //! Syntax highlighting stylesheet
|
|
|
+ CSS='code.css';
|
|
|
+
|
|
|
+ //@{ HTTP request types
|
|
|
+ const
|
|
|
+ REQ_SYNC=1,
|
|
|
+ REQ_AJAX=2;
|
|
|
+ //@}
|
|
|
+
|
|
|
+ //@{ Error messages
|
|
|
+ const
|
|
|
+ E_Pattern='Invalid routing pattern: %s',
|
|
|
+ E_Named='Named route does not exist: %s',
|
|
|
+ E_Fatal='Fatal error: %s',
|
|
|
+ E_Open='Unable to open %s',
|
|
|
+ E_Routes='No routes specified',
|
|
|
+ E_Class='Invalid class %s',
|
|
|
+ E_Method='Invalid method %s',
|
|
|
+ E_Hive='Invalid hive key %s';
|
|
|
+ //@}
|
|
|
+
|
|
|
+ private
|
|
|
+ //! Globals
|
|
|
+ $hive,
|
|
|
+ //! Initial settings
|
|
|
+ $init,
|
|
|
+ //! Language lookup sequence
|
|
|
+ $languages,
|
|
|
+ //! Default fallback language
|
|
|
+ $fallback='en';
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Sync PHP global with corresponding hive key
|
|
|
+ * @return array
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function sync($key) {
|
|
|
+ return $this->hive[$key]=&$GLOBALS['_'.$key];
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return the parts of specified hive key
|
|
|
+ * @return array
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ private function cut($key) {
|
|
|
+ return preg_split('/\[\h*[\'"]?(.+?)[\'"]?\h*\]|(->)|\./',
|
|
|
+ $key,NULL,PREG_SPLIT_NO_EMPTY|PREG_SPLIT_DELIM_CAPTURE);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Replace tokenized URL with current route's token values
|
|
|
+ * @return string
|
|
|
+ * @param $url array|string
|
|
|
+ **/
|
|
|
+ function build($url) {
|
|
|
+ if (is_array($url))
|
|
|
+ foreach ($url as &$var) {
|
|
|
+ $var=$this->build($var);
|
|
|
+ unset($var);
|
|
|
+ }
|
|
|
+ elseif (preg_match_all('/@(\w+)/',$url,$matches,PREG_SET_ORDER))
|
|
|
+ foreach ($matches as $match)
|
|
|
+ if (array_key_exists($match[1],$this->hive['PARAMS']))
|
|
|
+ $url=str_replace($match[0],
|
|
|
+ $this->hive['PARAMS'][$match[1]],$url);
|
|
|
+ return $url;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Parse string containing key-value pairs and use as routing tokens
|
|
|
+ * @return NULL
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function parse($str) {
|
|
|
+ preg_match_all('/(\w+)\h*=\h*(.+?)(?=,|$)/',
|
|
|
+ $str,$pairs,PREG_SET_ORDER);
|
|
|
+ foreach ($pairs as $pair)
|
|
|
+ $this->hive['PARAMS'][$pair[1]]=trim($pair[2]);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert JS-style token to PHP expression
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function compile($str) {
|
|
|
+ $fw=$this;
|
|
|
+ return preg_replace_callback(
|
|
|
+ '/(?<!\w)@(\w(?:[\w\.\[\]]|\->|::)*)/',
|
|
|
+ function($var) use($fw) {
|
|
|
+ return '$'.preg_replace_callback(
|
|
|
+ '/\.(\w+)|\[((?:[^\[\]]*|(?R))*)\]/',
|
|
|
+ function($expr) use($fw) {
|
|
|
+ return '['.var_export(
|
|
|
+ isset($expr[2])?
|
|
|
+ $fw->compile($expr[2]):
|
|
|
+ (ctype_digit($expr[1])?
|
|
|
+ (int)$expr[1]:
|
|
|
+ $expr[1]),TRUE).']';
|
|
|
+ },
|
|
|
+ $var[1]
|
|
|
+ );
|
|
|
+ },
|
|
|
+ $str
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Get hive key reference/contents; Add non-existent hive keys,
|
|
|
+ * array elements, and object properties by default
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ * @param $add bool
|
|
|
+ **/
|
|
|
+ function &ref($key,$add=TRUE) {
|
|
|
+ $parts=$this->cut($key);
|
|
|
+ if ($parts[0]=='SESSION') {
|
|
|
+ @session_start();
|
|
|
+ $this->sync('SESSION');
|
|
|
+ }
|
|
|
+ elseif (!preg_match('/^\w+$/',$parts[0]))
|
|
|
+ user_error(sprintf(self::E_Hive,$this->stringify($key)));
|
|
|
+ if ($add)
|
|
|
+ $var=&$this->hive;
|
|
|
+ else
|
|
|
+ $var=$this->hive;
|
|
|
+ $obj=FALSE;
|
|
|
+ foreach ($parts as $part)
|
|
|
+ if ($part=='->')
|
|
|
+ $obj=TRUE;
|
|
|
+ elseif ($obj) {
|
|
|
+ $obj=FALSE;
|
|
|
+ if (!is_object($var))
|
|
|
+ $var=new stdclass;
|
|
|
+ if ($add || property_exists($var,$part))
|
|
|
+ $var=&$var->$part;
|
|
|
+ else {
|
|
|
+ $var=&$this->null;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else {
|
|
|
+ if (!is_array($var))
|
|
|
+ $var=array();
|
|
|
+ if ($add || array_key_exists($part,$var))
|
|
|
+ $var=&$var[$part];
|
|
|
+ else {
|
|
|
+ $var=&$this->null;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ($parts[0]=='ALIASES')
|
|
|
+ $var=$this->build($var);
|
|
|
+ return $var;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return TRUE if hive key is not set
|
|
|
+ * (or return timestamp and TTL if cached)
|
|
|
+ * @return bool
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ **/
|
|
|
+ function exists($key,&$val=NULL) {
|
|
|
+ $val=$this->ref($key,FALSE);
|
|
|
+ return isset($val)?
|
|
|
+ TRUE:
|
|
|
+ (Cache::instance()->exists($this->hash($key).'.var',$val)?
|
|
|
+ $val:FALSE);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return TRUE if hive key is empty and not cached
|
|
|
+ * @return bool
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function devoid($key) {
|
|
|
+ $val=$this->ref($key,FALSE);
|
|
|
+ return empty($val) &&
|
|
|
+ (!Cache::instance()->exists($this->hash($key).'.var',$val) ||
|
|
|
+ !$val);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Bind value to hive key
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ * @param $ttl int
|
|
|
+ **/
|
|
|
+ function set($key,$val,$ttl=0) {
|
|
|
+ if (preg_match('/^(GET|POST|COOKIE)\b(.+)/',$key,$expr)) {
|
|
|
+ $this->set('REQUEST'.$expr[2],$val);
|
|
|
+ if ($expr[1]=='COOKIE') {
|
|
|
+ $parts=$this->cut($key);
|
|
|
+ $jar=$this->hive['JAR'];
|
|
|
+ if ($ttl)
|
|
|
+ $jar['expire']=time()+$ttl;
|
|
|
+ call_user_func_array('setcookie',array($parts[1],$val)+$jar);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else switch ($key) {
|
|
|
+ case 'CACHE':
|
|
|
+ $val=Cache::instance()->load($val,TRUE);
|
|
|
+ break;
|
|
|
+ case 'ENCODING':
|
|
|
+ $val=ini_set('default_charset',$val);
|
|
|
+ if (extension_loaded('mbstring'))
|
|
|
+ mb_internal_encoding($val);
|
|
|
+ break;
|
|
|
+ case 'FALLBACK':
|
|
|
+ $this->fallback=$val;
|
|
|
+ $lang=$this->language($this->hive['LANGUAGE']);
|
|
|
+ case 'LANGUAGE':
|
|
|
+ if (isset($lang) || $lang=$this->language($val))
|
|
|
+ $val=$this->language($val);
|
|
|
+ $lex=$this->lexicon($this->hive['LOCALES']);
|
|
|
+ case 'LOCALES':
|
|
|
+ if (isset($lex) || $lex=$this->lexicon($val))
|
|
|
+ $this->mset($lex,$this->hive['PREFIX'],$ttl);
|
|
|
+ break;
|
|
|
+ case 'TZ':
|
|
|
+ date_default_timezone_set($val);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ $ref=$val;
|
|
|
+ if (preg_match('/^JAR\b/',$key))
|
|
|
+ call_user_func_array(
|
|
|
+ 'session_set_cookie_params',$this->hive['JAR']);
|
|
|
+ $cache=Cache::instance();
|
|
|
+ if ($cache->exists($hash=$this->hash($key).'.var') || $ttl)
|
|
|
+ // Persist the key-value pair
|
|
|
+ $cache->set($hash,$val,$ttl);
|
|
|
+ return $ref;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Retrieve contents of hive key
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ * @param $args string|array
|
|
|
+ **/
|
|
|
+ function get($key,$args=NULL) {
|
|
|
+ if (is_string($val=$this->ref($key,FALSE)) && !is_null($args))
|
|
|
+ return call_user_func_array(
|
|
|
+ array($this,'format'),
|
|
|
+ array_merge(array($val),is_array($args)?$args:array($args))
|
|
|
+ );
|
|
|
+ if (is_null($val)) {
|
|
|
+ // Attempt to retrieve from cache
|
|
|
+ if (Cache::instance()->exists($this->hash($key).'.var',$data))
|
|
|
+ return $data;
|
|
|
+ }
|
|
|
+ return $val;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Unset hive key
|
|
|
+ * @return NULL
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function clear($key) {
|
|
|
+ // Normalize array literal
|
|
|
+ $cache=Cache::instance();
|
|
|
+ $parts=$this->cut($key);
|
|
|
+ if ($key=='CACHE')
|
|
|
+ // Clear cache contents
|
|
|
+ $cache->reset();
|
|
|
+ elseif (preg_match('/^(GET|POST|COOKIE)\b(.+)/',$key,$expr)) {
|
|
|
+ $this->clear('REQUEST'.$expr[2]);
|
|
|
+ if ($expr[1]=='COOKIE') {
|
|
|
+ $parts=$this->cut($key);
|
|
|
+ $jar=$this->hive['JAR'];
|
|
|
+ $jar['expire']=strtotime('-1 year');
|
|
|
+ call_user_func_array('setcookie',
|
|
|
+ array_merge(array($parts[1],''),$jar));
|
|
|
+ unset($_COOKIE[$parts[1]]);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ elseif ($parts[0]=='SESSION') {
|
|
|
+ @session_start();
|
|
|
+ if (empty($parts[1])) {
|
|
|
+ // End session
|
|
|
+ session_unset();
|
|
|
+ session_destroy();
|
|
|
+ unset($_COOKIE[session_name()]);
|
|
|
+ header_remove('Set-Cookie');
|
|
|
+ }
|
|
|
+ $this->sync('SESSION');
|
|
|
+ }
|
|
|
+ if (!isset($parts[1]) && array_key_exists($parts[0],$this->init))
|
|
|
+ // Reset global to default value
|
|
|
+ $this->hive[$parts[0]]=$this->init[$parts[0]];
|
|
|
+ else {
|
|
|
+ eval('unset('.$this->compile('@this->hive.'.$key).');');
|
|
|
+ if ($parts[0]=='SESSION') {
|
|
|
+ session_commit();
|
|
|
+ session_start();
|
|
|
+ }
|
|
|
+ if ($cache->exists($hash=$this->hash($key).'.var'))
|
|
|
+ // Remove from cache
|
|
|
+ $cache->clear($hash);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Multi-variable assignment using associative array
|
|
|
+ * @return NULL
|
|
|
+ * @param $vars array
|
|
|
+ * @param $prefix string
|
|
|
+ * @param $ttl int
|
|
|
+ **/
|
|
|
+ function mset(array $vars,$prefix='',$ttl=0) {
|
|
|
+ foreach ($vars as $key=>$val)
|
|
|
+ $this->set($prefix.$key,$val,$ttl);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Publish hive contents
|
|
|
+ * @return array
|
|
|
+ **/
|
|
|
+ function hive() {
|
|
|
+ return $this->hive;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Copy contents of hive variable to another
|
|
|
+ * @return mixed
|
|
|
+ * @param $src string
|
|
|
+ * @param $dst string
|
|
|
+ **/
|
|
|
+ function copy($src,$dst) {
|
|
|
+ $ref=&$this->ref($dst);
|
|
|
+ return $ref=$this->ref($src,FALSE);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Concatenate string to hive string variable
|
|
|
+ * @return string
|
|
|
+ * @param $key string
|
|
|
+ * @param $val string
|
|
|
+ **/
|
|
|
+ function concat($key,$val) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ $ref.=$val;
|
|
|
+ return $ref;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Swap keys and values of hive array variable
|
|
|
+ * @return array
|
|
|
+ * @param $key string
|
|
|
+ * @public
|
|
|
+ **/
|
|
|
+ function flip($key) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ return $ref=array_combine(array_values($ref),array_keys($ref));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Add element to the end of hive array variable
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ **/
|
|
|
+ function push($key,$val) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ array_push($ref,$val);
|
|
|
+ return $val;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Remove last element of hive array variable
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function pop($key) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ return array_pop($ref);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Add element to the beginning of hive array variable
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ **/
|
|
|
+ function unshift($key,$val) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ array_unshift($ref,$val);
|
|
|
+ return $val;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Remove first element of hive array variable
|
|
|
+ * @return mixed
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function shift($key) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ return array_shift($ref);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Merge array with hive array variable
|
|
|
+ * @return array
|
|
|
+ * @param $key string
|
|
|
+ * @param $src string|array
|
|
|
+ **/
|
|
|
+ function merge($key,$src) {
|
|
|
+ $ref=&$this->ref($key);
|
|
|
+ return array_merge($ref,is_string($src)?$this->hive[$src]:$src);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert backslashes to slashes
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function fixslashes($str) {
|
|
|
+ return $str?strtr($str,'\\','/'):$str;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Split comma-, semi-colon, or pipe-separated string
|
|
|
+ * @return array
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function split($str) {
|
|
|
+ return array_map('trim',
|
|
|
+ preg_split('/[,;|]/',$str,0,PREG_SPLIT_NO_EMPTY));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert PHP expression/value to compressed exportable string
|
|
|
+ * @return string
|
|
|
+ * @param $arg mixed
|
|
|
+ * @param $stack array
|
|
|
+ **/
|
|
|
+ function stringify($arg,array $stack=NULL) {
|
|
|
+ if ($stack) {
|
|
|
+ foreach ($stack as $node)
|
|
|
+ if ($arg===$node)
|
|
|
+ return '*RECURSION*';
|
|
|
+ }
|
|
|
+ else
|
|
|
+ $stack=array();
|
|
|
+ switch (gettype($arg)) {
|
|
|
+ case 'object':
|
|
|
+ $str='';
|
|
|
+ foreach (get_object_vars($arg) as $key=>$val)
|
|
|
+ $str.=($str?',':'').
|
|
|
+ var_export($key,TRUE).'=>'.
|
|
|
+ $this->stringify($val,
|
|
|
+ array_merge($stack,array($arg)));
|
|
|
+ return get_class($arg).'::__set_state(array('.$str.'))';
|
|
|
+ case 'array':
|
|
|
+ $str='';
|
|
|
+ $num=isset($arg[0]) &&
|
|
|
+ ctype_digit(implode('',array_keys($arg)));
|
|
|
+ foreach ($arg as $key=>$val)
|
|
|
+ $str.=($str?',':'').
|
|
|
+ ($num?'':(var_export($key,TRUE).'=>')).
|
|
|
+ $this->stringify($val,
|
|
|
+ array_merge($stack,array($arg)));
|
|
|
+ return 'array('.$str.')';
|
|
|
+ default:
|
|
|
+ return var_export($arg,TRUE);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Flatten array values and return as CSV string
|
|
|
+ * @return string
|
|
|
+ * @param $args array
|
|
|
+ **/
|
|
|
+ function csv(array $args) {
|
|
|
+ return implode(',',array_map('stripcslashes',
|
|
|
+ array_map(array($this,'stringify'),$args)));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert snakecase string to camelcase
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function camelcase($str) {
|
|
|
+ return preg_replace_callback(
|
|
|
+ '/_(\w)/',
|
|
|
+ function($match) {
|
|
|
+ return strtoupper($match[1]);
|
|
|
+ },
|
|
|
+ $str
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert camelcase string to snakecase
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function snakecase($str) {
|
|
|
+ return strtolower(preg_replace('/[[:upper:]]/','_\0',$str));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return -1 if specified number is negative, 0 if zero,
|
|
|
+ * or 1 if the number is positive
|
|
|
+ * @return int
|
|
|
+ * @param $num mixed
|
|
|
+ **/
|
|
|
+ function sign($num) {
|
|
|
+ return $num?($num/abs($num)):0;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Generate 64bit/base36 hash
|
|
|
+ * @return string
|
|
|
+ * @param $str
|
|
|
+ **/
|
|
|
+ function hash($str) {
|
|
|
+ return str_pad(base_convert(
|
|
|
+ hexdec(substr(sha1($str),-16)),10,36),11,'0',STR_PAD_LEFT);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return Base64-encoded equivalent
|
|
|
+ * @return string
|
|
|
+ * @param $data string
|
|
|
+ * @param $mime string
|
|
|
+ **/
|
|
|
+ function base64($data,$mime) {
|
|
|
+ return 'data:'.$mime.';base64,'.base64_encode($data);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert special characters to HTML entities
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function encode($str) {
|
|
|
+ return @htmlentities($str,$this->hive['BITMASK'],
|
|
|
+ $this->hive['ENCODING'],FALSE)?:$this->scrub($str);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert HTML entities back to characters
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function decode($str) {
|
|
|
+ return html_entity_decode($str,$this->hive['BITMASK'],
|
|
|
+ $this->hive['ENCODING']);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Invoke callback recursively for all data types
|
|
|
+ * @return mixed
|
|
|
+ * @param $arg mixed
|
|
|
+ * @param $func callback
|
|
|
+ * @param $stack array
|
|
|
+ **/
|
|
|
+ function recursive($arg,$func,$stack=NULL) {
|
|
|
+ if ($stack) {
|
|
|
+ foreach ($stack as $node)
|
|
|
+ if ($arg===$node)
|
|
|
+ return $arg;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ $stack=array();
|
|
|
+ switch (gettype($arg)) {
|
|
|
+ case 'object':
|
|
|
+ if (method_exists('ReflectionClass','iscloneable')) {
|
|
|
+ $ref=new ReflectionClass($arg);
|
|
|
+ if ($ref->iscloneable()) {
|
|
|
+ $arg=clone($arg);
|
|
|
+ foreach (get_object_vars($arg) as $key=>$val)
|
|
|
+ $arg->$key=$this->recursive($val,$func,
|
|
|
+ array_merge($stack,array($arg)));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return $arg;
|
|
|
+ case 'array':
|
|
|
+ $tmp=array();
|
|
|
+ foreach ($arg as $key=>$val)
|
|
|
+ $tmp[$key]=$this->recursive($val,$func,
|
|
|
+ array_merge($stack,array($arg)));
|
|
|
+ return $tmp;
|
|
|
+ }
|
|
|
+ return $func($arg);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Remove HTML tags (except those enumerated) and non-printable
|
|
|
+ * characters to mitigate XSS/code injection attacks
|
|
|
+ * @return mixed
|
|
|
+ * @param $arg mixed
|
|
|
+ * @param $tags string
|
|
|
+ **/
|
|
|
+ function clean($arg,$tags=NULL) {
|
|
|
+ $fw=$this;
|
|
|
+ return $this->recursive($arg,
|
|
|
+ function($val) use($fw,$tags) {
|
|
|
+ if ($tags!='*')
|
|
|
+ $val=trim(strip_tags($val,
|
|
|
+ '<'.implode('><',$fw->split($tags)).'>'));
|
|
|
+ return trim(preg_replace(
|
|
|
+ '/[\x00-\x08\x0B\x0C\x0E-\x1F]/','',$val));
|
|
|
+ }
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Similar to clean(), except that variable is passed by reference
|
|
|
+ * @return mixed
|
|
|
+ * @param $var mixed
|
|
|
+ * @param $tags string
|
|
|
+ **/
|
|
|
+ function scrub(&$var,$tags=NULL) {
|
|
|
+ return $var=$this->clean($var,$tags);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return locale-aware formatted string
|
|
|
+ * @return string
|
|
|
+ **/
|
|
|
+ function format() {
|
|
|
+ $args=func_get_args();
|
|
|
+ $val=array_shift($args);
|
|
|
+ // Get formatting rules
|
|
|
+ $conv=localeconv();
|
|
|
+ return preg_replace_callback(
|
|
|
+ '/\{(?P<pos>\d+)\s*(?:,\s*(?P<type>\w+)\s*'.
|
|
|
+ '(?:,\s*(?P<mod>(?:\w+(?:\s*\{.+?\}\s*,?)?)*)'.
|
|
|
+ '(?:,\s*(?P<prop>.+?))?)?)?\}/',
|
|
|
+ function($expr) use($args,$conv) {
|
|
|
+ extract($expr);
|
|
|
+ extract($conv);
|
|
|
+ if (!array_key_exists($pos,$args))
|
|
|
+ return $expr[0];
|
|
|
+ if (isset($type))
|
|
|
+ switch ($type) {
|
|
|
+ case 'plural':
|
|
|
+ preg_match_all('/(?<tag>\w+)'.
|
|
|
+ '(?:\s+\{\s*(?<data>.+?)\s*\})/',
|
|
|
+ $mod,$matches,PREG_SET_ORDER);
|
|
|
+ $ord=array('zero','one','two');
|
|
|
+ foreach ($matches as $match) {
|
|
|
+ extract($match);
|
|
|
+ if (isset($ord[$args[$pos]]) &&
|
|
|
+ $tag==$ord[$args[$pos]] || $tag=='other')
|
|
|
+ return str_replace('#',$args[$pos],$data);
|
|
|
+ }
|
|
|
+ case 'number':
|
|
|
+ if (isset($mod))
|
|
|
+ switch ($mod) {
|
|
|
+ case 'integer':
|
|
|
+ return number_format(
|
|
|
+ $args[$pos],0,'',$thousands_sep);
|
|
|
+ case 'currency':
|
|
|
+ if (function_exists('money_format'))
|
|
|
+ return money_format(
|
|
|
+ '%n',$args[$pos]);
|
|
|
+ $fmt=array(
|
|
|
+ 0=>'(nc)',1=>'(n c)',
|
|
|
+ 2=>'(nc)',10=>'+nc',
|
|
|
+ 11=>'+n c',12=>'+ nc',
|
|
|
+ 20=>'nc+',21=>'n c+',
|
|
|
+ 22=>'nc +',30=>'n+c',
|
|
|
+ 31=>'n +c',32=>'n+ c',
|
|
|
+ 40=>'nc+',41=>'n c+',
|
|
|
+ 42=>'nc +',100=>'(cn)',
|
|
|
+ 101=>'(c n)',102=>'(cn)',
|
|
|
+ 110=>'+cn',111=>'+c n',
|
|
|
+ 112=>'+ cn',120=>'cn+',
|
|
|
+ 121=>'c n+',122=>'cn +',
|
|
|
+ 130=>'+cn',131=>'+c n',
|
|
|
+ 132=>'+ cn',140=>'c+n',
|
|
|
+ 141=>'c+ n',142=>'c +n'
|
|
|
+ );
|
|
|
+ if ($args[$pos]<0) {
|
|
|
+ $sgn=$negative_sign;
|
|
|
+ $pre='n';
|
|
|
+ }
|
|
|
+ else {
|
|
|
+ $sgn=$positive_sign;
|
|
|
+ $pre='p';
|
|
|
+ }
|
|
|
+ return str_replace(
|
|
|
+ array('+','n','c'),
|
|
|
+ array($sgn,number_format(
|
|
|
+ abs($args[$pos]),
|
|
|
+ $frac_digits,
|
|
|
+ $decimal_point,
|
|
|
+ $thousands_sep),
|
|
|
+ $currency_symbol),
|
|
|
+ $fmt[(int)(
|
|
|
+ (${$pre.'_cs_precedes'}%2).
|
|
|
+ (${$pre.'_sign_posn'}%5).
|
|
|
+ (${$pre.'_sep_by_space'}%3)
|
|
|
+ )]
|
|
|
+ );
|
|
|
+ case 'percent':
|
|
|
+ return number_format(
|
|
|
+ $args[$pos]*100,0,$decimal_point,
|
|
|
+ $thousands_sep).'%';
|
|
|
+ case 'decimal':
|
|
|
+ return number_format(
|
|
|
+ $args[$pos],$prop,$decimal_point,
|
|
|
+ $thousands_sep);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case 'date':
|
|
|
+ if (empty($mod) || $mod=='short')
|
|
|
+ $prop='%x';
|
|
|
+ elseif ($mod=='long')
|
|
|
+ $prop='%A, %d %B %Y';
|
|
|
+ return strftime($prop,$args[$pos]);
|
|
|
+ case 'time':
|
|
|
+ if (empty($mod) || $mod=='short')
|
|
|
+ $prop='%X';
|
|
|
+ return strftime($prop,$args[$pos]);
|
|
|
+ default:
|
|
|
+ return $expr[0];
|
|
|
+ }
|
|
|
+ return $args[$pos];
|
|
|
+ },
|
|
|
+ $val
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Assign/auto-detect language
|
|
|
+ * @return string
|
|
|
+ * @param $code string
|
|
|
+ **/
|
|
|
+ function language($code) {
|
|
|
+ $code=preg_replace('/;q=.+?(?=,|$)/','',$code);
|
|
|
+ $code.=($code?',':'').$this->fallback;
|
|
|
+ $this->languages=array();
|
|
|
+ foreach (array_reverse(explode(',',$code)) as $lang) {
|
|
|
+ if (preg_match('/^(\w{2})(?:-(\w{2}))?\b/i',$lang,$parts)) {
|
|
|
+ // Generic language
|
|
|
+ array_unshift($this->languages,$parts[1]);
|
|
|
+ if (isset($parts[2])) {
|
|
|
+ // Specific language
|
|
|
+ $parts[0]=$parts[1].'-'.($parts[2]=strtoupper($parts[2]));
|
|
|
+ array_unshift($this->languages,$parts[0]);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ $this->languages=array_unique($this->languages);
|
|
|
+ $locales=array();
|
|
|
+ $windows=preg_match('/^win/i',PHP_OS);
|
|
|
+ foreach ($this->languages as $locale) {
|
|
|
+ if ($windows) {
|
|
|
+ $parts=explode('-',$locale);
|
|
|
+ $locale=@constant('ISO::LC_'.$parts[0]);
|
|
|
+ if (isset($parts[1]) &&
|
|
|
+ $country=@constant('ISO::CC_'.strtolower($parts[1])))
|
|
|
+ $locale.='-'.$country;
|
|
|
+ }
|
|
|
+ $locales[]=$locale;
|
|
|
+ $locales[]=$locale.'.'.ini_get('default_charset');
|
|
|
+ }
|
|
|
+ setlocale(LC_ALL,str_replace('-','_',$locales));
|
|
|
+ return implode(',',$this->languages);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Transfer lexicon entries to hive
|
|
|
+ * @return array
|
|
|
+ * @param $path string
|
|
|
+ **/
|
|
|
+ function lexicon($path) {
|
|
|
+ $lex=array();
|
|
|
+ foreach ($this->languages?:array($this->fallback) as $lang) {
|
|
|
+ if ((is_file($file=($base=$path.$lang).'.php') ||
|
|
|
+ is_file($file=$base.'.php')) &&
|
|
|
+ is_array($dict=require($file)))
|
|
|
+ $lex+=$dict;
|
|
|
+ elseif (is_file($file=$base.'.ini')) {
|
|
|
+ preg_match_all(
|
|
|
+ '/(?<=^|\n)(?:'.
|
|
|
+ '(.+?)\h*=\h*'.
|
|
|
+ '((?:\\\\\h*\r?\n|.+?)*)'.
|
|
|
+ ')(?=\r?\n|$)/',
|
|
|
+ $this->read($file),$matches,PREG_SET_ORDER);
|
|
|
+ if ($matches)
|
|
|
+ foreach ($matches as $match)
|
|
|
+ if (isset($match[1]) &&
|
|
|
+ !array_key_exists($match[1],$lex))
|
|
|
+ $lex[$match[1]]=trim(preg_replace(
|
|
|
+ '/(?<!\\\\)"|\\\\\h*\r?\n/','',$match[2]));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return $lex;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return string representation of PHP value
|
|
|
+ * @return string
|
|
|
+ * @param $arg mixed
|
|
|
+ **/
|
|
|
+ function serialize($arg) {
|
|
|
+ switch (strtolower($this->hive['SERIALIZER'])) {
|
|
|
+ case 'igbinary':
|
|
|
+ return igbinary_serialize($arg);
|
|
|
+ default:
|
|
|
+ return serialize($arg);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return PHP value derived from string
|
|
|
+ * @return string
|
|
|
+ * @param $arg mixed
|
|
|
+ **/
|
|
|
+ function unserialize($arg) {
|
|
|
+ switch (strtolower($this->hive['SERIALIZER'])) {
|
|
|
+ case 'igbinary':
|
|
|
+ return igbinary_unserialize($arg);
|
|
|
+ default:
|
|
|
+ return unserialize($arg);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Send HTTP/1.1 status header; Return text equivalent of status code
|
|
|
+ * @return string
|
|
|
+ * @param $code int
|
|
|
+ **/
|
|
|
+ function status($code) {
|
|
|
+ $reason=@constant('self::HTTP_'.$code);
|
|
|
+ if (PHP_SAPI!='cli')
|
|
|
+ header('HTTP/1.1 '.$code.' '.$reason);
|
|
|
+ return $reason;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Send cache metadata to HTTP client
|
|
|
+ * @return NULL
|
|
|
+ * @param $secs int
|
|
|
+ **/
|
|
|
+ function expire($secs=0) {
|
|
|
+ if (PHP_SAPI!='cli') {
|
|
|
+ header('X-Content-Type-Options: nosniff');
|
|
|
+ header('X-Frame-Options: '.$this->hive['XFRAME']);
|
|
|
+ header('X-Powered-By: '.$this->hive['PACKAGE']);
|
|
|
+ header('X-XSS-Protection: 1; mode=block');
|
|
|
+ if ($secs) {
|
|
|
+ $time=microtime(TRUE);
|
|
|
+ header_remove('Pragma');
|
|
|
+ header('Expires: '.gmdate('r',$time+$secs));
|
|
|
+ header('Cache-Control: max-age='.$secs);
|
|
|
+ header('Last-Modified: '.gmdate('r'));
|
|
|
+ $headers=$this->hive['HEADERS'];
|
|
|
+ if (isset($headers['If-Modified-Since']) &&
|
|
|
+ strtotime($headers['If-Modified-Since'])+$secs>$time) {
|
|
|
+ $this->status(304);
|
|
|
+ die;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ header('Cache-Control: no-cache, no-store, must-revalidate');
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Log error; Execute ONERROR handler if defined, else display
|
|
|
+ * default error page (HTML for synchronous requests, JSON string
|
|
|
+ * for AJAX requests)
|
|
|
+ * @return NULL
|
|
|
+ * @param $code int
|
|
|
+ * @param $text string
|
|
|
+ * @param $trace array
|
|
|
+ **/
|
|
|
+ function error($code,$text='',array $trace=NULL) {
|
|
|
+ $prior=$this->hive['ERROR'];
|
|
|
+ $header=$this->status($code);
|
|
|
+ $req=$this->hive['VERB'].' '.$this->hive['PATH'];
|
|
|
+ if (!$text)
|
|
|
+ $text='HTTP '.$code.' ('.$req.')';
|
|
|
+ error_log($text);
|
|
|
+ if (!$trace)
|
|
|
+ $trace=array_slice(debug_backtrace(FALSE),1);
|
|
|
+ $debug=$this->hive['DEBUG'];
|
|
|
+ $trace=array_filter(
|
|
|
+ $trace,
|
|
|
+ function($frame) use($debug) {
|
|
|
+ return $debug && isset($frame['file']) &&
|
|
|
+ ($frame['file']!=__FILE__ || $debug>1) &&
|
|
|
+ (empty($frame['function']) ||
|
|
|
+ !preg_match('/^(?:(?:trigger|user)_error|'.
|
|
|
+ '__call|call_user_func)/',$frame['function']));
|
|
|
+ }
|
|
|
+ );
|
|
|
+ $highlight=PHP_SAPI!='cli' &&
|
|
|
+ $this->hive['HIGHLIGHT'] && is_file($css=__DIR__.'/'.self::CSS);
|
|
|
+ $out='';
|
|
|
+ $eol="\n";
|
|
|
+ // Analyze stack trace
|
|
|
+ foreach ($trace as $frame) {
|
|
|
+ $line='';
|
|
|
+ if (isset($frame['class']))
|
|
|
+ $line.=$frame['class'].$frame['type'];
|
|
|
+ if (isset($frame['function']))
|
|
|
+ $line.=$frame['function'].'('.
|
|
|
+ ($debug>2 && isset($frame['args'])?
|
|
|
+ $this->csv($frame['args']):'').')';
|
|
|
+ $src=$this->fixslashes(str_replace($_SERVER['DOCUMENT_ROOT'].
|
|
|
+ '/','',$frame['file'])).':'.$frame['line'].' ';
|
|
|
+ error_log('- '.$src.$line);
|
|
|
+ $out.='• '.($highlight?
|
|
|
+ ($this->highlight($src).' '.$this->highlight($line)):
|
|
|
+ ($src.$line)).$eol;
|
|
|
+ }
|
|
|
+ $this->hive['ERROR']=array(
|
|
|
+ 'status'=>$header,
|
|
|
+ 'code'=>$code,
|
|
|
+ 'text'=>$text,
|
|
|
+ 'trace'=>$trace
|
|
|
+ );
|
|
|
+ $handler=$this->hive['ONERROR'];
|
|
|
+ $this->hive['ONERROR']=NULL;
|
|
|
+ if ((!$handler ||
|
|
|
+ $this->call($handler,$this,'beforeroute,afterroute')===FALSE) &&
|
|
|
+ !$prior && PHP_SAPI!='cli' && !$this->hive['QUIET'])
|
|
|
+ echo $this->hive['AJAX']?
|
|
|
+ json_encode($this->hive['ERROR']):
|
|
|
+ ('<!DOCTYPE html>'.$eol.
|
|
|
+ '<html>'.$eol.
|
|
|
+ '<head>'.
|
|
|
+ '<title>'.$code.' '.$header.'</title>'.
|
|
|
+ ($highlight?
|
|
|
+ ('<style>'.$this->read($css).'</style>'):'').
|
|
|
+ '</head>'.$eol.
|
|
|
+ '<body>'.$eol.
|
|
|
+ '<h1>'.$header.'</h1>'.$eol.
|
|
|
+ '<p>'.$this->encode($text?:$req).'</p>'.$eol.
|
|
|
+ ($debug?('<pre>'.$out.'</pre>'.$eol):'').
|
|
|
+ '</body>'.$eol.
|
|
|
+ '</html>');
|
|
|
+ if ($this->hive['HALT'])
|
|
|
+ die;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Mock HTTP request
|
|
|
+ * @return NULL
|
|
|
+ * @param $pattern string
|
|
|
+ * @param $args array
|
|
|
+ * @param $headers array
|
|
|
+ * @param $body string
|
|
|
+ **/
|
|
|
+ function mock($pattern,array $args=NULL,array $headers=NULL,$body=NULL) {
|
|
|
+ $types=array('sync','ajax');
|
|
|
+ preg_match('/([\|\w]+)\h+(?:@(\w+)(?:(\(.+?)\))*|([^\h]+))'.
|
|
|
+ '(?:\h+\[('.implode('|',$types).')\])?/',$pattern,$parts);
|
|
|
+ $verb=strtoupper($parts[1]);
|
|
|
+ if ($parts[2]) {
|
|
|
+ if (empty($this->hive['ALIASES'][$parts[2]]))
|
|
|
+ user_error(sprintf(self::E_Named,$parts[2]));
|
|
|
+ $parts[4]=$this->hive['ALIASES'][$parts[2]];
|
|
|
+ if (isset($parts[3]))
|
|
|
+ $this->parse($parts[3]);
|
|
|
+ $parts[4]=$this->build($parts[4]);
|
|
|
+ }
|
|
|
+ if (empty($parts[4]))
|
|
|
+ user_error(sprintf(self::E_Pattern,$pattern));
|
|
|
+ $url=parse_url($parts[4]);
|
|
|
+ $query='';
|
|
|
+ if ($args)
|
|
|
+ $query.=http_build_query($args);
|
|
|
+ $query.=isset($url['query'])?(($query?'&':'').$url['query']):'';
|
|
|
+ if ($query && preg_match('/GET|POST/',$verb)) {
|
|
|
+ parse_str($query,$GLOBALS['_'.$verb]);
|
|
|
+ parse_str($query,$GLOBALS['_REQUEST']);
|
|
|
+ }
|
|
|
+ foreach ($headers?:array() as $key=>$val)
|
|
|
+ $_SERVER['HTTP_'.strtr(strtoupper($key),'-','_')]=$val;
|
|
|
+ $this->hive['VERB']=$verb;
|
|
|
+ $this->hive['URI']=$this->hive['BASE'].$url['path'];
|
|
|
+ $this->hive['AJAX']=isset($parts[5]) &&
|
|
|
+ preg_match('/ajax/i',$parts[5]);
|
|
|
+ if (preg_match('/GET|HEAD/',$verb) && $query)
|
|
|
+ $this->hive['URI'].='?'.$query;
|
|
|
+ else
|
|
|
+ $this->hive['BODY']=$body?:$query;
|
|
|
+ $this->run();
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Bind handler to route pattern
|
|
|
+ * @return NULL
|
|
|
+ * @param $pattern string|array
|
|
|
+ * @param $handler callback
|
|
|
+ * @param $ttl int
|
|
|
+ * @param $kbps int
|
|
|
+ **/
|
|
|
+ function route($pattern,$handler,$ttl=0,$kbps=0) {
|
|
|
+ $types=array('sync','ajax');
|
|
|
+ if (is_array($pattern)) {
|
|
|
+ foreach ($pattern as $item)
|
|
|
+ $this->route($item,$handler,$ttl,$kbps);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ preg_match('/([\|\w]+)\h+(?:(?:@(\w+)\h*:\h*)?([^\h]+)|@(\w+))'.
|
|
|
+ '(?:\h+\[('.implode('|',$types).')\])?/',$pattern,$parts);
|
|
|
+ if ($parts[2])
|
|
|
+ $this->hive['ALIASES'][$parts[2]]=$parts[3];
|
|
|
+ elseif (!empty($parts[4])) {
|
|
|
+ if (empty($this->hive['ALIASES'][$parts[4]]))
|
|
|
+ user_error(sprintf(self::E_Named,$parts[4]));
|
|
|
+ $parts[3]=$this->hive['ALIASES'][$parts[4]];
|
|
|
+ }
|
|
|
+ if (empty($parts[3]))
|
|
|
+ user_error(sprintf(self::E_Pattern,$pattern));
|
|
|
+ $type=empty($parts[5])?
|
|
|
+ self::REQ_SYNC|self::REQ_AJAX:
|
|
|
+ constant('self::REQ_'.strtoupper($parts[5]));
|
|
|
+ foreach ($this->split($parts[1]) as $verb) {
|
|
|
+ if (!preg_match('/'.self::VERBS.'/',$verb))
|
|
|
+ $this->error(501,$verb.' '.$this->hive['URI']);
|
|
|
+ $this->hive['ROUTES'][str_replace('@',"\x00".'@',$parts[3])]
|
|
|
+ [$type][strtoupper($verb)]=array($handler,$ttl,$kbps);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Reroute to specified URI
|
|
|
+ * @return NULL
|
|
|
+ * @param $url string
|
|
|
+ * @param $permanent bool
|
|
|
+ **/
|
|
|
+ function reroute($url,$permanent=FALSE) {
|
|
|
+ if (PHP_SAPI!='cli') {
|
|
|
+ if (preg_match('/^(?:@(\w+)(?:(\(.+?)\))*|https?:\/\/)/',
|
|
|
+ $url,$parts)) {
|
|
|
+ if (isset($parts[1])) {
|
|
|
+ if (empty($this->hive['ALIASES'][$parts[1]]))
|
|
|
+ user_error(sprintf(self::E_Named,$parts[1]));
|
|
|
+ $url=$this->hive['BASE'].
|
|
|
+ $this->hive['ALIASES'][$parts[1]];
|
|
|
+ if (isset($parts[2]))
|
|
|
+ $this->parse($parts[2]);
|
|
|
+ $url=$this->build($url);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ $url=$this->hive['BASE'].$url;
|
|
|
+ header('Location: '.$url);
|
|
|
+ $this->status($permanent?301:302);
|
|
|
+ die;
|
|
|
+ }
|
|
|
+ $this->mock('GET '.$url);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Provide ReST interface by mapping HTTP verb to class method
|
|
|
+ * @return NULL
|
|
|
+ * @param $url string
|
|
|
+ * @param $class string
|
|
|
+ * @param $ttl int
|
|
|
+ * @param $kbps int
|
|
|
+ **/
|
|
|
+ function map($url,$class,$ttl=0,$kbps=0) {
|
|
|
+ if (is_array($url)) {
|
|
|
+ foreach ($url as $item)
|
|
|
+ $this->map($item,$class,$ttl,$kbps);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ foreach (explode('|',self::VERBS) as $method)
|
|
|
+ $this->route($method.' '.$url,
|
|
|
+ $class.'->'.strtolower($method),$ttl,$kbps);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Function to redirect a route to another url or route
|
|
|
+ * @return NULL
|
|
|
+ * @param $pattern string|array
|
|
|
+ * @param $url string
|
|
|
+ */
|
|
|
+ function redirect($pattern,$url) {
|
|
|
+ if (is_array($pattern)) {
|
|
|
+ foreach ($pattern as $item)
|
|
|
+ $this->redirect($item,$url);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ $this->route($pattern, function($this) use ($url) { $this->reroute($url); });
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return TRUE if IPv4 address exists in DNSBL
|
|
|
+ * @return bool
|
|
|
+ * @param $ip string
|
|
|
+ **/
|
|
|
+ function blacklisted($ip) {
|
|
|
+ if ($this->hive['DNSBL'] &&
|
|
|
+ !in_array($ip,
|
|
|
+ is_array($this->hive['EXEMPT'])?
|
|
|
+ $this->hive['EXEMPT']:
|
|
|
+ $this->split($this->hive['EXEMPT']))) {
|
|
|
+ // Reverse IPv4 dotted quad
|
|
|
+ $rev=implode('.',array_reverse(explode('.',$ip)));
|
|
|
+ foreach (is_array($this->hive['DNSBL'])?
|
|
|
+ $this->hive['DNSBL']:
|
|
|
+ $this->split($this->hive['DNSBL']) as $server)
|
|
|
+ // DNSBL lookup
|
|
|
+ if (checkdnsrr($rev.'.'.$server,'A'))
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Match routes against incoming URI
|
|
|
+ * @return NULL
|
|
|
+ **/
|
|
|
+ function run() {
|
|
|
+ if ($this->blacklisted($this->hive['IP']))
|
|
|
+ // Spammer detected
|
|
|
+ $this->error(403);
|
|
|
+ if (!$this->hive['ROUTES'])
|
|
|
+ // No routes defined
|
|
|
+ user_error(self::E_Routes);
|
|
|
+ // Match specific routes first
|
|
|
+ krsort($this->hive['ROUTES']);
|
|
|
+ // Convert to BASE-relative URL
|
|
|
+ $req=preg_replace(
|
|
|
+ '/^'.preg_quote($this->hive['BASE'],'/').'(\/.*|$)/','\1',
|
|
|
+ $this->hive['URI']
|
|
|
+ );
|
|
|
+ $allowed=array();
|
|
|
+ $case=$this->hive['CASELESS']?'i':'';
|
|
|
+ foreach ($this->hive['ROUTES'] as $url=>$routes) {
|
|
|
+ $url=str_replace("\x00".'@','@',$url);
|
|
|
+ if (!preg_match('/^'.
|
|
|
+ preg_replace('/@(\w+\b)/','(?P<\1>[^\/\?]+)',
|
|
|
+ str_replace('\*','([^\?]*)',preg_quote($url,'/'))).
|
|
|
+ '(?:[\?\/].*)?$/'.$case.'um',$req,$args))
|
|
|
+ continue;
|
|
|
+ $route=NULL;
|
|
|
+ if (isset($routes[$this->hive['AJAX']+1]))
|
|
|
+ $route=$routes[$this->hive['AJAX']+1];
|
|
|
+ elseif (isset($routes[self::REQ_SYNC|self::REQ_AJAX]))
|
|
|
+ $route=$routes[self::REQ_SYNC|self::REQ_AJAX];
|
|
|
+ if (!$route)
|
|
|
+ continue;
|
|
|
+ if ($this->hive['VERB']!='OPTIONS' &&
|
|
|
+ isset($route[$this->hive['VERB']])) {
|
|
|
+ $parts=parse_url($req);
|
|
|
+ if ($this->hive['VERB']=='GET' &&
|
|
|
+ preg_match('/.+\/$/',$parts['path']))
|
|
|
+ $this->reroute(substr($parts['path'],0,-1).
|
|
|
+ (isset($parts['query'])?('?'.$parts['query']):''));
|
|
|
+ list($handler,$ttl,$kbps)=$route[$this->hive['VERB']];
|
|
|
+ if (is_bool(strpos($url,'/*')))
|
|
|
+ foreach (array_keys($args) as $key)
|
|
|
+ if (is_numeric($key) && $key)
|
|
|
+ unset($args[$key]);
|
|
|
+ if (is_string($handler)) {
|
|
|
+ // Replace route pattern tokens in handler if any
|
|
|
+ $handler=preg_replace_callback('/@(\w+\b)/',
|
|
|
+ function($id) use($args) {
|
|
|
+ return isset($args[$id[1]])?$args[$id[1]]:$id[0];
|
|
|
+ },
|
|
|
+ $handler
|
|
|
+ );
|
|
|
+ if (preg_match('/(.+)\h*(?:->|::)/',$handler,$match) &&
|
|
|
+ !class_exists($match[1]))
|
|
|
+ $this->error(404);
|
|
|
+ }
|
|
|
+ // Capture values of route pattern tokens
|
|
|
+ $this->hive['PARAMS']=$args=array_map('urldecode',$args);
|
|
|
+ // Save matching route
|
|
|
+ $this->hive['PATTERN']=$url;
|
|
|
+ // Process request
|
|
|
+ $body='';
|
|
|
+ $now=microtime(TRUE);
|
|
|
+ if (preg_match('/GET|HEAD/',$this->hive['VERB']) &&
|
|
|
+ isset($ttl)) {
|
|
|
+ // Only GET and HEAD requests are cacheable
|
|
|
+ $headers=$this->hive['HEADERS'];
|
|
|
+ $cache=Cache::instance();
|
|
|
+ $cached=$cache->exists(
|
|
|
+ $hash=$this->hash($this->hive['VERB'].' '.
|
|
|
+ $this->hive['URI']).'.url',$data);
|
|
|
+ if ($cached && $cached[0]+$ttl>$now) {
|
|
|
+ // Retrieve from cache backend
|
|
|
+ list($headers,$body)=$data;
|
|
|
+ if (PHP_SAPI!='cli')
|
|
|
+ array_walk($headers,'header');
|
|
|
+ $this->expire($cached[0]+$ttl-$now);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ // Expire HTTP client-cached page
|
|
|
+ $this->expire($ttl);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ $this->expire(0);
|
|
|
+ if (!strlen($body)) {
|
|
|
+ if (!$this->hive['RAW'] && !$this->hive['BODY'])
|
|
|
+ $this->hive['BODY']=file_get_contents('php://input');
|
|
|
+ ob_start();
|
|
|
+ // Call route handler
|
|
|
+ $this->call($handler,array($this,$args),
|
|
|
+ 'beforeroute,afterroute');
|
|
|
+ $body=ob_get_clean();
|
|
|
+ if ($ttl && !error_get_last())
|
|
|
+ // Save to cache backend
|
|
|
+ $cache->set($hash,array(headers_list(),$body),$ttl);
|
|
|
+ }
|
|
|
+ $this->hive['RESPONSE']=$body;
|
|
|
+ if (!$this->hive['QUIET']) {
|
|
|
+ if ($kbps) {
|
|
|
+ $ctr=0;
|
|
|
+ foreach (str_split($body,1024) as $part) {
|
|
|
+ // Throttle output
|
|
|
+ $ctr++;
|
|
|
+ if ($ctr/$kbps>($elapsed=microtime(TRUE)-$now) &&
|
|
|
+ !connection_aborted())
|
|
|
+ usleep(1e6*($ctr/$kbps-$elapsed));
|
|
|
+ echo $part;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ echo $body;
|
|
|
+ }
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ $allowed=array_keys($route);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (!$allowed)
|
|
|
+ // URL doesn't match any route
|
|
|
+ $this->error(404);
|
|
|
+ elseif (PHP_SAPI!='cli') {
|
|
|
+ // Unhandled HTTP method
|
|
|
+ header('Allow: '.implode(',',$allowed));
|
|
|
+ if ($this->hive['VERB']!='OPTIONS')
|
|
|
+ $this->error(405);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Execute callback/hooks (supports 'class->method' format)
|
|
|
+ * @return mixed|FALSE
|
|
|
+ * @param $func callback
|
|
|
+ * @param $args mixed
|
|
|
+ * @param $hooks string
|
|
|
+ **/
|
|
|
+ function call($func,$args=NULL,$hooks='') {
|
|
|
+ if (!is_array($args))
|
|
|
+ $args=array($args);
|
|
|
+ // Execute function; abort if callback/hook returns FALSE
|
|
|
+ if (is_string($func) &&
|
|
|
+ preg_match('/(.+)\h*(->|::)\h*(.+)/s',$func,$parts)) {
|
|
|
+ // Convert string to executable PHP callback
|
|
|
+ if (!class_exists($parts[1]))
|
|
|
+ user_error(sprintf(self::E_Class,
|
|
|
+ is_string($func)?$parts[1]:$this->stringify()));
|
|
|
+ if ($parts[2]=='->')
|
|
|
+ $parts[1]=is_subclass_of($parts[1],'Prefab')?
|
|
|
+ call_user_func($parts[1].'::instance'):
|
|
|
+ new $parts[1]($this);
|
|
|
+ $func=array($parts[1],$parts[3]);
|
|
|
+ }
|
|
|
+ if (!is_callable($func))
|
|
|
+ // No route handler
|
|
|
+ if ($hooks=='beforeroute,afterroute')
|
|
|
+ $this->error(405);
|
|
|
+ else
|
|
|
+ user_error(sprintf(self::E_Method,
|
|
|
+ is_string($func)?$func:$this->stringify($func)));
|
|
|
+ $obj=FALSE;
|
|
|
+ if (is_array($func)) {
|
|
|
+ $hooks=$this->split($hooks);
|
|
|
+ $obj=TRUE;
|
|
|
+ }
|
|
|
+ // Execute pre-route hook if any
|
|
|
+ if ($obj && $hooks && in_array($hook='beforeroute',$hooks) &&
|
|
|
+ method_exists($func[0],$hook) &&
|
|
|
+ call_user_func_array(array($func[0],$hook),$args)===FALSE)
|
|
|
+ return FALSE;
|
|
|
+ // Execute callback
|
|
|
+ $out=call_user_func_array($func,$args?:array());
|
|
|
+ if ($out===FALSE)
|
|
|
+ return FALSE;
|
|
|
+ // Execute post-route hook if any
|
|
|
+ if ($obj && $hooks && in_array($hook='afterroute',$hooks) &&
|
|
|
+ method_exists($func[0],$hook) &&
|
|
|
+ call_user_func_array(array($func[0],$hook),$args)===FALSE)
|
|
|
+ return FALSE;
|
|
|
+ return $out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Execute specified callbacks in succession; Apply same arguments
|
|
|
+ * to all callbacks
|
|
|
+ * @return array
|
|
|
+ * @param $funcs array|string
|
|
|
+ * @param $args mixed
|
|
|
+ **/
|
|
|
+ function chain($funcs,$args=NULL) {
|
|
|
+ $out=array();
|
|
|
+ foreach (is_array($funcs)?$funcs:$this->split($funcs) as $func)
|
|
|
+ $out[]=$this->call($func,$args);
|
|
|
+ return $out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Execute specified callbacks in succession; Relay result of
|
|
|
+ * previous callback as argument to the next callback
|
|
|
+ * @return array
|
|
|
+ * @param $funcs array|string
|
|
|
+ * @param $args mixed
|
|
|
+ **/
|
|
|
+ function relay($funcs,$args=NULL) {
|
|
|
+ foreach (is_array($funcs)?$funcs:$this->split($funcs) as $func)
|
|
|
+ $args=array($this->call($func,$args));
|
|
|
+ return array_shift($args);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Configure framework according to .ini-style file settings
|
|
|
+ * @return NULL
|
|
|
+ * @param $file string
|
|
|
+ **/
|
|
|
+ function config($file) {
|
|
|
+ preg_match_all(
|
|
|
+ '/(?<=^|\n)(?:'.
|
|
|
+ '\[(?<section>.+?)\]|'.
|
|
|
+ '(?<lval>[^\h\r\n;].+?)\h*=\h*'.
|
|
|
+ '(?<rval>(?:\\\\\h*\r?\n|.+?)*)'.
|
|
|
+ ')(?=\r?\n|$)/',
|
|
|
+ $this->read($file),$matches,PREG_SET_ORDER);
|
|
|
+ if ($matches) {
|
|
|
+ $sec='globals';
|
|
|
+ foreach ($matches as $match) {
|
|
|
+ if ($match['section'])
|
|
|
+ $sec=$match['section'];
|
|
|
+ elseif (in_array($sec,array('routes','maps','redirects'))) {
|
|
|
+ call_user_func_array(
|
|
|
+ array($this,rtrim($sec,'s')),
|
|
|
+ array_merge(array($match['lval']),
|
|
|
+ str_getcsv($match['rval'])));
|
|
|
+ }
|
|
|
+ else {
|
|
|
+ $args=array_map(
|
|
|
+ function($val) {
|
|
|
+ if (is_numeric($val))
|
|
|
+ return $val+0;
|
|
|
+ $val=ltrim($val);
|
|
|
+ if (preg_match('/^\w+$/i',$val) && defined($val))
|
|
|
+ return constant($val);
|
|
|
+ return preg_replace('/\\\\\h*(\r?\n)/','\1',$val);
|
|
|
+ },
|
|
|
+ // Mark quoted strings with 0x00 whitespace
|
|
|
+ str_getcsv(preg_replace('/(?<!\\\\)(")(.*?)\1/',
|
|
|
+ "\\1\x00\\2\\1",$match['rval']))
|
|
|
+ );
|
|
|
+ call_user_func_array(array($this,'set'),
|
|
|
+ array_merge(
|
|
|
+ array($match['lval']),
|
|
|
+ count($args)>1?array($args):$args));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Create mutex, invoke callback then drop ownership when done
|
|
|
+ * @return mixed
|
|
|
+ * @param $id string
|
|
|
+ * @param $func callback
|
|
|
+ * @param $args mixed
|
|
|
+ **/
|
|
|
+ function mutex($id,$func,$args=NULL) {
|
|
|
+ if (!is_dir($tmp=$this->hive['TEMP']))
|
|
|
+ mkdir($tmp,self::MODE,TRUE);
|
|
|
+ // Use filesystem lock
|
|
|
+ if (is_file($lock=$tmp.
|
|
|
+ $this->hash($this->hive['ROOT'].$this->hive['BASE']).'.'.
|
|
|
+ $this->hash($id).'.lock') &&
|
|
|
+ filemtime($lock)+ini_get('max_execution_time')<microtime(TRUE))
|
|
|
+ // Stale lock
|
|
|
+ @unlink($lock);
|
|
|
+ while (!($handle=@fopen($lock,'x')) && !connection_aborted())
|
|
|
+ usleep(mt_rand(0,100));
|
|
|
+ $out=$this->call($func,$args);
|
|
|
+ fclose($handle);
|
|
|
+ @unlink($lock);
|
|
|
+ return $out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Read file (with option to apply Unix LF as standard line ending)
|
|
|
+ * @return string
|
|
|
+ * @param $file string
|
|
|
+ * @param $lf bool
|
|
|
+ **/
|
|
|
+ function read($file,$lf=FALSE) {
|
|
|
+ $out=@file_get_contents($file);
|
|
|
+ return $lf?preg_replace('/\r\n|\r/',"\n",$out):$out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Exclusive file write
|
|
|
+ * @return int|FALSE
|
|
|
+ * @param $file string
|
|
|
+ * @param $data mixed
|
|
|
+ * @param $append bool
|
|
|
+ **/
|
|
|
+ function write($file,$data,$append=FALSE) {
|
|
|
+ return file_put_contents($file,$data,LOCK_EX|($append?FILE_APPEND:0));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Apply syntax highlighting
|
|
|
+ * @return string
|
|
|
+ * @param $text string
|
|
|
+ **/
|
|
|
+ function highlight($text) {
|
|
|
+ $out='';
|
|
|
+ $pre=FALSE;
|
|
|
+ $text=trim($text);
|
|
|
+ if (!preg_match('/^<\?php/',$text)) {
|
|
|
+ $text='<?php '.$text;
|
|
|
+ $pre=TRUE;
|
|
|
+ }
|
|
|
+ foreach (token_get_all($text) as $token)
|
|
|
+ if ($pre)
|
|
|
+ $pre=FALSE;
|
|
|
+ else
|
|
|
+ $out.='<span'.
|
|
|
+ (is_array($token)?
|
|
|
+ (' class="'.
|
|
|
+ substr(strtolower(token_name($token[0])),2).'">'.
|
|
|
+ $this->encode($token[1]).''):
|
|
|
+ ('>'.$this->encode($token))).
|
|
|
+ '</span>';
|
|
|
+ return $out?('<code>'.$out.'</code>'):$text;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Dump expression with syntax highlighting
|
|
|
+ * @return NULL
|
|
|
+ * @param $expr mixed
|
|
|
+ **/
|
|
|
+ function dump($expr) {
|
|
|
+ echo $this->highlight($this->stringify($expr));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return path relative to the base directory
|
|
|
+ * @return string
|
|
|
+ * @param $url string
|
|
|
+ **/
|
|
|
+ function rel($url) {
|
|
|
+ return preg_replace('/(?:https?:\/\/)?'.
|
|
|
+ preg_quote($this->hive['BASE'],'/').'/','',rtrim($url,'/'));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Namespace-aware class autoloader
|
|
|
+ * @return mixed
|
|
|
+ * @param $class string
|
|
|
+ **/
|
|
|
+ protected function autoload($class) {
|
|
|
+ $class=$this->fixslashes(ltrim($class,'\\'));
|
|
|
+ foreach ($this->split($this->hive['PLUGINS'].';'.
|
|
|
+ $this->hive['AUTOLOAD']) as $auto)
|
|
|
+ if (is_file($file=$auto.$class.'.php') ||
|
|
|
+ is_file($file=$auto.strtolower($class).'.php') ||
|
|
|
+ is_file($file=strtolower($auto.$class).'.php'))
|
|
|
+ return require($file);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Execute framework/application shutdown sequence
|
|
|
+ * @return NULL
|
|
|
+ * @param $cwd string
|
|
|
+ **/
|
|
|
+ function unload($cwd) {
|
|
|
+ chdir($cwd);
|
|
|
+ if (!$error=error_get_last())
|
|
|
+ @session_commit();
|
|
|
+ $handler=$this->hive['UNLOAD'];
|
|
|
+ if ((!$handler || $this->call($handler,$this)===FALSE) &&
|
|
|
+ $error && in_array($error['type'],
|
|
|
+ array(E_ERROR,E_PARSE,E_CORE_ERROR,E_COMPILE_ERROR)))
|
|
|
+ // Fatal error detected
|
|
|
+ $this->error(sprintf(self::E_Fatal,$error['message']));
|
|
|
+ }
|
|
|
+
|
|
|
+ //! Prohibit cloning
|
|
|
+ private function __clone() {
|
|
|
+ }
|
|
|
+
|
|
|
+ //! Bootstrap
|
|
|
+ function __construct() {
|
|
|
+ // Managed directives
|
|
|
+ ini_set('default_charset',$charset='UTF-8');
|
|
|
+ if (extension_loaded('mbstring'))
|
|
|
+ mb_internal_encoding($charset);
|
|
|
+ ini_set('display_errors',0);
|
|
|
+ // Deprecated directives
|
|
|
+ @ini_set('magic_quotes_gpc',0);
|
|
|
+ @ini_set('register_globals',0);
|
|
|
+ // Abort on startup error
|
|
|
+ // Intercept errors/exceptions; PHP5.3-compatible
|
|
|
+ error_reporting(E_ALL|E_STRICT);
|
|
|
+ $fw=$this;
|
|
|
+ set_exception_handler(
|
|
|
+ function($obj) use($fw) {
|
|
|
+ $fw->error(500,$obj->getmessage(),$obj->gettrace());
|
|
|
+ }
|
|
|
+ );
|
|
|
+ set_error_handler(
|
|
|
+ function($code,$text) use($fw) {
|
|
|
+ if (error_reporting())
|
|
|
+ $fw->error(500,$text);
|
|
|
+ }
|
|
|
+ );
|
|
|
+ if (!isset($_SERVER['SERVER_NAME']))
|
|
|
+ $_SERVER['SERVER_NAME']=gethostname();
|
|
|
+ if (PHP_SAPI=='cli') {
|
|
|
+ // Emulate HTTP request
|
|
|
+ if (isset($_SERVER['argc']) && $_SERVER['argc']<2) {
|
|
|
+ $_SERVER['argc']++;
|
|
|
+ $_SERVER['argv'][1]='/';
|
|
|
+ }
|
|
|
+ $_SERVER['REQUEST_METHOD']='GET';
|
|
|
+ $_SERVER['REQUEST_URI']=$_SERVER['argv'][1];
|
|
|
+ }
|
|
|
+ $headers=array();
|
|
|
+ if (PHP_SAPI!='cli')
|
|
|
+ foreach (array_keys($_SERVER) as $key)
|
|
|
+ if (substr($key,0,5)=='HTTP_')
|
|
|
+ $headers[strtr(ucwords(strtolower(strtr(
|
|
|
+ substr($key,5),'_',' '))),' ','-')]=&$_SERVER[$key];
|
|
|
+ if (isset($headers['X-HTTP-Method-Override']))
|
|
|
+ $_SERVER['REQUEST_METHOD']=$headers['X-HTTP-Method-Override'];
|
|
|
+ elseif ($_SERVER['REQUEST_METHOD']=='POST' && isset($_POST['_method']))
|
|
|
+ $_SERVER['REQUEST_METHOD']=$_POST['_method'];
|
|
|
+ $scheme=isset($_SERVER['HTTPS']) && $_SERVER['HTTPS']=='on' ||
|
|
|
+ isset($headers['X-Forwarded-Proto']) &&
|
|
|
+ $headers['X-Forwarded-Proto']=='https'?'https':'http';
|
|
|
+ if (function_exists('apache_setenv')) {
|
|
|
+ // Work around Apache pre-2.4 VirtualDocumentRoot bug
|
|
|
+ $_SERVER['DOCUMENT_ROOT']=str_replace($_SERVER['SCRIPT_NAME'],'',
|
|
|
+ $_SERVER['SCRIPT_FILENAME']);
|
|
|
+ apache_setenv("DOCUMENT_ROOT",$_SERVER['DOCUMENT_ROOT']);
|
|
|
+ }
|
|
|
+ $_SERVER['DOCUMENT_ROOT']=realpath($_SERVER['DOCUMENT_ROOT']);
|
|
|
+ $base='';
|
|
|
+ if (PHP_SAPI!='cli')
|
|
|
+ $base=implode('/',array_map('urlencode',
|
|
|
+ explode('/',rtrim($this->fixslashes(
|
|
|
+ dirname($_SERVER['SCRIPT_NAME'])),'/'))));
|
|
|
+ $path=preg_replace('/^'.preg_quote($base,'/').'/','',
|
|
|
+ parse_url($_SERVER['REQUEST_URI'],PHP_URL_PATH));
|
|
|
+ call_user_func_array('session_set_cookie_params',
|
|
|
+ $jar=array(
|
|
|
+ 'expire'=>0,
|
|
|
+ 'path'=>$base?:'/',
|
|
|
+ 'domain'=>is_int(strpos($_SERVER['SERVER_NAME'],'.')) &&
|
|
|
+ !filter_var($_SERVER['SERVER_NAME'],FILTER_VALIDATE_IP)?
|
|
|
+ $_SERVER['SERVER_NAME']:'',
|
|
|
+ 'secure'=>($scheme=='https'),
|
|
|
+ 'httponly'=>TRUE
|
|
|
+ )
|
|
|
+ );
|
|
|
+ // Default configuration
|
|
|
+ $this->hive=array(
|
|
|
+ 'AGENT'=>isset($headers['X-Operamini-Phone-UA'])?
|
|
|
+ $headers['X-Operamini-Phone-UA']:
|
|
|
+ (isset($headers['X-Skyfire-Phone'])?
|
|
|
+ $headers['X-Skyfire-Phone']:
|
|
|
+ (isset($headers['User-Agent'])?
|
|
|
+ $headers['User-Agent']:'')),
|
|
|
+ 'AJAX'=>isset($headers['X-Requested-With']) &&
|
|
|
+ $headers['X-Requested-With']=='XMLHttpRequest',
|
|
|
+ 'ALIASES'=>array(),
|
|
|
+ 'AUTOLOAD'=>'./',
|
|
|
+ 'BASE'=>$base,
|
|
|
+ 'BITMASK'=>ENT_COMPAT,
|
|
|
+ 'BODY'=>NULL,
|
|
|
+ 'CACHE'=>FALSE,
|
|
|
+ 'CASELESS'=>TRUE,
|
|
|
+ 'DEBUG'=>0,
|
|
|
+ 'DIACRITICS'=>array(),
|
|
|
+ 'DNSBL'=>'',
|
|
|
+ 'EMOJI'=>array(),
|
|
|
+ 'ENCODING'=>$charset,
|
|
|
+ 'ERROR'=>NULL,
|
|
|
+ 'ESCAPE'=>TRUE,
|
|
|
+ 'EXEMPT'=>NULL,
|
|
|
+ 'FALLBACK'=>$this->fallback,
|
|
|
+ 'HEADERS'=>$headers,
|
|
|
+ 'HALT'=>TRUE,
|
|
|
+ 'HIGHLIGHT'=>TRUE,
|
|
|
+ 'HOST'=>$_SERVER['SERVER_NAME'],
|
|
|
+ 'IP'=>isset($headers['Client-IP'])?
|
|
|
+ $headers['Client-IP']:
|
|
|
+ (isset($headers['X-Forwarded-For'])?
|
|
|
+ $headers['X-Forwarded-For']:
|
|
|
+ (isset($_SERVER['REMOTE_ADDR'])?
|
|
|
+ $_SERVER['REMOTE_ADDR']:'')),
|
|
|
+ 'JAR'=>$jar,
|
|
|
+ 'LANGUAGE'=>isset($headers['Accept-Language'])?
|
|
|
+ $this->language($headers['Accept-Language']):
|
|
|
+ $this->fallback,
|
|
|
+ 'LOCALES'=>'./',
|
|
|
+ 'LOGS'=>'./',
|
|
|
+ 'ONERROR'=>NULL,
|
|
|
+ 'PACKAGE'=>self::PACKAGE,
|
|
|
+ 'PARAMS'=>array(),
|
|
|
+ 'PATH'=>$path,
|
|
|
+ 'PATTERN'=>NULL,
|
|
|
+ 'PLUGINS'=>$this->fixslashes(__DIR__).'/',
|
|
|
+ 'PORT'=>isset($_SERVER['SERVER_PORT'])?
|
|
|
+ $_SERVER['SERVER_PORT']:NULL,
|
|
|
+ 'PREFIX'=>NULL,
|
|
|
+ 'QUIET'=>FALSE,
|
|
|
+ 'RAW'=>FALSE,
|
|
|
+ 'REALM'=>$scheme.'://'.
|
|
|
+ $_SERVER['SERVER_NAME'].$_SERVER['REQUEST_URI'],
|
|
|
+ 'RESPONSE'=>'',
|
|
|
+ 'ROOT'=>$_SERVER['DOCUMENT_ROOT'],
|
|
|
+ 'ROUTES'=>array(),
|
|
|
+ 'SCHEME'=>$scheme,
|
|
|
+ 'SERIALIZER'=>extension_loaded($ext='igbinary')?$ext:'php',
|
|
|
+ 'TEMP'=>'tmp/',
|
|
|
+ 'TIME'=>microtime(TRUE),
|
|
|
+ 'TZ'=>(@ini_get('date.timezone'))?:'UTC',
|
|
|
+ 'UI'=>'./',
|
|
|
+ 'UNLOAD'=>NULL,
|
|
|
+ 'UPLOADS'=>'./',
|
|
|
+ 'URI'=>&$_SERVER['REQUEST_URI'],
|
|
|
+ 'VERB'=>&$_SERVER['REQUEST_METHOD'],
|
|
|
+ 'VERSION'=>self::VERSION,
|
|
|
+ 'XFRAME'=>'SAMEORIGIN'
|
|
|
+ );
|
|
|
+ if (PHP_SAPI=='cli-server' &&
|
|
|
+ preg_match('/^'.preg_quote($base,'/').'$/',$this->hive['URI']))
|
|
|
+ $this->reroute('/');
|
|
|
+ if (ini_get('auto_globals_jit'))
|
|
|
+ // Override setting
|
|
|
+ $GLOBALS+=array('_ENV'=>$_ENV,'_REQUEST'=>$_REQUEST);
|
|
|
+ // Sync PHP globals with corresponding hive keys
|
|
|
+ $this->init=$this->hive;
|
|
|
+ foreach (explode('|',self::GLOBALS) as $global) {
|
|
|
+ $sync=$this->sync($global);
|
|
|
+ $this->init+=array(
|
|
|
+ $global=>preg_match('/SERVER|ENV/',$global)?$sync:array()
|
|
|
+ );
|
|
|
+ }
|
|
|
+ if ($error=error_get_last())
|
|
|
+ // Error detected
|
|
|
+ $this->error(500,sprintf(self::E_Fatal,$error['message']),
|
|
|
+ array($error));
|
|
|
+ date_default_timezone_set($this->hive['TZ']);
|
|
|
+ // Register framework autoloader
|
|
|
+ spl_autoload_register(array($this,'autoload'));
|
|
|
+ // Register shutdown handler
|
|
|
+ register_shutdown_function(array($this,'unload'),getcwd());
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! Cache engine
|
|
|
+class Cache extends Prefab {
|
|
|
+
|
|
|
+ protected
|
|
|
+ //! Cache DSN
|
|
|
+ $dsn,
|
|
|
+ //! Prefix for cache entries
|
|
|
+ $prefix,
|
|
|
+ //! MemCache or Redis object
|
|
|
+ $ref;
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return timestamp and TTL of cache entry or FALSE if not found
|
|
|
+ * @return array|FALSE
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ **/
|
|
|
+ function exists($key,&$val=NULL) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ if (!$this->dsn)
|
|
|
+ return FALSE;
|
|
|
+ $ndx=$this->prefix.'.'.$key;
|
|
|
+ $parts=explode('=',$this->dsn,2);
|
|
|
+ switch ($parts[0]) {
|
|
|
+ case 'apc':
|
|
|
+ case 'apcu':
|
|
|
+ $raw=apc_fetch($ndx);
|
|
|
+ break;
|
|
|
+ case 'redis':
|
|
|
+ $raw=$this->ref->get($ndx);
|
|
|
+ break;
|
|
|
+ case 'memcache':
|
|
|
+ $raw=memcache_get($this->ref,$ndx);
|
|
|
+ break;
|
|
|
+ case 'wincache':
|
|
|
+ $raw=wincache_ucache_get($ndx);
|
|
|
+ break;
|
|
|
+ case 'xcache':
|
|
|
+ $raw=xcache_get($ndx);
|
|
|
+ break;
|
|
|
+ case 'folder':
|
|
|
+ $raw=$fw->read($parts[1].$ndx);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (!empty($raw)) {
|
|
|
+ list($val,$time,$ttl)=(array)$fw->unserialize($raw);
|
|
|
+ if ($ttl===0 || $time+$ttl>microtime(TRUE))
|
|
|
+ return array($time,$ttl);
|
|
|
+ $this->clear($key);
|
|
|
+ }
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Store value in cache
|
|
|
+ * @return mixed|FALSE
|
|
|
+ * @param $key string
|
|
|
+ * @param $val mixed
|
|
|
+ * @param $ttl int
|
|
|
+ **/
|
|
|
+ function set($key,$val,$ttl=0) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ if (!$this->dsn)
|
|
|
+ return TRUE;
|
|
|
+ $ndx=$this->prefix.'.'.$key;
|
|
|
+ $time=microtime(TRUE);
|
|
|
+ if ($cached=$this->exists($key))
|
|
|
+ list($time,$ttl)=$cached;
|
|
|
+ $data=$fw->serialize(array($val,$time,$ttl));
|
|
|
+ $parts=explode('=',$this->dsn,2);
|
|
|
+ switch ($parts[0]) {
|
|
|
+ case 'apc':
|
|
|
+ case 'apcu':
|
|
|
+ return apc_store($ndx,$data,$ttl);
|
|
|
+ case 'redis':
|
|
|
+ return $this->ref->set($ndx,$data,array('ex'=>$ttl));
|
|
|
+ case 'memcache':
|
|
|
+ return memcache_set($this->ref,$ndx,$data,0,$ttl);
|
|
|
+ case 'wincache':
|
|
|
+ return wincache_ucache_set($ndx,$data,$ttl);
|
|
|
+ case 'xcache':
|
|
|
+ return xcache_set($ndx,$data,$ttl);
|
|
|
+ case 'folder':
|
|
|
+ return $fw->write($parts[1].$ndx,$data);
|
|
|
+ }
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Retrieve value of cache entry
|
|
|
+ * @return mixed|FALSE
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function get($key) {
|
|
|
+ return $this->dsn && $this->exists($key,$data)?$data:FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Delete cache entry
|
|
|
+ * @return bool
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ function clear($key) {
|
|
|
+ if (!$this->dsn)
|
|
|
+ return;
|
|
|
+ $ndx=$this->prefix.'.'.$key;
|
|
|
+ $parts=explode('=',$this->dsn,2);
|
|
|
+ switch ($parts[0]) {
|
|
|
+ case 'apc':
|
|
|
+ case 'apcu':
|
|
|
+ return apc_delete($ndx);
|
|
|
+ case 'redis':
|
|
|
+ return $this->ref->del($ndx);
|
|
|
+ case 'memcache':
|
|
|
+ return memcache_delete($this->ref,$ndx);
|
|
|
+ case 'wincache':
|
|
|
+ return wincache_ucache_delete($ndx);
|
|
|
+ case 'xcache':
|
|
|
+ return xcache_unset($ndx);
|
|
|
+ case 'folder':
|
|
|
+ return @unlink($parts[1].$ndx);
|
|
|
+ }
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Clear contents of cache backend
|
|
|
+ * @return bool
|
|
|
+ * @param $suffix string
|
|
|
+ * @param $lifetime int
|
|
|
+ **/
|
|
|
+ function reset($suffix=NULL,$lifetime=0) {
|
|
|
+ if (!$this->dsn)
|
|
|
+ return TRUE;
|
|
|
+ $regex='/'.preg_quote($this->prefix.'.','/').'.+?'.
|
|
|
+ preg_quote($suffix,'/').'/';
|
|
|
+ $parts=explode('=',$this->dsn,2);
|
|
|
+ switch ($parts[0]) {
|
|
|
+ case 'apc':
|
|
|
+ $key='info';
|
|
|
+ case 'apcu':
|
|
|
+ if (empty($key))
|
|
|
+ $key='key';
|
|
|
+ $info=apc_cache_info('user');
|
|
|
+ foreach ($info['cache_list'] as $item)
|
|
|
+ if (preg_match($regex,$item[$key]) &&
|
|
|
+ $item['mtime']+$lifetime<time())
|
|
|
+ apc_delete($item[$key]);
|
|
|
+ return TRUE;
|
|
|
+ case 'redis':
|
|
|
+ $fw=Base::instance();
|
|
|
+ $keys=$this->ref->keys($this->prefix.'.*'.$suffix);
|
|
|
+ foreach($keys as $key) {
|
|
|
+ $val=$fw->unserialize($this->ref->get($key));
|
|
|
+ if ($val[1]+$lifetime<time())
|
|
|
+ $this->ref->del($key);
|
|
|
+ }
|
|
|
+ return TRUE;
|
|
|
+ case 'memcache':
|
|
|
+ foreach (memcache_get_extended_stats(
|
|
|
+ $this->ref,'slabs') as $slabs)
|
|
|
+ foreach (array_filter(array_keys($slabs),'is_numeric')
|
|
|
+ as $id)
|
|
|
+ foreach (memcache_get_extended_stats(
|
|
|
+ $this->ref,'cachedump',$id) as $data)
|
|
|
+ if (is_array($data))
|
|
|
+ foreach ($data as $key=>$val)
|
|
|
+ if (preg_match($regex,$key) &&
|
|
|
+ $val[1]+$lifetime<time())
|
|
|
+ memcache_delete($this->ref,$key);
|
|
|
+ return TRUE;
|
|
|
+ case 'wincache':
|
|
|
+ $info=wincache_ucache_info();
|
|
|
+ foreach ($info['ucache_entries'] as $item)
|
|
|
+ if (preg_match($regex,$item['key_name']) &&
|
|
|
+ $item['use_time']+$lifetime<time())
|
|
|
+ wincache_ucache_delete($item['key_name']);
|
|
|
+ return TRUE;
|
|
|
+ case 'xcache':
|
|
|
+ return TRUE; /* Not supported */
|
|
|
+ case 'folder':
|
|
|
+ if ($glob=@glob($parts[1].'*'))
|
|
|
+ foreach ($glob as $file)
|
|
|
+ if (preg_match($regex,basename($file)) &&
|
|
|
+ filemtime($file)+$lifetime<time())
|
|
|
+ @unlink($file);
|
|
|
+ return TRUE;
|
|
|
+ }
|
|
|
+ return FALSE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Load/auto-detect cache backend
|
|
|
+ * @return string
|
|
|
+ * @param $dsn bool|string
|
|
|
+ **/
|
|
|
+ function load($dsn) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ if ($dsn=trim($dsn)) {
|
|
|
+ if (preg_match('/^redis=(.+)/',$dsn,$parts) &&
|
|
|
+ extension_loaded('redis')) {
|
|
|
+ $port=6379;
|
|
|
+ $parts=explode(':',$parts[1],2);
|
|
|
+ if (count($parts)>1)
|
|
|
+ list($host,$port)=$parts;
|
|
|
+ else
|
|
|
+ $host=$parts[0];
|
|
|
+ $this->ref=new Redis;
|
|
|
+ if(!$this->ref->connect($host,$port,2))
|
|
|
+ $this->ref=NULL;
|
|
|
+ }
|
|
|
+ elseif (preg_match('/^memcache=(.+)/',$dsn,$parts) &&
|
|
|
+ extension_loaded('memcache'))
|
|
|
+ foreach ($fw->split($parts[1]) as $server) {
|
|
|
+ $port=11211;
|
|
|
+ $parts=explode(':',$server,2);
|
|
|
+ if (count($parts)>1)
|
|
|
+ list($host,$port)=$parts;
|
|
|
+ else
|
|
|
+ $host=$parts[0];
|
|
|
+ if (empty($this->ref))
|
|
|
+ $this->ref=@memcache_connect($host,$port)?:NULL;
|
|
|
+ else
|
|
|
+ memcache_add_server($this->ref,$host,$port);
|
|
|
+ }
|
|
|
+ if (empty($this->ref) && !preg_match('/^folder\h*=/',$dsn))
|
|
|
+ $dsn=($grep=preg_grep('/^(apc|wincache|xcache)/',
|
|
|
+ array_map('strtolower',get_loaded_extensions())))?
|
|
|
+ // Auto-detect
|
|
|
+ current($grep):
|
|
|
+ // Use filesystem as fallback
|
|
|
+ ('folder='.$fw->get('TEMP').'cache/');
|
|
|
+ if (preg_match('/^folder\h*=\h*(.+)/',$dsn,$parts) &&
|
|
|
+ !is_dir($parts[1]))
|
|
|
+ mkdir($parts[1],Base::MODE,TRUE);
|
|
|
+ }
|
|
|
+ $this->prefix=$fw->hash($_SERVER['SERVER_NAME'].$fw->get('BASE'));
|
|
|
+ return $this->dsn=$dsn;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Class constructor
|
|
|
+ * @return object
|
|
|
+ * @param $dsn bool|string
|
|
|
+ **/
|
|
|
+ function __construct($dsn=FALSE) {
|
|
|
+ if ($dsn)
|
|
|
+ $this->load($dsn);
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! View handler
|
|
|
+class View extends Prefab {
|
|
|
+
|
|
|
+ protected
|
|
|
+ //! Template file
|
|
|
+ $view;
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Encode characters to equivalent HTML entities
|
|
|
+ * @return string
|
|
|
+ * @param $arg mixed
|
|
|
+ **/
|
|
|
+ function esc($arg) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ return $fw->recursive($arg,
|
|
|
+ function($val) use($fw) {
|
|
|
+ return is_string($val)?$fw->encode($val):$val;
|
|
|
+ }
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Decode HTML entities to equivalent characters
|
|
|
+ * @return string
|
|
|
+ * @param $arg mixed
|
|
|
+ **/
|
|
|
+ function raw($arg) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ return $fw->recursive($arg,
|
|
|
+ function($val) use($fw) {
|
|
|
+ return is_string($val)?$fw->decode($val):$val;
|
|
|
+ }
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Create sandbox for template execution
|
|
|
+ * @return string
|
|
|
+ * @param $hive array
|
|
|
+ **/
|
|
|
+ protected function sandbox(array $hive=NULL) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ if (!$hive)
|
|
|
+ $hive=$fw->hive();
|
|
|
+ if ($fw->get('ESCAPE'))
|
|
|
+ $hive=$this->esc($hive);
|
|
|
+ if (isset($hive['ALIASES']))
|
|
|
+ $hive['ALIASES']=$fw->build($hive['ALIASES']);
|
|
|
+ extract($hive);
|
|
|
+ unset($fw);
|
|
|
+ unset($hive);
|
|
|
+ ob_start();
|
|
|
+ require($this->view);
|
|
|
+ return ob_get_clean();
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Render template
|
|
|
+ * @return string
|
|
|
+ * @param $file string
|
|
|
+ * @param $mime string
|
|
|
+ * @param $hive array
|
|
|
+ * @param $ttl int
|
|
|
+ **/
|
|
|
+ function render($file,$mime='text/html',array $hive=NULL,$ttl=0) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ $cache=Cache::instance();
|
|
|
+ $cached=$cache->exists($hash=$fw->hash($file),$data);
|
|
|
+ if ($cached && $cached[0]+$ttl>microtime(TRUE))
|
|
|
+ return $data;
|
|
|
+ foreach ($fw->split($fw->get('UI').';./') as $dir)
|
|
|
+ if (is_file($this->view=$fw->fixslashes($dir.$file))) {
|
|
|
+ if (isset($_COOKIE[session_name()]))
|
|
|
+ @session_start();
|
|
|
+ $fw->sync('SESSION');
|
|
|
+ if ($mime && PHP_SAPI!='cli')
|
|
|
+ header('Content-Type: '.$mime.'; '.
|
|
|
+ 'charset='.$fw->get('ENCODING'));
|
|
|
+ $data=$this->sandbox($hive);
|
|
|
+ if ($ttl)
|
|
|
+ $cache->set($hash,$data);
|
|
|
+ return $data;
|
|
|
+ }
|
|
|
+ user_error(sprintf(Base::E_Open,$file));
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! Lightweight template engine
|
|
|
+class Preview extends View {
|
|
|
+
|
|
|
+ protected
|
|
|
+ //! MIME type
|
|
|
+ $mime;
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert token to variable
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ **/
|
|
|
+ function token($str) {
|
|
|
+ return trim(preg_replace('/\{\{(.+?)\}\}/s',trim('\1'),
|
|
|
+ Base::instance()->compile($str)));
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Assemble markup
|
|
|
+ * @return string
|
|
|
+ * @param $node string
|
|
|
+ **/
|
|
|
+ protected function build($node) {
|
|
|
+ $self=$this;
|
|
|
+ return preg_replace_callback(
|
|
|
+ '/\{\{(.+?)\}\}/s',
|
|
|
+ function($expr) use($self) {
|
|
|
+ $str=trim($self->token($expr[1]));
|
|
|
+ if (preg_match('/^(.+?)\h*\|(\h*\w+(?:\h*[,;]\h*\w+)*)/',
|
|
|
+ $str,$parts)) {
|
|
|
+ $str=$parts[1];
|
|
|
+ foreach (Base::instance()->split($parts[2]) as $func)
|
|
|
+ $str=(($func=='format')?'\Base::instance()':'$this').
|
|
|
+ '->'.$func.'('.$str.')';
|
|
|
+ }
|
|
|
+ return '<?php echo '.$str.'; ?>';
|
|
|
+ },
|
|
|
+ preg_replace_callback(
|
|
|
+ '/\{~(.+?)~\}/s',
|
|
|
+ function($expr) use($self) {
|
|
|
+ return '<?php '.$self->token($expr[1]).' ?>';
|
|
|
+ },
|
|
|
+ $node
|
|
|
+ )
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Render template string
|
|
|
+ * @return string
|
|
|
+ * @param $str string
|
|
|
+ * @param $hive array
|
|
|
+ **/
|
|
|
+ function resolve($str,array $hive=NULL) {
|
|
|
+ if (!$hive)
|
|
|
+ $hive=\Base::instance()->hive();
|
|
|
+ extract($hive);
|
|
|
+ ob_start();
|
|
|
+ eval(' ?>'.$this->build($str).'<?php ');
|
|
|
+ return ob_get_clean();
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Render template
|
|
|
+ * @return string
|
|
|
+ * @param $file string
|
|
|
+ * @param $mime string
|
|
|
+ * @param $hive array
|
|
|
+ * @param $ttl int
|
|
|
+ **/
|
|
|
+ function render($file,$mime='text/html',array $hive=NULL,$ttl=0) {
|
|
|
+ $fw=Base::instance();
|
|
|
+ $cache=Cache::instance();
|
|
|
+ $cached=$cache->exists($hash=$fw->hash($file),$data);
|
|
|
+ if ($cached && $cached[0]+$ttl>microtime(TRUE))
|
|
|
+ return $data;
|
|
|
+ if (!is_dir($tmp=$fw->get('TEMP')))
|
|
|
+ mkdir($tmp,Base::MODE,TRUE);
|
|
|
+ foreach ($fw->split($fw->get('UI')) as $dir)
|
|
|
+ if (is_file($view=$fw->fixslashes($dir.$file))) {
|
|
|
+ if (!is_file($this->view=($tmp.
|
|
|
+ $fw->hash($fw->get('ROOT').$fw->get('BASE')).'.'.
|
|
|
+ $fw->hash($view).'.php')) ||
|
|
|
+ filemtime($this->view)<filemtime($view)) {
|
|
|
+ // Remove PHP code and comments
|
|
|
+ $text=preg_replace(
|
|
|
+ '/(?<!["\'])\h*<\?(?:php|\s*=).+?\?>\h*(?!["\'])|'.
|
|
|
+ '\{\*.+?\*\}/is','',
|
|
|
+ $fw->read($view));
|
|
|
+ if (method_exists($this,'parse'))
|
|
|
+ $text=$this->parse($text);
|
|
|
+ $fw->write($this->view,$this->build($text));
|
|
|
+ }
|
|
|
+ if (isset($_COOKIE[session_name()]))
|
|
|
+ @session_start();
|
|
|
+ $fw->sync('SESSION');
|
|
|
+ if ($mime && PHP_SAPI!='cli')
|
|
|
+ header('Content-Type: '.($this->mime=$mime).'; '.
|
|
|
+ 'charset='.$fw->get('ENCODING'));
|
|
|
+ $data=$this->sandbox($hive);
|
|
|
+ if ($ttl)
|
|
|
+ $cache->set($hash,$data);
|
|
|
+ return $data;
|
|
|
+ }
|
|
|
+ user_error(sprintf(Base::E_Open,$file));
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! ISO language/country codes
|
|
|
+class ISO extends Prefab {
|
|
|
+
|
|
|
+ //@{ ISO 3166-1 country codes
|
|
|
+ const
|
|
|
+ CC_af='Afghanistan',
|
|
|
+ CC_ax='Åland Islands',
|
|
|
+ CC_al='Albania',
|
|
|
+ CC_dz='Algeria',
|
|
|
+ CC_as='American Samoa',
|
|
|
+ CC_ad='Andorra',
|
|
|
+ CC_ao='Angola',
|
|
|
+ CC_ai='Anguilla',
|
|
|
+ CC_aq='Antarctica',
|
|
|
+ CC_ag='Antigua and Barbuda',
|
|
|
+ CC_ar='Argentina',
|
|
|
+ CC_am='Armenia',
|
|
|
+ CC_aw='Aruba',
|
|
|
+ CC_au='Australia',
|
|
|
+ CC_at='Austria',
|
|
|
+ CC_az='Azerbaijan',
|
|
|
+ CC_bs='Bahamas',
|
|
|
+ CC_bh='Bahrain',
|
|
|
+ CC_bd='Bangladesh',
|
|
|
+ CC_bb='Barbados',
|
|
|
+ CC_by='Belarus',
|
|
|
+ CC_be='Belgium',
|
|
|
+ CC_bz='Belize',
|
|
|
+ CC_bj='Benin',
|
|
|
+ CC_bm='Bermuda',
|
|
|
+ CC_bt='Bhutan',
|
|
|
+ CC_bo='Bolivia',
|
|
|
+ CC_bq='Bonaire, Sint Eustatius and Saba',
|
|
|
+ CC_ba='Bosnia and Herzegovina',
|
|
|
+ CC_bw='Botswana',
|
|
|
+ CC_bv='Bouvet Island',
|
|
|
+ CC_br='Brazil',
|
|
|
+ CC_io='British Indian Ocean Territory',
|
|
|
+ CC_bn='Brunei Darussalam',
|
|
|
+ CC_bg='Bulgaria',
|
|
|
+ CC_bf='Burkina Faso',
|
|
|
+ CC_bi='Burundi',
|
|
|
+ CC_kh='Cambodia',
|
|
|
+ CC_cm='Cameroon',
|
|
|
+ CC_ca='Canada',
|
|
|
+ CC_cv='Cape Verde',
|
|
|
+ CC_ky='Cayman Islands',
|
|
|
+ CC_cf='Central African Republic',
|
|
|
+ CC_td='Chad',
|
|
|
+ CC_cl='Chile',
|
|
|
+ CC_cn='China',
|
|
|
+ CC_cx='Christmas Island',
|
|
|
+ CC_cc='Cocos (Keeling) Islands',
|
|
|
+ CC_co='Colombia',
|
|
|
+ CC_km='Comoros',
|
|
|
+ CC_cg='Congo',
|
|
|
+ CC_cd='Congo, The Democratic Republic of',
|
|
|
+ CC_ck='Cook Islands',
|
|
|
+ CC_cr='Costa Rica',
|
|
|
+ CC_ci='Côte d\'ivoire',
|
|
|
+ CC_hr='Croatia',
|
|
|
+ CC_cu='Cuba',
|
|
|
+ CC_cw='Curaçao',
|
|
|
+ CC_cy='Cyprus',
|
|
|
+ CC_cz='Czech Republic',
|
|
|
+ CC_dk='Denmark',
|
|
|
+ CC_dj='Djibouti',
|
|
|
+ CC_dm='Dominica',
|
|
|
+ CC_do='Dominican Republic',
|
|
|
+ CC_ec='Ecuador',
|
|
|
+ CC_eg='Egypt',
|
|
|
+ CC_sv='El Salvador',
|
|
|
+ CC_gq='Equatorial Guinea',
|
|
|
+ CC_er='Eritrea',
|
|
|
+ CC_ee='Estonia',
|
|
|
+ CC_et='Ethiopia',
|
|
|
+ CC_fk='Falkland Islands (Malvinas)',
|
|
|
+ CC_fo='Faroe Islands',
|
|
|
+ CC_fj='Fiji',
|
|
|
+ CC_fi='Finland',
|
|
|
+ CC_fr='France',
|
|
|
+ CC_gf='French Guiana',
|
|
|
+ CC_pf='French Polynesia',
|
|
|
+ CC_tf='French Southern Territories',
|
|
|
+ CC_ga='Gabon',
|
|
|
+ CC_gm='Gambia',
|
|
|
+ CC_ge='Georgia',
|
|
|
+ CC_de='Germany',
|
|
|
+ CC_gh='Ghana',
|
|
|
+ CC_gi='Gibraltar',
|
|
|
+ CC_gr='Greece',
|
|
|
+ CC_gl='Greenland',
|
|
|
+ CC_gd='Grenada',
|
|
|
+ CC_gp='Guadeloupe',
|
|
|
+ CC_gu='Guam',
|
|
|
+ CC_gt='Guatemala',
|
|
|
+ CC_gg='Guernsey',
|
|
|
+ CC_gn='Guinea',
|
|
|
+ CC_gw='Guinea-Bissau',
|
|
|
+ CC_gy='Guyana',
|
|
|
+ CC_ht='Haiti',
|
|
|
+ CC_hm='Heard Island and McDonald Islands',
|
|
|
+ CC_va='Holy See (Vatican City State)',
|
|
|
+ CC_hn='Honduras',
|
|
|
+ CC_hk='Hong Kong',
|
|
|
+ CC_hu='Hungary',
|
|
|
+ CC_is='Iceland',
|
|
|
+ CC_in='India',
|
|
|
+ CC_id='Indonesia',
|
|
|
+ CC_ir='Iran, Islamic Republic of',
|
|
|
+ CC_iq='Iraq',
|
|
|
+ CC_ie='Ireland',
|
|
|
+ CC_im='Isle of Man',
|
|
|
+ CC_il='Israel',
|
|
|
+ CC_it='Italy',
|
|
|
+ CC_jm='Jamaica',
|
|
|
+ CC_jp='Japan',
|
|
|
+ CC_je='Jersey',
|
|
|
+ CC_jo='Jordan',
|
|
|
+ CC_kz='Kazakhstan',
|
|
|
+ CC_ke='Kenya',
|
|
|
+ CC_ki='Kiribati',
|
|
|
+ CC_kp='Korea, Democratic People\'s Republic of',
|
|
|
+ CC_kr='Korea, Republic of',
|
|
|
+ CC_kw='Kuwait',
|
|
|
+ CC_kg='Kyrgyzstan',
|
|
|
+ CC_la='Lao People\'s Democratic Republic',
|
|
|
+ CC_lv='Latvia',
|
|
|
+ CC_lb='Lebanon',
|
|
|
+ CC_ls='Lesotho',
|
|
|
+ CC_lr='Liberia',
|
|
|
+ CC_ly='Libya',
|
|
|
+ CC_li='Liechtenstein',
|
|
|
+ CC_lt='Lithuania',
|
|
|
+ CC_lu='Luxembourg',
|
|
|
+ CC_mo='Macao',
|
|
|
+ CC_mk='Macedonia, The Former Yugoslav Republic of',
|
|
|
+ CC_mg='Madagascar',
|
|
|
+ CC_mw='Malawi',
|
|
|
+ CC_my='Malaysia',
|
|
|
+ CC_mv='Maldives',
|
|
|
+ CC_ml='Mali',
|
|
|
+ CC_mt='Malta',
|
|
|
+ CC_mh='Marshall Islands',
|
|
|
+ CC_mq='Martinique',
|
|
|
+ CC_mr='Mauritania',
|
|
|
+ CC_mu='Mauritius',
|
|
|
+ CC_yt='Mayotte',
|
|
|
+ CC_mx='Mexico',
|
|
|
+ CC_fm='Micronesia, Federated States of',
|
|
|
+ CC_md='Moldova, Republic of',
|
|
|
+ CC_mc='Monaco',
|
|
|
+ CC_mn='Mongolia',
|
|
|
+ CC_me='Montenegro',
|
|
|
+ CC_ms='Montserrat',
|
|
|
+ CC_ma='Morocco',
|
|
|
+ CC_mz='Mozambique',
|
|
|
+ CC_mm='Myanmar',
|
|
|
+ CC_na='Namibia',
|
|
|
+ CC_nr='Nauru',
|
|
|
+ CC_np='Nepal',
|
|
|
+ CC_nl='Netherlands',
|
|
|
+ CC_nc='New Caledonia',
|
|
|
+ CC_nz='New Zealand',
|
|
|
+ CC_ni='Nicaragua',
|
|
|
+ CC_ne='Niger',
|
|
|
+ CC_ng='Nigeria',
|
|
|
+ CC_nu='Niue',
|
|
|
+ CC_nf='Norfolk Island',
|
|
|
+ CC_mp='Northern Mariana Islands',
|
|
|
+ CC_no='Norway',
|
|
|
+ CC_om='Oman',
|
|
|
+ CC_pk='Pakistan',
|
|
|
+ CC_pw='Palau',
|
|
|
+ CC_ps='Palestinian Territory, Occupied',
|
|
|
+ CC_pa='Panama',
|
|
|
+ CC_pg='Papua New Guinea',
|
|
|
+ CC_py='Paraguay',
|
|
|
+ CC_pe='Peru',
|
|
|
+ CC_ph='Philippines',
|
|
|
+ CC_pn='Pitcairn',
|
|
|
+ CC_pl='Poland',
|
|
|
+ CC_pt='Portugal',
|
|
|
+ CC_pr='Puerto Rico',
|
|
|
+ CC_qa='Qatar',
|
|
|
+ CC_re='Réunion',
|
|
|
+ CC_ro='Romania',
|
|
|
+ CC_ru='Russian Federation',
|
|
|
+ CC_rw='Rwanda',
|
|
|
+ CC_bl='Saint Barthélemy',
|
|
|
+ CC_sh='Saint Helena, Ascension and Tristan da Cunha',
|
|
|
+ CC_kn='Saint Kitts and Nevis',
|
|
|
+ CC_lc='Saint Lucia',
|
|
|
+ CC_mf='Saint Martin (French Part)',
|
|
|
+ CC_pm='Saint Pierre and Miquelon',
|
|
|
+ CC_vc='Saint Vincent and The Grenadines',
|
|
|
+ CC_ws='Samoa',
|
|
|
+ CC_sm='San Marino',
|
|
|
+ CC_st='Sao Tome and Principe',
|
|
|
+ CC_sa='Saudi Arabia',
|
|
|
+ CC_sn='Senegal',
|
|
|
+ CC_rs='Serbia',
|
|
|
+ CC_sc='Seychelles',
|
|
|
+ CC_sl='Sierra Leone',
|
|
|
+ CC_sg='Singapore',
|
|
|
+ CC_sk='Slovakia',
|
|
|
+ CC_sx='Sint Maarten (Dutch Part)',
|
|
|
+ CC_si='Slovenia',
|
|
|
+ CC_sb='Solomon Islands',
|
|
|
+ CC_so='Somalia',
|
|
|
+ CC_za='South Africa',
|
|
|
+ CC_gs='South Georgia and The South Sandwich Islands',
|
|
|
+ CC_ss='South Sudan',
|
|
|
+ CC_es='Spain',
|
|
|
+ CC_lk='Sri Lanka',
|
|
|
+ CC_sd='Sudan',
|
|
|
+ CC_sr='Suriname',
|
|
|
+ CC_sj='Svalbard and Jan Mayen',
|
|
|
+ CC_sz='Swaziland',
|
|
|
+ CC_se='Sweden',
|
|
|
+ CC_ch='Switzerland',
|
|
|
+ CC_sy='Syrian Arab Republic',
|
|
|
+ CC_tw='Taiwan, Province of China',
|
|
|
+ CC_tj='Tajikistan',
|
|
|
+ CC_tz='Tanzania, United Republic of',
|
|
|
+ CC_th='Thailand',
|
|
|
+ CC_tl='Timor-Leste',
|
|
|
+ CC_tg='Togo',
|
|
|
+ CC_tk='Tokelau',
|
|
|
+ CC_to='Tonga',
|
|
|
+ CC_tt='Trinidad and Tobago',
|
|
|
+ CC_tn='Tunisia',
|
|
|
+ CC_tr='Turkey',
|
|
|
+ CC_tm='Turkmenistan',
|
|
|
+ CC_tc='Turks and Caicos Islands',
|
|
|
+ CC_tv='Tuvalu',
|
|
|
+ CC_ug='Uganda',
|
|
|
+ CC_ua='Ukraine',
|
|
|
+ CC_ae='United Arab Emirates',
|
|
|
+ CC_gb='United Kingdom',
|
|
|
+ CC_us='United States',
|
|
|
+ CC_um='United States Minor Outlying Islands',
|
|
|
+ CC_uy='Uruguay',
|
|
|
+ CC_uz='Uzbekistan',
|
|
|
+ CC_vu='Vanuatu',
|
|
|
+ CC_ve='Venezuela',
|
|
|
+ CC_vn='Viet Nam',
|
|
|
+ CC_vg='Virgin Islands, British',
|
|
|
+ CC_vi='Virgin Islands, U.S.',
|
|
|
+ CC_wf='Wallis and Futuna',
|
|
|
+ CC_eh='Western Sahara',
|
|
|
+ CC_ye='Yemen',
|
|
|
+ CC_zm='Zambia',
|
|
|
+ CC_zw='Zimbabwe';
|
|
|
+ //@}
|
|
|
+
|
|
|
+ //@{ ISO 639-1 language codes (Windows-compatibility subset)
|
|
|
+ const
|
|
|
+ LC_af='Afrikaans',
|
|
|
+ LC_am='Amharic',
|
|
|
+ LC_ar='Arabic',
|
|
|
+ LC_as='Assamese',
|
|
|
+ LC_ba='Bashkir',
|
|
|
+ LC_be='Belarusian',
|
|
|
+ LC_bg='Bulgarian',
|
|
|
+ LC_bn='Bengali',
|
|
|
+ LC_bo='Tibetan',
|
|
|
+ LC_br='Breton',
|
|
|
+ LC_ca='Catalan',
|
|
|
+ LC_co='Corsican',
|
|
|
+ LC_cs='Czech',
|
|
|
+ LC_cy='Welsh',
|
|
|
+ LC_da='Danish',
|
|
|
+ LC_de='German',
|
|
|
+ LC_dv='Divehi',
|
|
|
+ LC_el='Greek',
|
|
|
+ LC_en='English',
|
|
|
+ LC_es='Spanish',
|
|
|
+ LC_et='Estonian',
|
|
|
+ LC_eu='Basque',
|
|
|
+ LC_fa='Persian',
|
|
|
+ LC_fi='Finnish',
|
|
|
+ LC_fo='Faroese',
|
|
|
+ LC_fr='French',
|
|
|
+ LC_gd='Scottish Gaelic',
|
|
|
+ LC_gl='Galician',
|
|
|
+ LC_gu='Gujarati',
|
|
|
+ LC_he='Hebrew',
|
|
|
+ LC_hi='Hindi',
|
|
|
+ LC_hr='Croatian',
|
|
|
+ LC_hu='Hungarian',
|
|
|
+ LC_hy='Armenian',
|
|
|
+ LC_id='Indonesian',
|
|
|
+ LC_ig='Igbo',
|
|
|
+ LC_is='Icelandic',
|
|
|
+ LC_it='Italian',
|
|
|
+ LC_ja='Japanese',
|
|
|
+ LC_ka='Georgian',
|
|
|
+ LC_kk='Kazakh',
|
|
|
+ LC_km='Khmer',
|
|
|
+ LC_kn='Kannada',
|
|
|
+ LC_ko='Korean',
|
|
|
+ LC_lb='Luxembourgish',
|
|
|
+ LC_lo='Lao',
|
|
|
+ LC_lt='Lithuanian',
|
|
|
+ LC_lv='Latvian',
|
|
|
+ LC_mi='Maori',
|
|
|
+ LC_ml='Malayalam',
|
|
|
+ LC_mr='Marathi',
|
|
|
+ LC_ms='Malay',
|
|
|
+ LC_mt='Maltese',
|
|
|
+ LC_ne='Nepali',
|
|
|
+ LC_nl='Dutch',
|
|
|
+ LC_no='Norwegian',
|
|
|
+ LC_oc='Occitan',
|
|
|
+ LC_or='Oriya',
|
|
|
+ LC_pl='Polish',
|
|
|
+ LC_ps='Pashto',
|
|
|
+ LC_pt='Portuguese',
|
|
|
+ LC_qu='Quechua',
|
|
|
+ LC_ro='Romanian',
|
|
|
+ LC_ru='Russian',
|
|
|
+ LC_rw='Kinyarwanda',
|
|
|
+ LC_sa='Sanskrit',
|
|
|
+ LC_si='Sinhala',
|
|
|
+ LC_sk='Slovak',
|
|
|
+ LC_sl='Slovenian',
|
|
|
+ LC_sq='Albanian',
|
|
|
+ LC_sv='Swedish',
|
|
|
+ LC_ta='Tamil',
|
|
|
+ LC_te='Telugu',
|
|
|
+ LC_th='Thai',
|
|
|
+ LC_tk='Turkmen',
|
|
|
+ LC_tr='Turkish',
|
|
|
+ LC_tt='Tatar',
|
|
|
+ LC_uk='Ukrainian',
|
|
|
+ LC_ur='Urdu',
|
|
|
+ LC_vi='Vietnamese',
|
|
|
+ LC_wo='Wolof',
|
|
|
+ LC_yo='Yoruba',
|
|
|
+ LC_zh='Chinese';
|
|
|
+ //@}
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Convert class constants to array
|
|
|
+ * @return array
|
|
|
+ * @param $prefix string
|
|
|
+ **/
|
|
|
+ protected function constants($prefix) {
|
|
|
+ $ref=new ReflectionClass($this);
|
|
|
+ $out=array();
|
|
|
+ foreach (preg_grep('/^'.$prefix.'/',array_keys($ref->getconstants()))
|
|
|
+ as $val) {
|
|
|
+ $out[$key=substr($val,strlen($prefix))]=
|
|
|
+ constant('self::'.$prefix.$key);
|
|
|
+ }
|
|
|
+ unset($ref);
|
|
|
+ return $out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return list of languages indexed by ISO 639-1 language code
|
|
|
+ * @return array
|
|
|
+ **/
|
|
|
+ function languages() {
|
|
|
+ return $this->constants('LC_');
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return list of countries indexed by ISO 3166-1 country code
|
|
|
+ * @return array
|
|
|
+ **/
|
|
|
+ function countries() {
|
|
|
+ return $this->constants('CC_');
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+//! Container for singular object instances
|
|
|
+final class Registry {
|
|
|
+
|
|
|
+ private static
|
|
|
+ //! Object catalog
|
|
|
+ $table;
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Return TRUE if object exists in catalog
|
|
|
+ * @return bool
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ static function exists($key) {
|
|
|
+ return isset(self::$table[$key]);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Add object to catalog
|
|
|
+ * @return object
|
|
|
+ * @param $key string
|
|
|
+ * @param $obj object
|
|
|
+ **/
|
|
|
+ static function set($key,$obj) {
|
|
|
+ return self::$table[$key]=$obj;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Retrieve object from catalog
|
|
|
+ * @return object
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ static function get($key) {
|
|
|
+ return self::$table[$key];
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Delete object from catalog
|
|
|
+ * @return NULL
|
|
|
+ * @param $key string
|
|
|
+ **/
|
|
|
+ static function clear($key) {
|
|
|
+ self::$table[$key]=NULL;
|
|
|
+ unset(self::$table[$key]);
|
|
|
+ }
|
|
|
+
|
|
|
+ //! Prohibit cloning
|
|
|
+ private function __clone() {
|
|
|
+ }
|
|
|
+
|
|
|
+ //! Prohibit instantiation
|
|
|
+ private function __construct() {
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+return Base::instance();
|