12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997 |
- <?php
- /**
- * Common DokuWiki functions
- *
- * @license GPL 2 (http://www.gnu.org/licenses/gpl.html)
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- use dokuwiki\Cache\CacheInstructions;
- use dokuwiki\Cache\CacheRenderer;
- use dokuwiki\ChangeLog\PageChangeLog;
- use dokuwiki\File\PageFile;
- use dokuwiki\Logger;
- use dokuwiki\Subscriptions\PageSubscriptionSender;
- use dokuwiki\Subscriptions\SubscriberManager;
- use dokuwiki\Extension\AuthPlugin;
- use dokuwiki\Extension\Event;
- /**
- * 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 | ENT_SUBSTITUTE | ENT_HTML401, 'UTF-8');
- }
- /**
- * A safer explode for fixed length lists
- *
- * This works just like explode(), but will always return the wanted number of elements.
- * If the $input string does not contain enough elements, the missing elements will be
- * filled up with the $default value. If the input string contains more elements, the last
- * one will NOT be split up and will still contain $separator
- *
- * @param string $separator The boundary string
- * @param string $string The input string
- * @param int $limit The number of expected elements
- * @param mixed $default The value to use when filling up missing elements
- * @see explode
- * @return array
- */
- function sexplode($separator, $string, $limit, $default = null)
- {
- return array_pad(explode($separator, $string, $limit), $limit, $default);
- }
- /**
- * 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 \dokuwiki\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(empty($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;
- $subManager = new SubscriberManager();
- $info['subscribed'] = $subManager->userSubscription();
- $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));
- }
- if ($revinfo !== false) {
- $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'].
- $info['editor'] = $revinfo['user'] ?: $revinfo['ip'];
- } else {
- $info['ip'] = null;
- $info['user'] = null;
- $info['sum'] = null;
- $info['editor'] = null;
- }
- // draft
- $draft = new \dokuwiki\Draft($ID, $info['client']);
- if ($draft->isDraftAvailable()) {
- $info['draft'] = $draft->getDraftFilename();
- }
- 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'] = isset($INFO) ? (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 $skipEmptyStrings skip empty string values?
- * @return string
- */
- function buildAttributes($params, $skipEmptyStrings = false) {
- $url = '';
- $white = false;
- foreach($params as $key => $val) {
- if($key[0] == '_') continue;
- if($val === '' && $skipEmptyStrings) continue;
- if($white) $url .= ' ';
- $url .= $key.'="';
- $url .= hsc($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;
- global $INFO;
- //first visit?
- $crumbs = isset($_SESSION[DOKU_COOKIE]['bc']) ? $_SESSION[DOKU_COOKIE]['bc'] : array();
- //we only save on show and existing visible readable wiki documents
- $file = wikiFN($ID);
- if($ACT != 'show' || $INFO['perm'] < AUTH_READ || 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/dokuwiki/dokuwiki/pull/84 and
- * https://github.com/dokuwiki/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;
- $id = (string) $id;
- 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
- $w = isset($more['w']) ? $more['w'] : null;
- $h = isset($more['h']) ? $more['h'] : null;
- if($w || $h || $isexternalimage){
- $more['tok'] = media_get_token($id, $w, $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
- // phpcs:disable Generic.Files.LineLength.TooLong
- $text = preg_replace(
- '!(\b)(www\.[\w.:?\-;,]+?\.[\w.:?\-;,]+?[\w/\#~:.?+=&%@\!\-.:?\-;,]+?)([.:?\-;,]*[^\w/\#~:.?+=&%@\!\-.:?\-;,])!i',
- '\1http://\2 \2\3',
- $text
- );
- // phpcs:enable
- $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 Event::createAndTrigger('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, $conf;
- $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'))));
- }
- // remove any non-IP stuff
- $cnt = count($ip);
- for($i = 0; $i < $cnt; $i++) {
- if(filter_var($ip[$i], FILTER_VALIDATE_IP) === false) {
- unset($ip[$i]);
- }
- }
- $ip = array_values(array_unique($ip));
- if(empty($ip) || !$ip[0]) $ip[0] = '0.0.0.0'; // for some strange reason we don't have a IP
- if(!$single) return join(',', $ip);
- // skip trusted local addresses
- foreach($ip as $i) {
- if(!empty($conf['trustedproxy']) && preg_match('/'.$conf['trustedproxy'].'/', $i)) {
- continue;
- } else {
- return $i;
- }
- }
- // still here? just use the last address
- // this case all ips in the list are trusted
- return $ip[count($ip)-1];
- }
- /**
- * 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
- *
- * @deprecated 2018-04-27 you probably want media queries instead anyway
- * @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 = join(
- '|',
- [
- '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') || (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') || $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(!\dokuwiki\Utf8\Clean::isUtf8($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 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@',
- '@!CURNS@',
- '@!!CURNS@',
- '@!CURNS!@',
- '@FILE@',
- '@!FILE@',
- '@!FILE!@',
- '@PAGE@',
- '@!PAGE@',
- '@!!PAGE@',
- '@!PAGE!@',
- '@USER@',
- '@NAME@',
- '@MAIL@',
- '@DATE@',
- ),
- array(
- $id,
- getNS($id),
- curNS($id),
- \dokuwiki\Utf8\PhpString::ucfirst(curNS($id)),
- \dokuwiki\Utf8\PhpString::ucwords(curNS($id)),
- \dokuwiki\Utf8\PhpString::strtoupper(curNS($id)),
- $file,
- \dokuwiki\Utf8\PhpString::ucfirst($file),
- \dokuwiki\Utf8\PhpString::strtoupper($file),
- $page,
- \dokuwiki\Utf8\PhpString::ucfirst($page),
- \dokuwiki\Utf8\PhpString::ucwords($page),
- \dokuwiki\Utf8\PhpString::strtoupper($page),
- $INPUT->server->str('REMOTE_USER'),
- $USERINFO ? $USERINFO['name'] : '',
- $USERINFO ? $USERINFO['mail'] : '',
- $conf['dformat'],
- ), $tpl
- );
- // we need the callback to work around strftime's char limit
- $tpl = preg_replace_callback(
- '/%./',
- function ($m) {
- return dformat(null, $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) = sexplode('-', $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
- * @deprecated 2021-11-28
- */
- function detectExternalEdit($id) {
- dbg_deprecated(PageFile::class .'::detectExternalEdit()');
- (new PageFile($id))->detectExternalEdit();
- }
- /**
- * 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) {
- // get COMMON_WIKIPAGE_SAVE event data
- $data = (new PageFile($id))->saveWikiText($text, $summary, $minor);
- if(!$data) return; // save was cancelled (for no changes or by a plugin)
- // send notify mails
- list('oldRevision' => $rev, 'newRevision' => $new_rev, 'summary' => $summary) = $data;
- notify($id, 'admin', $rev, $summary, $minor, $new_rev);
- notify($id, 'subscribers', $rev, $summary, $minor, $new_rev);
- // 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 CacheRenderer($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
- * @deprecated 2021-11-28
- */
- function saveOldRevision($id) {
- dbg_deprecated(PageFile::class .'::saveOldRevision()');
- return (new PageFile($id))->saveOldRevision();
- }
- /**
- * 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
- * @param int|string $current_rev New page revision
- * @return bool
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- function notify($id, $who, $rev = '', $summary = '', $minor = false, $replace = array(), $current_rev = false) {
- 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);
- Event::createAndTrigger(
- 'COMMON_NOTIFY_ADDRESSLIST', $data,
- array(new SubscriberManager(), 'notifyAddresses')
- );
- $to = $data['addresslist'];
- if (empty($to)) return false;
- $tpl = 'subscr_single';
- } else {
- return false; //just to be safe
- }
- // prepare content
- $subscription = new PageSubscriptionSender();
- return $subscription->sendPageDiff($to, $tpl, $id, $rev, $summary, $current_rev);
- }
- /**
- * 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(!array_key_exists('host', $url)) return '';
- if(!preg_match('/(google|bing|yahoo|ask|duckduckgo|babylon|aol|yandex)/',$url['host'])) return '';
- $query = array();
- if(!array_key_exists('query', $url)) return '';
- parse_str($url['query'], $query);
- $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 '';
- // ignore if query includes a full URL
- if(strpos($q, '//') !== false) 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' :
- return \dokuwiki\Utf8\Conversion::toHtml($email, true);
- 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
- *
- * On 32 bit systems values >= 2GB will fail!
- *
- * -1 (infinite size) will be reported as -1
- *
- * @link https://www.php.net/manual/en/faq.using.php#faq.using.shorthandbytes
- * @param string $value PHP size shorthand
- * @return int
- */
- function php_to_byte($value) {
- switch (strtoupper(substr($value,-1))) {
- case 'G':
- $ret = intval(substr($value, 0, -1)) * 1024 * 1024 * 1024;
- break;
- case 'M':
- $ret = intval(substr($value, 0, -1)) * 1024 * 1024;
- break;
- case 'K':
- $ret = intval(substr($value, 0, -1)) * 1024;
- break;
- default:
- $ret = intval($value);
- 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 - \dokuwiki\Utf8\PhpString::strlen($keep);
- if($max < $min) return $keep;
- $len = \dokuwiki\Utf8\PhpString::strlen($short);
- if($len <= $max) return $keep.$short;
- $half = floor($max / 2);
- return $keep .
- \dokuwiki\Utf8\PhpString::substr($short, 0, $half - 1) .
- $char .
- \dokuwiki\Utf8\PhpString::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 AuthPlugin $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 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!
- if($limit == -1) return true; // unlimited
- // 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);
- // due to #2721 there might be duplicate entries,
- // so we read from the end
- for($i = $cnt-2; $i >= 0; $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 !== false && ($orig !== $val)) {
- $parts = explode('#', $_COOKIE['DOKU_PREFS']);
- $cnt = count($parts);
- // urlencode $pref for the comparison
- $enc_pref = rawurlencode($pref);
- $seen = false;
- for ($i = 0; $i < $cnt; $i += 2) {
- if ($parts[$i] == $enc_pref) {
- if (!$seen){
- if ($val !== false) {
- $parts[$i + 1] = rawurlencode($val ?? '');
- } else {
- unset($parts[$i]);
- unset($parts[$i + 1]);
- }
- $seen = true;
- } else {
- // no break because we want to remove duplicate entries
- unset($parts[$i]);
- unset($parts[$i + 1]);
- }
- }
- }
- $cookieVal = implode('#', $parts);
- } else if ($orig === false && $val !== false) {
- $cookieVal = (isset($_COOKIE['DOKU_PREFS']) ? $_COOKIE['DOKU_PREFS'] . '#' : '') .
- rawurlencode($pref) . '#' . rawurlencode($val);
- }
- $cookieDir = empty($conf['cookiedir']) ? DOKU_REL : $conf['cookiedir'];
- if(defined('DOKU_UNITTEST')) {
- $_COOKIE['DOKU_PREFS'] = $cookieVal;
- }else{
- 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 :
|