diff options
Diffstat (limited to 'wiki/inc/common.php')
-rw-r--r-- | wiki/inc/common.php | 2087 |
1 files changed, 0 insertions, 2087 deletions
diff --git a/wiki/inc/common.php b/wiki/inc/common.php deleted file mode 100644 index 1fd0154..0000000 --- a/wiki/inc/common.php +++ /dev/null @@ -1,2087 +0,0 @@ -<?php -/** - * Common DokuWiki functions - * - * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) - * @author Andreas Gohr <andi@splitbrain.org> - */ - -if(!defined('DOKU_INC')) die('meh.'); - -/** - * These constants are used with the recents function - */ -define('RECENTS_SKIP_DELETED', 2); -define('RECENTS_SKIP_MINORS', 4); -define('RECENTS_SKIP_SUBSPACES', 8); -define('RECENTS_MEDIA_CHANGES', 16); -define('RECENTS_MEDIA_PAGES_MIXED', 32); - -/** - * Wrapper around htmlspecialchars() - * - * @author Andreas Gohr <andi@splitbrain.org> - * @see htmlspecialchars() - * - * @param string $string the string being converted - * @return string converted string - */ -function hsc($string) { - return htmlspecialchars($string, ENT_QUOTES, 'UTF-8'); -} - -/** - * Checks if the given input is blank - * - * This is similar to empty() but will return false for "0". - * - * Please note: when you pass uninitialized variables, they will implicitly be created - * with a NULL value without warning. - * - * To avoid this it's recommended to guard the call with isset like this: - * - * (isset($foo) && !blank($foo)) - * (!isset($foo) || blank($foo)) - * - * @param $in - * @param bool $trim Consider a string of whitespace to be blank - * @return bool - */ -function blank(&$in, $trim = false) { - if(is_null($in)) return true; - if(is_array($in)) return empty($in); - if($in === "\0") return true; - if($trim && trim($in) === '') return true; - if(strlen($in) > 0) return false; - return empty($in); -} - -/** - * print a newline terminated string - * - * You can give an indention as optional parameter - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $string line of text - * @param int $indent number of spaces indention - */ -function ptln($string, $indent = 0) { - echo str_repeat(' ', $indent)."$string\n"; -} - -/** - * strips control characters (<32) from the given string - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $string being stripped - * @return string - */ -function stripctl($string) { - return preg_replace('/[\x00-\x1F]+/s', '', $string); -} - -/** - * Return a secret token to be used for CSRF attack prevention - * - * @author Andreas Gohr <andi@splitbrain.org> - * @link http://en.wikipedia.org/wiki/Cross-site_request_forgery - * @link http://christ1an.blogspot.com/2007/04/preventing-csrf-efficiently.html - * - * @return string - */ -function getSecurityToken() { - /** @var Input $INPUT */ - global $INPUT; - - $user = $INPUT->server->str('REMOTE_USER'); - $session = session_id(); - - // CSRF checks are only for logged in users - do not generate for anonymous - if(trim($user) == '' || trim($session) == '') return ''; - return PassHash::hmac('md5', $session.$user, auth_cookiesalt()); -} - -/** - * Check the secret CSRF token - * - * @param null|string $token security token or null to read it from request variable - * @return bool success if the token matched - */ -function checkSecurityToken($token = null) { - /** @var Input $INPUT */ - global $INPUT; - if(!$INPUT->server->str('REMOTE_USER')) return true; // no logged in user, no need for a check - - if(is_null($token)) $token = $INPUT->str('sectok'); - if(getSecurityToken() != $token) { - msg('Security Token did not match. Possible CSRF attack.', -1); - return false; - } - return true; -} - -/** - * Print a hidden form field with a secret CSRF token - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param bool $print if true print the field, otherwise html of the field is returned - * @return string html of hidden form field - */ -function formSecurityToken($print = true) { - $ret = '<div class="no"><input type="hidden" name="sectok" value="'.getSecurityToken().'" /></div>'."\n"; - if($print) echo $ret; - return $ret; -} - -/** - * Determine basic information for a request of $id - * - * @author Andreas Gohr <andi@splitbrain.org> - * @author Chris Smith <chris@jalakai.co.uk> - * - * @param string $id pageid - * @param bool $htmlClient add info about whether is mobile browser - * @return array with info for a request of $id - * - */ -function basicinfo($id, $htmlClient=true){ - global $USERINFO; - /* @var Input $INPUT */ - global $INPUT; - - // set info about manager/admin status. - $info = array(); - $info['isadmin'] = false; - $info['ismanager'] = false; - if($INPUT->server->has('REMOTE_USER')) { - $info['userinfo'] = $USERINFO; - $info['perm'] = auth_quickaclcheck($id); - $info['client'] = $INPUT->server->str('REMOTE_USER'); - - if($info['perm'] == AUTH_ADMIN) { - $info['isadmin'] = true; - $info['ismanager'] = true; - } elseif(auth_ismanager()) { - $info['ismanager'] = true; - } - - // if some outside auth were used only REMOTE_USER is set - if(!$info['userinfo']['name']) { - $info['userinfo']['name'] = $INPUT->server->str('REMOTE_USER'); - } - - } else { - $info['perm'] = auth_aclcheck($id, '', null); - $info['client'] = clientIP(true); - } - - $info['namespace'] = getNS($id); - - // mobile detection - if ($htmlClient) { - $info['ismobile'] = clientismobile(); - } - - return $info; - } - -/** - * Return info about the current document as associative - * array. - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @return array with info about current document - */ -function pageinfo() { - global $ID; - global $REV; - global $RANGE; - global $lang; - /* @var Input $INPUT */ - global $INPUT; - - $info = basicinfo($ID); - - // include ID & REV not redundant, as some parts of DokuWiki may temporarily change $ID, e.g. p_wiki_xhtml - // FIXME ... perhaps it would be better to ensure the temporary changes weren't necessary - $info['id'] = $ID; - $info['rev'] = $REV; - - if($INPUT->server->has('REMOTE_USER')) { - $sub = new Subscription(); - $info['subscribed'] = $sub->user_subscription(); - } else { - $info['subscribed'] = false; - } - - $info['locked'] = checklock($ID); - $info['filepath'] = wikiFN($ID); - $info['exists'] = file_exists($info['filepath']); - $info['currentrev'] = @filemtime($info['filepath']); - if($REV) { - //check if current revision was meant - if($info['exists'] && ($info['currentrev'] == $REV)) { - $REV = ''; - } elseif($RANGE) { - //section editing does not work with old revisions! - $REV = ''; - $RANGE = ''; - msg($lang['nosecedit'], 0); - } else { - //really use old revision - $info['filepath'] = wikiFN($ID, $REV); - $info['exists'] = file_exists($info['filepath']); - } - } - $info['rev'] = $REV; - if($info['exists']) { - $info['writable'] = (is_writable($info['filepath']) && - ($info['perm'] >= AUTH_EDIT)); - } else { - $info['writable'] = ($info['perm'] >= AUTH_CREATE); - } - $info['editable'] = ($info['writable'] && empty($info['locked'])); - $info['lastmod'] = @filemtime($info['filepath']); - - //load page meta data - $info['meta'] = p_get_metadata($ID); - - //who's the editor - $pagelog = new PageChangeLog($ID, 1024); - if($REV) { - $revinfo = $pagelog->getRevisionInfo($REV); - } else { - if(!empty($info['meta']['last_change']) && is_array($info['meta']['last_change'])) { - $revinfo = $info['meta']['last_change']; - } else { - $revinfo = $pagelog->getRevisionInfo($info['lastmod']); - // cache most recent changelog line in metadata if missing and still valid - if($revinfo !== false) { - $info['meta']['last_change'] = $revinfo; - p_set_metadata($ID, array('last_change' => $revinfo)); - } - } - } - //and check for an external edit - if($revinfo !== false && $revinfo['date'] != $info['lastmod']) { - // cached changelog line no longer valid - $revinfo = false; - $info['meta']['last_change'] = $revinfo; - p_set_metadata($ID, array('last_change' => $revinfo)); - } - - $info['ip'] = $revinfo['ip']; - $info['user'] = $revinfo['user']; - $info['sum'] = $revinfo['sum']; - // See also $INFO['meta']['last_change'] which is the most recent log line for page $ID. - // Use $INFO['meta']['last_change']['type']===DOKU_CHANGE_TYPE_MINOR_EDIT in place of $info['minor']. - - if($revinfo['user']) { - $info['editor'] = $revinfo['user']; - } else { - $info['editor'] = $revinfo['ip']; - } - - // draft - $draft = getCacheName($info['client'].$ID, '.draft'); - if(file_exists($draft)) { - if(@filemtime($draft) < @filemtime(wikiFN($ID))) { - // remove stale draft - @unlink($draft); - } else { - $info['draft'] = $draft; - } - } - - return $info; -} - -/** - * Initialize and/or fill global $JSINFO with some basic info to be given to javascript - */ -function jsinfo() { - global $JSINFO, $ID, $INFO, $ACT; - - if (!is_array($JSINFO)) { - $JSINFO = []; - } - //export minimal info to JS, plugins can add more - $JSINFO['id'] = $ID; - $JSINFO['namespace'] = (string) $INFO['namespace']; - $JSINFO['ACT'] = act_clean($ACT); - $JSINFO['useHeadingNavigation'] = (int) useHeading('navigation'); - $JSINFO['useHeadingContent'] = (int) useHeading('content'); -} - -/** - * Return information about the current media item as an associative array. - * - * @return array with info about current media item - */ -function mediainfo(){ - global $NS; - global $IMG; - - $info = basicinfo("$NS:*"); - $info['image'] = $IMG; - - return $info; -} - -/** - * Build an string of URL parameters - * - * @author Andreas Gohr - * - * @param array $params array with key-value pairs - * @param string $sep series of pairs are separated by this character - * @return string query string - */ -function buildURLparams($params, $sep = '&') { - $url = ''; - $amp = false; - foreach($params as $key => $val) { - if($amp) $url .= $sep; - - $url .= rawurlencode($key).'='; - $url .= rawurlencode((string) $val); - $amp = true; - } - return $url; -} - -/** - * Build an string of html tag attributes - * - * Skips keys starting with '_', values get HTML encoded - * - * @author Andreas Gohr - * - * @param array $params array with (attribute name-attribute value) pairs - * @param bool $skipempty skip empty string values? - * @return string - */ -function buildAttributes($params, $skipempty = false) { - $url = ''; - $white = false; - foreach($params as $key => $val) { - if($key{0} == '_') continue; - if($val === '' && $skipempty) continue; - if($white) $url .= ' '; - - $url .= $key.'="'; - $url .= htmlspecialchars($val); - $url .= '"'; - $white = true; - } - return $url; -} - -/** - * This builds the breadcrumb trail and returns it as array - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @return string[] with the data: array(pageid=>name, ... ) - */ -function breadcrumbs() { - // we prepare the breadcrumbs early for quick session closing - static $crumbs = null; - if($crumbs != null) return $crumbs; - - global $ID; - global $ACT; - global $conf; - - //first visit? - $crumbs = isset($_SESSION[DOKU_COOKIE]['bc']) ? $_SESSION[DOKU_COOKIE]['bc'] : array(); - //we only save on show and existing visible wiki documents - $file = wikiFN($ID); - if($ACT != 'show' || isHiddenPage($ID) || !file_exists($file)) { - $_SESSION[DOKU_COOKIE]['bc'] = $crumbs; - return $crumbs; - } - - // page names - $name = noNSorNS($ID); - if(useHeading('navigation')) { - // get page title - $title = p_get_first_heading($ID, METADATA_RENDER_USING_SIMPLE_CACHE); - if($title) { - $name = $title; - } - } - - //remove ID from array - if(isset($crumbs[$ID])) { - unset($crumbs[$ID]); - } - - //add to array - $crumbs[$ID] = $name; - //reduce size - while(count($crumbs) > $conf['breadcrumbs']) { - array_shift($crumbs); - } - //save to session - $_SESSION[DOKU_COOKIE]['bc'] = $crumbs; - return $crumbs; -} - -/** - * Filter for page IDs - * - * This is run on a ID before it is outputted somewhere - * currently used to replace the colon with something else - * on Windows (non-IIS) systems and to have proper URL encoding - * - * See discussions at https://github.com/splitbrain/dokuwiki/pull/84 and - * https://github.com/splitbrain/dokuwiki/pull/173 why we use a whitelist of - * unaffected servers instead of blacklisting affected servers here. - * - * Urlencoding is ommitted when the second parameter is false - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id pageid being filtered - * @param bool $ue apply urlencoding? - * @return string - */ -function idfilter($id, $ue = true) { - global $conf; - /* @var Input $INPUT */ - global $INPUT; - - if($conf['useslash'] && $conf['userewrite']) { - $id = strtr($id, ':', '/'); - } elseif(strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' && - $conf['userewrite'] && - strpos($INPUT->server->str('SERVER_SOFTWARE'), 'Microsoft-IIS') === false - ) { - $id = strtr($id, ':', ';'); - } - if($ue) { - $id = rawurlencode($id); - $id = str_replace('%3A', ':', $id); //keep as colon - $id = str_replace('%3B', ';', $id); //keep as semicolon - $id = str_replace('%2F', '/', $id); //keep as slash - } - return $id; -} - -/** - * This builds a link to a wikipage - * - * It handles URL rewriting and adds additional parameters - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id, defaults to start page - * @param string|array $urlParameters URL parameters, associative array recommended - * @param bool $absolute request an absolute URL instead of relative - * @param string $separator parameter separator - * @return string - */ -function wl($id = '', $urlParameters = '', $absolute = false, $separator = '&') { - global $conf; - if(is_array($urlParameters)) { - if(isset($urlParameters['rev']) && !$urlParameters['rev']) unset($urlParameters['rev']); - if(isset($urlParameters['at']) && $conf['date_at_format']) $urlParameters['at'] = date($conf['date_at_format'],$urlParameters['at']); - $urlParameters = buildURLparams($urlParameters, $separator); - } else { - $urlParameters = str_replace(',', $separator, $urlParameters); - } - if($id === '') { - $id = $conf['start']; - } - $id = idfilter($id); - if($absolute) { - $xlink = DOKU_URL; - } else { - $xlink = DOKU_BASE; - } - - if($conf['userewrite'] == 2) { - $xlink .= DOKU_SCRIPT.'/'.$id; - if($urlParameters) $xlink .= '?'.$urlParameters; - } elseif($conf['userewrite']) { - $xlink .= $id; - if($urlParameters) $xlink .= '?'.$urlParameters; - } elseif($id) { - $xlink .= DOKU_SCRIPT.'?id='.$id; - if($urlParameters) $xlink .= $separator.$urlParameters; - } else { - $xlink .= DOKU_SCRIPT; - if($urlParameters) $xlink .= '?'.$urlParameters; - } - - return $xlink; -} - -/** - * This builds a link to an alternate page format - * - * Handles URL rewriting if enabled. Follows the style of wl(). - * - * @author Ben Coburn <btcoburn@silicodon.net> - * @param string $id page id, defaults to start page - * @param string $format the export renderer to use - * @param string|array $urlParameters URL parameters, associative array recommended - * @param bool $abs request an absolute URL instead of relative - * @param string $sep parameter separator - * @return string - */ -function exportlink($id = '', $format = 'raw', $urlParameters = '', $abs = false, $sep = '&') { - global $conf; - if(is_array($urlParameters)) { - $urlParameters = buildURLparams($urlParameters, $sep); - } else { - $urlParameters = str_replace(',', $sep, $urlParameters); - } - - $format = rawurlencode($format); - $id = idfilter($id); - if($abs) { - $xlink = DOKU_URL; - } else { - $xlink = DOKU_BASE; - } - - if($conf['userewrite'] == 2) { - $xlink .= DOKU_SCRIPT.'/'.$id.'?do=export_'.$format; - if($urlParameters) $xlink .= $sep.$urlParameters; - } elseif($conf['userewrite'] == 1) { - $xlink .= '_export/'.$format.'/'.$id; - if($urlParameters) $xlink .= '?'.$urlParameters; - } else { - $xlink .= DOKU_SCRIPT.'?do=export_'.$format.$sep.'id='.$id; - if($urlParameters) $xlink .= $sep.$urlParameters; - } - - return $xlink; -} - -/** - * Build a link to a media file - * - * Will return a link to the detail page if $direct is false - * - * The $more parameter should always be given as array, the function then - * will strip default parameters to produce even cleaner URLs - * - * @param string $id the media file id or URL - * @param mixed $more string or array with additional parameters - * @param bool $direct link to detail page if false - * @param string $sep URL parameter separator - * @param bool $abs Create an absolute URL - * @return string - */ -function ml($id = '', $more = '', $direct = true, $sep = '&', $abs = false) { - global $conf; - $isexternalimage = media_isexternal($id); - if(!$isexternalimage) { - $id = cleanID($id); - } - - if(is_array($more)) { - // add token for resized images - if(!empty($more['w']) || !empty($more['h']) || $isexternalimage){ - $more['tok'] = media_get_token($id,$more['w'],$more['h']); - } - // strip defaults for shorter URLs - if(isset($more['cache']) && $more['cache'] == 'cache') unset($more['cache']); - if(empty($more['w'])) unset($more['w']); - if(empty($more['h'])) unset($more['h']); - if(isset($more['id']) && $direct) unset($more['id']); - if(isset($more['rev']) && !$more['rev']) unset($more['rev']); - $more = buildURLparams($more, $sep); - } else { - $matches = array(); - if (preg_match_all('/\b(w|h)=(\d*)\b/',$more,$matches,PREG_SET_ORDER) || $isexternalimage){ - $resize = array('w'=>0, 'h'=>0); - foreach ($matches as $match){ - $resize[$match[1]] = $match[2]; - } - $more .= $more === '' ? '' : $sep; - $more .= 'tok='.media_get_token($id,$resize['w'],$resize['h']); - } - $more = str_replace('cache=cache', '', $more); //skip default - $more = str_replace(',,', ',', $more); - $more = str_replace(',', $sep, $more); - } - - if($abs) { - $xlink = DOKU_URL; - } else { - $xlink = DOKU_BASE; - } - - // external URLs are always direct without rewriting - if($isexternalimage) { - $xlink .= 'lib/exe/fetch.php'; - $xlink .= '?'.$more; - $xlink .= $sep.'media='.rawurlencode($id); - return $xlink; - } - - $id = idfilter($id); - - // decide on scriptname - if($direct) { - if($conf['userewrite'] == 1) { - $script = '_media'; - } else { - $script = 'lib/exe/fetch.php'; - } - } else { - if($conf['userewrite'] == 1) { - $script = '_detail'; - } else { - $script = 'lib/exe/detail.php'; - } - } - - // build URL based on rewrite mode - if($conf['userewrite']) { - $xlink .= $script.'/'.$id; - if($more) $xlink .= '?'.$more; - } else { - if($more) { - $xlink .= $script.'?'.$more; - $xlink .= $sep.'media='.$id; - } else { - $xlink .= $script.'?media='.$id; - } - } - - return $xlink; -} - -/** - * Returns the URL to the DokuWiki base script - * - * Consider using wl() instead, unless you absoutely need the doku.php endpoint - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @return string - */ -function script() { - return DOKU_BASE.DOKU_SCRIPT; -} - -/** - * Spamcheck against wordlist - * - * Checks the wikitext against a list of blocked expressions - * returns true if the text contains any bad words - * - * Triggers COMMON_WORDBLOCK_BLOCKED - * - * Action Plugins can use this event to inspect the blocked data - * and gain information about the user who was blocked. - * - * Event data: - * data['matches'] - array of matches - * data['userinfo'] - information about the blocked user - * [ip] - ip address - * [user] - username (if logged in) - * [mail] - mail address (if logged in) - * [name] - real name (if logged in) - * - * @author Andreas Gohr <andi@splitbrain.org> - * @author Michael Klier <chi@chimeric.de> - * - * @param string $text - optional text to check, if not given the globals are used - * @return bool - true if a spam word was found - */ -function checkwordblock($text = '') { - global $TEXT; - global $PRE; - global $SUF; - global $SUM; - global $conf; - global $INFO; - /* @var Input $INPUT */ - global $INPUT; - - if(!$conf['usewordblock']) return false; - - if(!$text) $text = "$PRE $TEXT $SUF $SUM"; - - // we prepare the text a tiny bit to prevent spammers circumventing URL checks - $text = preg_replace('!(\b)(www\.[\w.:?\-;,]+?\.[\w.:?\-;,]+?[\w/\#~:.?+=&%@\!\-.:?\-;,]+?)([.:?\-;,]*[^\w/\#~:.?+=&%@\!\-.:?\-;,])!i', '\1http://\2 \2\3', $text); - - $wordblocks = getWordblocks(); - // how many lines to read at once (to work around some PCRE limits) - if(version_compare(phpversion(), '4.3.0', '<')) { - // old versions of PCRE define a maximum of parenthesises even if no - // backreferences are used - the maximum is 99 - // this is very bad performancewise and may even be too high still - $chunksize = 40; - } else { - // read file in chunks of 200 - this should work around the - // MAX_PATTERN_SIZE in modern PCRE - $chunksize = 200; - } - while($blocks = array_splice($wordblocks, 0, $chunksize)) { - $re = array(); - // build regexp from blocks - foreach($blocks as $block) { - $block = preg_replace('/#.*$/', '', $block); - $block = trim($block); - if(empty($block)) continue; - $re[] = $block; - } - if(count($re) && preg_match('#('.join('|', $re).')#si', $text, $matches)) { - // prepare event data - $data = array(); - $data['matches'] = $matches; - $data['userinfo']['ip'] = $INPUT->server->str('REMOTE_ADDR'); - if($INPUT->server->str('REMOTE_USER')) { - $data['userinfo']['user'] = $INPUT->server->str('REMOTE_USER'); - $data['userinfo']['name'] = $INFO['userinfo']['name']; - $data['userinfo']['mail'] = $INFO['userinfo']['mail']; - } - $callback = function () { - return true; - }; - return trigger_event('COMMON_WORDBLOCK_BLOCKED', $data, $callback, true); - } - } - return false; -} - -/** - * Return the IP of the client - * - * Honours X-Forwarded-For and X-Real-IP Proxy Headers - * - * It returns a comma separated list of IPs if the above mentioned - * headers are set. If the single parameter is set, it tries to return - * a routable public address, prefering the ones suplied in the X - * headers - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param boolean $single If set only a single IP is returned - * @return string - */ -function clientIP($single = false) { - /* @var Input $INPUT */ - global $INPUT; - - $ip = array(); - $ip[] = $INPUT->server->str('REMOTE_ADDR'); - if($INPUT->server->str('HTTP_X_FORWARDED_FOR')) { - $ip = array_merge($ip, explode(',', str_replace(' ', '', $INPUT->server->str('HTTP_X_FORWARDED_FOR')))); - } - if($INPUT->server->str('HTTP_X_REAL_IP')) { - $ip = array_merge($ip, explode(',', str_replace(' ', '', $INPUT->server->str('HTTP_X_REAL_IP')))); - } - - // some IPv4/v6 regexps borrowed from Feyd - // see: http://forums.devnetwork.net/viewtopic.php?f=38&t=53479 - $dec_octet = '(?:25[0-5]|2[0-4]\d|1\d\d|[1-9]\d|[0-9])'; - $hex_digit = '[A-Fa-f0-9]'; - $h16 = "{$hex_digit}{1,4}"; - $IPv4Address = "$dec_octet\\.$dec_octet\\.$dec_octet\\.$dec_octet"; - $ls32 = "(?:$h16:$h16|$IPv4Address)"; - $IPv6Address = - "(?:(?:{$IPv4Address})|(?:". - "(?:$h16:){6}$ls32". - "|::(?:$h16:){5}$ls32". - "|(?:$h16)?::(?:$h16:){4}$ls32". - "|(?:(?:$h16:){0,1}$h16)?::(?:$h16:){3}$ls32". - "|(?:(?:$h16:){0,2}$h16)?::(?:$h16:){2}$ls32". - "|(?:(?:$h16:){0,3}$h16)?::(?:$h16:){1}$ls32". - "|(?:(?:$h16:){0,4}$h16)?::$ls32". - "|(?:(?:$h16:){0,5}$h16)?::$h16". - "|(?:(?:$h16:){0,6}$h16)?::". - ")(?:\\/(?:12[0-8]|1[0-1][0-9]|[1-9][0-9]|[0-9]))?)"; - - // remove any non-IP stuff - $cnt = count($ip); - $match = array(); - for($i = 0; $i < $cnt; $i++) { - if(preg_match("/^$IPv4Address$/", $ip[$i], $match) || preg_match("/^$IPv6Address$/", $ip[$i], $match)) { - $ip[$i] = $match[0]; - } else { - $ip[$i] = ''; - } - if(empty($ip[$i])) unset($ip[$i]); - } - $ip = array_values(array_unique($ip)); - if(!$ip[0]) $ip[0] = '0.0.0.0'; // for some strange reason we don't have a IP - - if(!$single) return join(',', $ip); - - // decide which IP to use, trying to avoid local addresses - $ip = array_reverse($ip); - foreach($ip as $i) { - if(preg_match('/^(::1|[fF][eE]80:|127\.|10\.|192\.168\.|172\.((1[6-9])|(2[0-9])|(3[0-1]))\.)/', $i)) { - continue; - } else { - return $i; - } - } - // still here? just use the first (last) address - return $ip[0]; -} - -/** - * Check if the browser is on a mobile device - * - * Adapted from the example code at url below - * - * @link http://www.brainhandles.com/2007/10/15/detecting-mobile-browsers/#code - * - * @return bool if true, client is mobile browser; otherwise false - */ -function clientismobile() { - /* @var Input $INPUT */ - global $INPUT; - - if($INPUT->server->has('HTTP_X_WAP_PROFILE')) return true; - - if(preg_match('/wap\.|\.wap/i', $INPUT->server->str('HTTP_ACCEPT'))) return true; - - if(!$INPUT->server->has('HTTP_USER_AGENT')) return false; - - $uamatches = 'midp|j2me|avantg|docomo|novarra|palmos|palmsource|240x320|opwv|chtml|pda|windows ce|mmp\/|blackberry|mib\/|symbian|wireless|nokia|hand|mobi|phone|cdm|up\.b|audio|SIE\-|SEC\-|samsung|HTC|mot\-|mitsu|sagem|sony|alcatel|lg|erics|vx|NEC|philips|mmm|xx|panasonic|sharp|wap|sch|rover|pocket|benq|java|pt|pg|vox|amoi|bird|compal|kg|voda|sany|kdd|dbt|sendo|sgh|gradi|jb|\d\d\di|moto'; - - if(preg_match("/$uamatches/i", $INPUT->server->str('HTTP_USER_AGENT'))) return true; - - return false; -} - -/** - * check if a given link is interwiki link - * - * @param string $link the link, e.g. "wiki>page" - * @return bool - */ -function link_isinterwiki($link){ - if (preg_match('/^[a-zA-Z0-9\.]+>/u',$link)) return true; - return false; -} - -/** - * Convert one or more comma separated IPs to hostnames - * - * If $conf['dnslookups'] is disabled it simply returns the input string - * - * @author Glen Harris <astfgl@iamnota.org> - * - * @param string $ips comma separated list of IP addresses - * @return string a comma separated list of hostnames - */ -function gethostsbyaddrs($ips) { - global $conf; - if(!$conf['dnslookups']) return $ips; - - $hosts = array(); - $ips = explode(',', $ips); - - if(is_array($ips)) { - foreach($ips as $ip) { - $hosts[] = gethostbyaddr(trim($ip)); - } - return join(',', $hosts); - } else { - return gethostbyaddr(trim($ips)); - } -} - -/** - * Checks if a given page is currently locked. - * - * removes stale lockfiles - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id - * @return bool page is locked? - */ -function checklock($id) { - global $conf; - /* @var Input $INPUT */ - global $INPUT; - - $lock = wikiLockFN($id); - - //no lockfile - if(!file_exists($lock)) return false; - - //lockfile expired - if((time() - filemtime($lock)) > $conf['locktime']) { - @unlink($lock); - return false; - } - - //my own lock - @list($ip, $session) = explode("\n", io_readFile($lock)); - if($ip == $INPUT->server->str('REMOTE_USER') || $ip == clientIP() || (session_id() && $session == session_id())) { - return false; - } - - return $ip; -} - -/** - * Lock a page for editing - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id to lock - */ -function lock($id) { - global $conf; - /* @var Input $INPUT */ - global $INPUT; - - if($conf['locktime'] == 0) { - return; - } - - $lock = wikiLockFN($id); - if($INPUT->server->str('REMOTE_USER')) { - io_saveFile($lock, $INPUT->server->str('REMOTE_USER')); - } else { - io_saveFile($lock, clientIP()."\n".session_id()); - } -} - -/** - * Unlock a page if it was locked by the user - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id to unlock - * @return bool true if a lock was removed - */ -function unlock($id) { - /* @var Input $INPUT */ - global $INPUT; - - $lock = wikiLockFN($id); - if(file_exists($lock)) { - @list($ip, $session) = explode("\n", io_readFile($lock)); - if($ip == $INPUT->server->str('REMOTE_USER') || $ip == clientIP() || $session == session_id()) { - @unlink($lock); - return true; - } - } - return false; -} - -/** - * convert line ending to unix format - * - * also makes sure the given text is valid UTF-8 - * - * @see formText() for 2crlf conversion - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $text - * @return string - */ -function cleanText($text) { - $text = preg_replace("/(\015\012)|(\015)/", "\012", $text); - - // if the text is not valid UTF-8 we simply assume latin1 - // this won't break any worse than it breaks with the wrong encoding - // but might actually fix the problem in many cases - if(!utf8_check($text)) $text = utf8_encode($text); - - return $text; -} - -/** - * Prepares text for print in Webforms by encoding special chars. - * It also converts line endings to Windows format which is - * pseudo standard for webforms. - * - * @see cleanText() for 2unix conversion - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $text - * @return string - */ -function formText($text) { - $text = str_replace("\012", "\015\012", $text); - return htmlspecialchars($text); -} - -/** - * Returns the specified local text in raw format - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id - * @param string $ext extension of file being read, default 'txt' - * @return string - */ -function rawLocale($id, $ext = 'txt') { - return io_readFile(localeFN($id, $ext)); -} - -/** - * Returns the raw WikiText - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id - * @param string|int $rev timestamp when a revision of wikitext is desired - * @return string - */ -function rawWiki($id, $rev = '') { - return io_readWikiPage(wikiFN($id, $rev), $id, $rev); -} - -/** - * Returns the pagetemplate contents for the ID's namespace - * - * @triggers COMMON_PAGETPL_LOAD - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id the id of the page to be created - * @return string parsed pagetemplate content - */ -function pageTemplate($id) { - global $conf; - - if(is_array($id)) $id = $id[0]; - - // prepare initial event data - $data = array( - 'id' => $id, // the id of the page to be created - 'tpl' => '', // the text used as template - 'tplfile' => '', // the file above text was/should be loaded from - 'doreplace' => true // should wildcard replacements be done on the text? - ); - - $evt = new Doku_Event('COMMON_PAGETPL_LOAD', $data); - if($evt->advise_before(true)) { - // the before event might have loaded the content already - if(empty($data['tpl'])) { - // if the before event did not set a template file, try to find one - if(empty($data['tplfile'])) { - $path = dirname(wikiFN($id)); - if(file_exists($path.'/_template.txt')) { - $data['tplfile'] = $path.'/_template.txt'; - } else { - // search upper namespaces for templates - $len = strlen(rtrim($conf['datadir'], '/')); - while(strlen($path) >= $len) { - if(file_exists($path.'/__template.txt')) { - $data['tplfile'] = $path.'/__template.txt'; - break; - } - $path = substr($path, 0, strrpos($path, '/')); - } - } - } - // load the content - $data['tpl'] = io_readFile($data['tplfile']); - } - if($data['doreplace']) parsePageTemplate($data); - } - $evt->advise_after(); - unset($evt); - - return $data['tpl']; -} - -/** - * Performs common page template replacements - * This works on data from COMMON_PAGETPL_LOAD - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param array $data array with event data - * @return string - */ -function parsePageTemplate(&$data) { - /** - * @var string $id the id of the page to be created - * @var string $tpl the text used as template - * @var string $tplfile the file above text was/should be loaded from - * @var bool $doreplace should wildcard replacements be done on the text? - */ - extract($data); - - global $USERINFO; - global $conf; - /* @var Input $INPUT */ - global $INPUT; - - // replace placeholders - $file = noNS($id); - $page = strtr($file, $conf['sepchar'], ' '); - - $tpl = str_replace( - array( - '@ID@', - '@NS@', - '@CURNS@', - '@FILE@', - '@!FILE@', - '@!FILE!@', - '@PAGE@', - '@!PAGE@', - '@!!PAGE@', - '@!PAGE!@', - '@USER@', - '@NAME@', - '@MAIL@', - '@DATE@', - ), - array( - $id, - getNS($id), - curNS($id), - $file, - utf8_ucfirst($file), - utf8_strtoupper($file), - $page, - utf8_ucfirst($page), - utf8_ucwords($page), - utf8_strtoupper($page), - $INPUT->server->str('REMOTE_USER'), - $USERINFO['name'], - $USERINFO['mail'], - $conf['dformat'], - ), $tpl - ); - - // we need the callback to work around strftime's char limit - $tpl = preg_replace_callback( - '/%./', - function ($m) { - return strftime($m[0]); - }, - $tpl - ); - $data['tpl'] = $tpl; - return $tpl; -} - -/** - * Returns the raw Wiki Text in three slices. - * - * The range parameter needs to have the form "from-to" - * and gives the range of the section in bytes - no - * UTF-8 awareness is needed. - * The returned order is prefix, section and suffix. - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $range in form "from-to" - * @param string $id page id - * @param string $rev optional, the revision timestamp - * @return string[] with three slices - */ -function rawWikiSlices($range, $id, $rev = '') { - $text = io_readWikiPage(wikiFN($id, $rev), $id, $rev); - - // Parse range - list($from, $to) = explode('-', $range, 2); - // Make range zero-based, use defaults if marker is missing - $from = !$from ? 0 : ($from - 1); - $to = !$to ? strlen($text) : ($to - 1); - - $slices = array(); - $slices[0] = substr($text, 0, $from); - $slices[1] = substr($text, $from, $to - $from); - $slices[2] = substr($text, $to); - return $slices; -} - -/** - * Joins wiki text slices - * - * function to join the text slices. - * When the pretty parameter is set to true it adds additional empty - * lines between sections if needed (used on saving). - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $pre prefix - * @param string $text text in the middle - * @param string $suf suffix - * @param bool $pretty add additional empty lines between sections - * @return string - */ -function con($pre, $text, $suf, $pretty = false) { - if($pretty) { - if($pre !== '' && substr($pre, -1) !== "\n" && - substr($text, 0, 1) !== "\n" - ) { - $pre .= "\n"; - } - if($suf !== '' && substr($text, -1) !== "\n" && - substr($suf, 0, 1) !== "\n" - ) { - $text .= "\n"; - } - } - - return $pre.$text.$suf; -} - -/** - * Checks if the current page version is newer than the last entry in the page's - * changelog. If so, we assume it has been an external edit and we create an - * attic copy and add a proper changelog line. - * - * This check is only executed when the page is about to be saved again from the - * wiki, triggered in @see saveWikiText() - * - * @param string $id the page ID - */ -function detectExternalEdit($id) { - global $lang; - - $fileLastMod = wikiFN($id); - $lastMod = @filemtime($fileLastMod); // from page - $pagelog = new PageChangeLog($id, 1024); - $lastRev = $pagelog->getRevisions(-1, 1); // from changelog - $lastRev = (int) (empty($lastRev) ? 0 : $lastRev[0]); - - if(!file_exists(wikiFN($id, $lastMod)) && file_exists($fileLastMod) && $lastMod >= $lastRev) { - // add old revision to the attic if missing - saveOldRevision($id); - // add a changelog entry if this edit came from outside dokuwiki - if($lastMod > $lastRev) { - $fileLastRev = wikiFN($id, $lastRev); - $revinfo = $pagelog->getRevisionInfo($lastRev); - if(empty($lastRev) || !file_exists($fileLastRev) || $revinfo['type'] == DOKU_CHANGE_TYPE_DELETE) { - $filesize_old = 0; - } else { - $filesize_old = io_getSizeFile($fileLastRev); - } - $filesize_new = filesize($fileLastMod); - $sizechange = $filesize_new - $filesize_old; - - addLogEntry($lastMod, $id, DOKU_CHANGE_TYPE_EDIT, $lang['external_edit'], '', array('ExternalEdit'=> true), $sizechange); - // remove soon to be stale instructions - $cache = new cache_instructions($id, $fileLastMod); - $cache->removeCache(); - } - } -} - -/** - * Saves a wikitext by calling io_writeWikiPage. - * Also directs changelog and attic updates. - * - * @author Andreas Gohr <andi@splitbrain.org> - * @author Ben Coburn <btcoburn@silicodon.net> - * - * @param string $id page id - * @param string $text wikitext being saved - * @param string $summary summary of text update - * @param bool $minor mark this saved version as minor update - */ -function saveWikiText($id, $text, $summary, $minor = false) { - /* Note to developers: - This code is subtle and delicate. Test the behavior of - the attic and changelog with dokuwiki and external edits - after any changes. External edits change the wiki page - directly without using php or dokuwiki. - */ - global $conf; - global $lang; - global $REV; - /* @var Input $INPUT */ - global $INPUT; - - // prepare data for event - $svdta = array(); - $svdta['id'] = $id; - $svdta['file'] = wikiFN($id); - $svdta['revertFrom'] = $REV; - $svdta['oldRevision'] = @filemtime($svdta['file']); - $svdta['newRevision'] = 0; - $svdta['newContent'] = $text; - $svdta['oldContent'] = rawWiki($id); - $svdta['summary'] = $summary; - $svdta['contentChanged'] = ($svdta['newContent'] != $svdta['oldContent']); - $svdta['changeInfo'] = ''; - $svdta['changeType'] = DOKU_CHANGE_TYPE_EDIT; - $svdta['sizechange'] = null; - - // select changelog line type - if($REV) { - $svdta['changeType'] = DOKU_CHANGE_TYPE_REVERT; - $svdta['changeInfo'] = $REV; - } else if(!file_exists($svdta['file'])) { - $svdta['changeType'] = DOKU_CHANGE_TYPE_CREATE; - } else if(trim($text) == '') { - // empty or whitespace only content deletes - $svdta['changeType'] = DOKU_CHANGE_TYPE_DELETE; - // autoset summary on deletion - if(blank($svdta['summary'])) { - $svdta['summary'] = $lang['deleted']; - } - } else if($minor && $conf['useacl'] && $INPUT->server->str('REMOTE_USER')) { - //minor edits only for logged in users - $svdta['changeType'] = DOKU_CHANGE_TYPE_MINOR_EDIT; - } - - $event = new Doku_Event('COMMON_WIKIPAGE_SAVE', $svdta); - if(!$event->advise_before()) return; - - // if the content has not been changed, no save happens (plugins may override this) - if(!$svdta['contentChanged']) return; - - detectExternalEdit($id); - - if( - $svdta['changeType'] == DOKU_CHANGE_TYPE_CREATE || - ($svdta['changeType'] == DOKU_CHANGE_TYPE_REVERT && !file_exists($svdta['file'])) - ) { - $filesize_old = 0; - } else { - $filesize_old = filesize($svdta['file']); - } - if($svdta['changeType'] == DOKU_CHANGE_TYPE_DELETE) { - // Send "update" event with empty data, so plugins can react to page deletion - $data = array(array($svdta['file'], '', false), getNS($id), noNS($id), false); - trigger_event('IO_WIKIPAGE_WRITE', $data); - // pre-save deleted revision - @touch($svdta['file']); - clearstatcache(); - $svdta['newRevision'] = saveOldRevision($id); - // remove empty file - @unlink($svdta['file']); - $filesize_new = 0; - // don't remove old meta info as it should be saved, plugins can use IO_WIKIPAGE_WRITE for removing their metadata... - // purge non-persistant meta data - p_purge_metadata($id); - // remove empty namespaces - io_sweepNS($id, 'datadir'); - io_sweepNS($id, 'mediadir'); - } else { - // save file (namespace dir is created in io_writeWikiPage) - io_writeWikiPage($svdta['file'], $svdta['newContent'], $id); - // pre-save the revision, to keep the attic in sync - $svdta['newRevision'] = saveOldRevision($id); - $filesize_new = filesize($svdta['file']); - } - $svdta['sizechange'] = $filesize_new - $filesize_old; - - $event->advise_after(); - - addLogEntry($svdta['newRevision'], $svdta['id'], $svdta['changeType'], $svdta['summary'], $svdta['changeInfo'], null, $svdta['sizechange']); - - // send notify mails - notify($svdta['id'], 'admin', $svdta['oldRevision'], $svdta['summary'], $minor); - notify($svdta['id'], 'subscribers', $svdta['oldRevision'], $svdta['summary'], $minor); - - // update the purgefile (timestamp of the last time anything within the wiki was changed) - io_saveFile($conf['cachedir'].'/purgefile', time()); - - // if useheading is enabled, purge the cache of all linking pages - if(useHeading('content')) { - $pages = ft_backlinks($id, true); - foreach($pages as $page) { - $cache = new cache_renderer($page, wikiFN($page), 'xhtml'); - $cache->removeCache(); - } - } -} - -/** - * moves the current version to the attic and returns its - * revision date - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $id page id - * @return int|string revision timestamp - */ -function saveOldRevision($id) { - $oldf = wikiFN($id); - if(!file_exists($oldf)) return ''; - $date = filemtime($oldf); - $newf = wikiFN($id, $date); - io_writeWikiPage($newf, rawWiki($id), $id, $date); - return $date; -} - -/** - * Sends a notify mail on page change or registration - * - * @param string $id The changed page - * @param string $who Who to notify (admin|subscribers|register) - * @param int|string $rev Old page revision - * @param string $summary What changed - * @param boolean $minor Is this a minor edit? - * @param string[] $replace Additional string substitutions, @KEY@ to be replaced by value - * @return bool - * - * @author Andreas Gohr <andi@splitbrain.org> - */ -function notify($id, $who, $rev = '', $summary = '', $minor = false, $replace = array()) { - global $conf; - /* @var Input $INPUT */ - global $INPUT; - - // decide if there is something to do, eg. whom to mail - if($who == 'admin') { - if(empty($conf['notify'])) return false; //notify enabled? - $tpl = 'mailtext'; - $to = $conf['notify']; - } elseif($who == 'subscribers') { - if(!actionOK('subscribe')) return false; //subscribers enabled? - if($conf['useacl'] && $INPUT->server->str('REMOTE_USER') && $minor) return false; //skip minors - $data = array('id' => $id, 'addresslist' => '', 'self' => false, 'replacements' => $replace); - trigger_event( - 'COMMON_NOTIFY_ADDRESSLIST', $data, - array(new Subscription(), 'notifyaddresses') - ); - $to = $data['addresslist']; - if(empty($to)) return false; - $tpl = 'subscr_single'; - } else { - return false; //just to be safe - } - - // prepare content - $subscription = new Subscription(); - return $subscription->send_diff($to, $tpl, $id, $rev, $summary); -} - -/** - * extracts the query from a search engine referrer - * - * @author Andreas Gohr <andi@splitbrain.org> - * @author Todd Augsburger <todd@rollerorgans.com> - * - * @return array|string - */ -function getGoogleQuery() { - /* @var Input $INPUT */ - global $INPUT; - - if(!$INPUT->server->has('HTTP_REFERER')) { - return ''; - } - $url = parse_url($INPUT->server->str('HTTP_REFERER')); - - // only handle common SEs - if(!preg_match('/(google|bing|yahoo|ask|duckduckgo|babylon|aol|yandex)/',$url['host'])) return ''; - - $query = array(); - // temporary workaround against PHP bug #49733 - // see http://bugs.php.net/bug.php?id=49733 - if(UTF8_MBSTRING) $enc = mb_internal_encoding(); - parse_str($url['query'], $query); - if(UTF8_MBSTRING) mb_internal_encoding($enc); - - $q = ''; - if(isset($query['q'])){ - $q = $query['q']; - }elseif(isset($query['p'])){ - $q = $query['p']; - }elseif(isset($query['query'])){ - $q = $query['query']; - } - $q = trim($q); - - if(!$q) return ''; - $q = preg_split('/[\s\'"\\\\`()\]\[?:!\.{};,#+*<>\\/]+/', $q, -1, PREG_SPLIT_NO_EMPTY); - return $q; -} - -/** - * Return the human readable size of a file - * - * @param int $size A file size - * @param int $dec A number of decimal places - * @return string human readable size - * - * @author Martin Benjamin <b.martin@cybernet.ch> - * @author Aidan Lister <aidan@php.net> - * @version 1.0.0 - */ -function filesize_h($size, $dec = 1) { - $sizes = array('B', 'KB', 'MB', 'GB'); - $count = count($sizes); - $i = 0; - - while($size >= 1024 && ($i < $count - 1)) { - $size /= 1024; - $i++; - } - - return round($size, $dec)."\xC2\xA0".$sizes[$i]; //non-breaking space -} - -/** - * Return the given timestamp as human readable, fuzzy age - * - * @author Andreas Gohr <gohr@cosmocode.de> - * - * @param int $dt timestamp - * @return string - */ -function datetime_h($dt) { - global $lang; - - $ago = time() - $dt; - if($ago > 24 * 60 * 60 * 30 * 12 * 2) { - return sprintf($lang['years'], round($ago / (24 * 60 * 60 * 30 * 12))); - } - if($ago > 24 * 60 * 60 * 30 * 2) { - return sprintf($lang['months'], round($ago / (24 * 60 * 60 * 30))); - } - if($ago > 24 * 60 * 60 * 7 * 2) { - return sprintf($lang['weeks'], round($ago / (24 * 60 * 60 * 7))); - } - if($ago > 24 * 60 * 60 * 2) { - return sprintf($lang['days'], round($ago / (24 * 60 * 60))); - } - if($ago > 60 * 60 * 2) { - return sprintf($lang['hours'], round($ago / (60 * 60))); - } - if($ago > 60 * 2) { - return sprintf($lang['minutes'], round($ago / (60))); - } - return sprintf($lang['seconds'], $ago); -} - -/** - * Wraps around strftime but provides support for fuzzy dates - * - * The format default to $conf['dformat']. It is passed to - * strftime - %f can be used to get the value from datetime_h() - * - * @see datetime_h - * @author Andreas Gohr <gohr@cosmocode.de> - * - * @param int|null $dt timestamp when given, null will take current timestamp - * @param string $format empty default to $conf['dformat'], or provide format as recognized by strftime() - * @return string - */ -function dformat($dt = null, $format = '') { - global $conf; - - if(is_null($dt)) $dt = time(); - $dt = (int) $dt; - if(!$format) $format = $conf['dformat']; - - $format = str_replace('%f', datetime_h($dt), $format); - return strftime($format, $dt); -} - -/** - * Formats a timestamp as ISO 8601 date - * - * @author <ungu at terong dot com> - * @link http://php.net/manual/en/function.date.php#54072 - * - * @param int $int_date current date in UNIX timestamp - * @return string - */ -function date_iso8601($int_date) { - $date_mod = date('Y-m-d\TH:i:s', $int_date); - $pre_timezone = date('O', $int_date); - $time_zone = substr($pre_timezone, 0, 3).":".substr($pre_timezone, 3, 2); - $date_mod .= $time_zone; - return $date_mod; -} - -/** - * return an obfuscated email address in line with $conf['mailguard'] setting - * - * @author Harry Fuecks <hfuecks@gmail.com> - * @author Christopher Smith <chris@jalakai.co.uk> - * - * @param string $email email address - * @return string - */ -function obfuscate($email) { - global $conf; - - switch($conf['mailguard']) { - case 'visible' : - $obfuscate = array('@' => ' [at] ', '.' => ' [dot] ', '-' => ' [dash] '); - return strtr($email, $obfuscate); - - case 'hex' : - $encode = ''; - $len = strlen($email); - for($x = 0; $x < $len; $x++) { - $encode .= '&#x'.bin2hex($email{$x}).';'; - } - return $encode; - - case 'none' : - default : - return $email; - } -} - -/** - * Removes quoting backslashes - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $string - * @param string $char backslashed character - * @return string - */ -function unslash($string, $char = "'") { - return str_replace('\\'.$char, $char, $string); -} - -/** - * Convert php.ini shorthands to byte - * - * @author <gilthans dot NO dot SPAM at gmail dot com> - * @link http://php.net/manual/en/ini.core.php#79564 - * - * @param string $v shorthands - * @return int|string - */ -function php_to_byte($v) { - $l = substr($v, -1); - $ret = substr($v, 0, -1); - switch(strtoupper($l)) { - /** @noinspection PhpMissingBreakStatementInspection */ - case 'P': - $ret *= 1024; - /** @noinspection PhpMissingBreakStatementInspection */ - case 'T': - $ret *= 1024; - /** @noinspection PhpMissingBreakStatementInspection */ - case 'G': - $ret *= 1024; - /** @noinspection PhpMissingBreakStatementInspection */ - case 'M': - $ret *= 1024; - /** @noinspection PhpMissingBreakStatementInspection */ - case 'K': - $ret *= 1024; - break; - default; - $ret *= 10; - break; - } - return $ret; -} - -/** - * Wrapper around preg_quote adding the default delimiter - * - * @param string $string - * @return string - */ -function preg_quote_cb($string) { - return preg_quote($string, '/'); -} - -/** - * Shorten a given string by removing data from the middle - * - * You can give the string in two parts, the first part $keep - * will never be shortened. The second part $short will be cut - * in the middle to shorten but only if at least $min chars are - * left to display it. Otherwise it will be left off. - * - * @param string $keep the part to keep - * @param string $short the part to shorten - * @param int $max maximum chars you want for the whole string - * @param int $min minimum number of chars to have left for middle shortening - * @param string $char the shortening character to use - * @return string - */ -function shorten($keep, $short, $max, $min = 9, $char = '…') { - $max = $max - utf8_strlen($keep); - if($max < $min) return $keep; - $len = utf8_strlen($short); - if($len <= $max) return $keep.$short; - $half = floor($max / 2); - return $keep.utf8_substr($short, 0, $half - 1).$char.utf8_substr($short, $len - $half); -} - -/** - * Return the users real name or e-mail address for use - * in page footer and recent changes pages - * - * @param string|null $username or null when currently logged-in user should be used - * @param bool $textonly true returns only plain text, true allows returning html - * @return string html or plain text(not escaped) of formatted user name - * - * @author Andy Webber <dokuwiki AT andywebber DOT com> - */ -function editorinfo($username, $textonly = false) { - return userlink($username, $textonly); -} - -/** - * Returns users realname w/o link - * - * @param string|null $username or null when currently logged-in user should be used - * @param bool $textonly true returns only plain text, true allows returning html - * @return string html or plain text(not escaped) of formatted user name - * - * @triggers COMMON_USER_LINK - */ -function userlink($username = null, $textonly = false) { - global $conf, $INFO; - /** @var DokuWiki_Auth_Plugin $auth */ - global $auth; - /** @var Input $INPUT */ - global $INPUT; - - // prepare initial event data - $data = array( - 'username' => $username, // the unique user name - 'name' => '', - 'link' => array( //setting 'link' to false disables linking - 'target' => '', - 'pre' => '', - 'suf' => '', - 'style' => '', - 'more' => '', - 'url' => '', - 'title' => '', - 'class' => '' - ), - 'userlink' => '', // formatted user name as will be returned - 'textonly' => $textonly - ); - if($username === null) { - $data['username'] = $username = $INPUT->server->str('REMOTE_USER'); - if($textonly){ - $data['name'] = $INFO['userinfo']['name']. ' (' . $INPUT->server->str('REMOTE_USER') . ')'; - }else { - $data['name'] = '<bdi>' . hsc($INFO['userinfo']['name']) . '</bdi> (<bdi>' . hsc($INPUT->server->str('REMOTE_USER')) . '</bdi>)'; - } - } - - $evt = new Doku_Event('COMMON_USER_LINK', $data); - if($evt->advise_before(true)) { - if(empty($data['name'])) { - if($auth) $info = $auth->getUserData($username); - if($conf['showuseras'] != 'loginname' && isset($info) && $info) { - switch($conf['showuseras']) { - case 'username': - case 'username_link': - $data['name'] = $textonly ? $info['name'] : hsc($info['name']); - break; - case 'email': - case 'email_link': - $data['name'] = obfuscate($info['mail']); - break; - } - } else { - $data['name'] = $textonly ? $data['username'] : hsc($data['username']); - } - } - - /** @var Doku_Renderer_xhtml $xhtml_renderer */ - static $xhtml_renderer = null; - - if(!$data['textonly'] && empty($data['link']['url'])) { - - if(in_array($conf['showuseras'], array('email_link', 'username_link'))) { - if(!isset($info)) { - if($auth) $info = $auth->getUserData($username); - } - if(isset($info) && $info) { - if($conf['showuseras'] == 'email_link') { - $data['link']['url'] = 'mailto:' . obfuscate($info['mail']); - } else { - if(is_null($xhtml_renderer)) { - $xhtml_renderer = p_get_renderer('xhtml'); - } - if(empty($xhtml_renderer->interwiki)) { - $xhtml_renderer->interwiki = getInterwiki(); - } - $shortcut = 'user'; - $exists = null; - $data['link']['url'] = $xhtml_renderer->_resolveInterWiki($shortcut, $username, $exists); - $data['link']['class'] .= ' interwiki iw_user'; - if($exists !== null) { - if($exists) { - $data['link']['class'] .= ' wikilink1'; - } else { - $data['link']['class'] .= ' wikilink2'; - $data['link']['rel'] = 'nofollow'; - } - } - } - } else { - $data['textonly'] = true; - } - - } else { - $data['textonly'] = true; - } - } - - if($data['textonly']) { - $data['userlink'] = $data['name']; - } else { - $data['link']['name'] = $data['name']; - if(is_null($xhtml_renderer)) { - $xhtml_renderer = p_get_renderer('xhtml'); - } - $data['userlink'] = $xhtml_renderer->_formatLink($data['link']); - } - } - $evt->advise_after(); - unset($evt); - - return $data['userlink']; -} - -/** - * Returns the path to a image file for the currently chosen license. - * When no image exists, returns an empty string - * - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $type - type of image 'badge' or 'button' - * @return string - */ -function license_img($type) { - global $license; - global $conf; - if(!$conf['license']) return ''; - if(!is_array($license[$conf['license']])) return ''; - $try = array(); - $try[] = 'lib/images/license/'.$type.'/'.$conf['license'].'.png'; - $try[] = 'lib/images/license/'.$type.'/'.$conf['license'].'.gif'; - if(substr($conf['license'], 0, 3) == 'cc-') { - $try[] = 'lib/images/license/'.$type.'/cc.png'; - } - foreach($try as $src) { - if(file_exists(DOKU_INC.$src)) return $src; - } - return ''; -} - -/** - * Checks if the given amount of memory is available - * - * If the memory_get_usage() function is not available the - * function just assumes $bytes of already allocated memory - * - * @author Filip Oscadal <webmaster@illusionsoftworks.cz> - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param int $mem Size of memory you want to allocate in bytes - * @param int $bytes already allocated memory (see above) - * @return bool - */ -function is_mem_available($mem, $bytes = 1048576) { - $limit = trim(ini_get('memory_limit')); - if(empty($limit)) return true; // no limit set! - - // parse limit to bytes - $limit = php_to_byte($limit); - - // get used memory if possible - if(function_exists('memory_get_usage')) { - $used = memory_get_usage(); - } else { - $used = $bytes; - } - - if($used + $mem > $limit) { - return false; - } - - return true; -} - -/** - * Send a HTTP redirect to the browser - * - * Works arround Microsoft IIS cookie sending bug. Exits the script. - * - * @link http://support.microsoft.com/kb/q176113/ - * @author Andreas Gohr <andi@splitbrain.org> - * - * @param string $url url being directed to - */ -function send_redirect($url) { - $url = stripctl($url); // defend against HTTP Response Splitting - - /* @var Input $INPUT */ - global $INPUT; - - //are there any undisplayed messages? keep them in session for display - global $MSG; - if(isset($MSG) && count($MSG) && !defined('NOSESSION')) { - //reopen session, store data and close session again - @session_start(); - $_SESSION[DOKU_COOKIE]['msg'] = $MSG; - } - - // always close the session - session_write_close(); - - // check if running on IIS < 6 with CGI-PHP - if($INPUT->server->has('SERVER_SOFTWARE') && $INPUT->server->has('GATEWAY_INTERFACE') && - (strpos($INPUT->server->str('GATEWAY_INTERFACE'), 'CGI') !== false) && - (preg_match('|^Microsoft-IIS/(\d)\.\d$|', trim($INPUT->server->str('SERVER_SOFTWARE')), $matches)) && - $matches[1] < 6 - ) { - header('Refresh: 0;url='.$url); - } else { - header('Location: '.$url); - } - - // no exits during unit tests - if(defined('DOKU_UNITTEST')) { - // pass info about the redirect back to the test suite - $testRequest = TestRequest::getRunning(); - if($testRequest !== null) { - $testRequest->addData('send_redirect', $url); - } - return; - } - - exit; -} - -/** - * Validate a value using a set of valid values - * - * This function checks whether a specified value is set and in the array - * $valid_values. If not, the function returns a default value or, if no - * default is specified, throws an exception. - * - * @param string $param The name of the parameter - * @param array $valid_values A set of valid values; Optionally a default may - * be marked by the key “default”. - * @param array $array The array containing the value (typically $_POST - * or $_GET) - * @param string $exc The text of the raised exception - * - * @throws Exception - * @return mixed - * @author Adrian Lang <lang@cosmocode.de> - */ -function valid_input_set($param, $valid_values, $array, $exc = '') { - if(isset($array[$param]) && in_array($array[$param], $valid_values)) { - return $array[$param]; - } elseif(isset($valid_values['default'])) { - return $valid_values['default']; - } else { - throw new Exception($exc); - } -} - -/** - * Read a preference from the DokuWiki cookie - * (remembering both keys & values are urlencoded) - * - * @param string $pref preference key - * @param mixed $default value returned when preference not found - * @return string preference value - */ -function get_doku_pref($pref, $default) { - $enc_pref = urlencode($pref); - if(isset($_COOKIE['DOKU_PREFS']) && strpos($_COOKIE['DOKU_PREFS'], $enc_pref) !== false) { - $parts = explode('#', $_COOKIE['DOKU_PREFS']); - $cnt = count($parts); - for($i = 0; $i < $cnt; $i += 2) { - if($parts[$i] == $enc_pref) { - return urldecode($parts[$i + 1]); - } - } - } - return $default; -} - -/** - * Add a preference to the DokuWiki cookie - * (remembering $_COOKIE['DOKU_PREFS'] is urlencoded) - * Remove it by setting $val to false - * - * @param string $pref preference key - * @param string $val preference value - */ -function set_doku_pref($pref, $val) { - global $conf; - $orig = get_doku_pref($pref, false); - $cookieVal = ''; - - if($orig && ($orig != $val)) { - $parts = explode('#', $_COOKIE['DOKU_PREFS']); - $cnt = count($parts); - // urlencode $pref for the comparison - $enc_pref = rawurlencode($pref); - for($i = 0; $i < $cnt; $i += 2) { - if($parts[$i] == $enc_pref) { - if ($val !== false) { - $parts[$i + 1] = rawurlencode($val); - } else { - unset($parts[$i]); - unset($parts[$i + 1]); - } - break; - } - } - $cookieVal = implode('#', $parts); - } else if (!$orig && $val !== false) { - $cookieVal = ($_COOKIE['DOKU_PREFS'] ? $_COOKIE['DOKU_PREFS'].'#' : '').rawurlencode($pref).'#'.rawurlencode($val); - } - - if (!empty($cookieVal)) { - $cookieDir = empty($conf['cookiedir']) ? DOKU_REL : $conf['cookiedir']; - setcookie('DOKU_PREFS', $cookieVal, time()+365*24*3600, $cookieDir, '', ($conf['securecookie'] && is_ssl())); - } -} - -/** - * Strips source mapping declarations from given text #601 - * - * @param string &$text reference to the CSS or JavaScript code to clean - */ -function stripsourcemaps(&$text){ - $text = preg_replace('/^(\/\/|\/\*)[@#]\s+sourceMappingURL=.*?(\*\/)?$/im', '\\1\\2', $text); -} - -/** - * Returns the contents of a given SVG file for embedding - * - * Inlining SVGs saves on HTTP requests and more importantly allows for styling them through - * CSS. However it should used with small SVGs only. The $maxsize setting ensures only small - * files are embedded. - * - * This strips unneeded headers, comments and newline. The result is not a vaild standalone SVG! - * - * @param string $file full path to the SVG file - * @param int $maxsize maximum allowed size for the SVG to be embedded - * @return string|false the SVG content, false if the file couldn't be loaded - */ -function inlineSVG($file, $maxsize = 2048) { - $file = trim($file); - if($file === '') return false; - if(!file_exists($file)) return false; - if(filesize($file) > $maxsize) return false; - if(!is_readable($file)) return false; - $content = file_get_contents($file); - $content = preg_replace('/<!--.*?(-->)/s','', $content); // comments - $content = preg_replace('/<\?xml .*?\?>/i', '', $content); // xml header - $content = preg_replace('/<!DOCTYPE .*?>/i', '', $content); // doc type - $content = preg_replace('/>\s+</s', '><', $content); // newlines between tags - $content = trim($content); - if(substr($content, 0, 5) !== '<svg ') return false; - return $content; -} - -//Setup VIM: ex: et ts=2 : |