1*ed7b5f09Sandi<?php 215fae107Sandi/** 315fae107Sandi * Common DokuWiki functions 415fae107Sandi * 515fae107Sandi * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 715fae107Sandi */ 815fae107Sandi 9*ed7b5f09Sandi if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/'); 10*ed7b5f09Sandi require_once(DOKU_INC.'conf/dokuwiki.php'); 11*ed7b5f09Sandi require_once(DOKU_INC.'inc/io.php'); 12*ed7b5f09Sandi require_once(DOKU_INC.'inc/utf8.php'); 13*ed7b5f09Sandi require_once(DOKU_INC.'inc/mail.php'); 14f3f0262cSandi 15f3f0262cSandi/** 1615fae107Sandi * Return info about the current document as associative 17f3f0262cSandi * array. 1815fae107Sandi * 1915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 20f3f0262cSandi */ 21f3f0262cSandifunction pageinfo(){ 22f3f0262cSandi global $ID; 23f3f0262cSandi global $REV; 24f3f0262cSandi global $USERINFO; 25f3f0262cSandi global $conf; 26f3f0262cSandi 27f3f0262cSandi if($_SERVER['REMOTE_USER']){ 28f3f0262cSandi $info['user'] = $_SERVER['REMOTE_USER']; 29f3f0262cSandi $info['userinfo'] = $USERINFO; 30f3f0262cSandi $info['perm'] = auth_quickaclcheck($ID); 31f3f0262cSandi }else{ 32f3f0262cSandi $info['user'] = ''; 33f3f0262cSandi $info['perm'] = auth_aclcheck($ID,'',null); 34f3f0262cSandi } 35f3f0262cSandi 36f3f0262cSandi $info['namespace'] = getNS($ID); 37f3f0262cSandi $info['locked'] = checklock($ID); 38f3f0262cSandi $info['filepath'] = realpath(wikiFN($ID,$REV)); 39f3f0262cSandi $info['exists'] = @file_exists($info['filepath']); 40f3f0262cSandi if($REV && !$info['exists']){ 41f3f0262cSandi //check if current revision was meant 42f3f0262cSandi $cur = wikiFN($ID); 43f3f0262cSandi if(@file_exists($cur) && (@filemtime($cur) == $REV)){ 44f3f0262cSandi $info['filepath'] = realpath($cur); 45f3f0262cSandi $info['exists'] = true; 46f3f0262cSandi $REV = ''; 47f3f0262cSandi } 48f3f0262cSandi } 49f3f0262cSandi if($info['exists']){ 50f3f0262cSandi $info['writable'] = (is_writable($info['filepath']) && 51f3f0262cSandi ($info['perm'] >= AUTH_EDIT)); 52f3f0262cSandi }else{ 53f3f0262cSandi $info['writable'] = ($info['perm'] >= AUTH_CREATE); 54f3f0262cSandi } 55f3f0262cSandi $info['editable'] = ($info['writable'] && empty($info['lock'])); 56f3f0262cSandi $info['lastmod'] = @filemtime($info['filepath']); 57f3f0262cSandi 58652610a2Sandi //who's the editor 59652610a2Sandi if($REV){ 60652610a2Sandi $revinfo = getRevisionInfo($ID,$REV); 61652610a2Sandi }else{ 62652610a2Sandi $revinfo = getRevisionInfo($ID,$info['lastmod']); 63652610a2Sandi } 64652610a2Sandi $info['ip'] = $revinfo['ip']; 65652610a2Sandi $info['user'] = $revinfo['user']; 66652610a2Sandi $info['sum'] = $revinfo['sum']; 67652610a2Sandi $info['editor'] = $revinfo['ip']; 68652610a2Sandi if($revinfo['user']) $info['editor'].= ' ('.$revinfo['user'].')'; 69652610a2Sandi 70f3f0262cSandi return $info; 71f3f0262cSandi} 72f3f0262cSandi 73f3f0262cSandi/** 740396becbSandi * print a message 750396becbSandi * 760396becbSandi * If HTTP headers were not sent yet the message is added 770396becbSandi * to the global message array else it's printed directly 780396becbSandi * using html_msgarea() 790396becbSandi * 80f3f0262cSandi * 81f3f0262cSandi * Levels can be: 82f3f0262cSandi * 83f3f0262cSandi * -1 error 84f3f0262cSandi * 0 info 85f3f0262cSandi * 1 success 8615fae107Sandi * 8715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 880396becbSandi * @see html_msgarea 89f3f0262cSandi */ 90f3f0262cSandifunction msg($message,$lvl=0){ 91f3f0262cSandi global $MSG; 92f3f0262cSandi $errors[-1] = 'error'; 93f3f0262cSandi $errors[0] = 'info'; 94f3f0262cSandi $errors[1] = 'success'; 95f3f0262cSandi 96cc20ad51Sandi if(!headers_sent()){ 97f3f0262cSandi if(!isset($MSG)) $MSG = array(); 98f3f0262cSandi $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message); 990396becbSandi }else{ 1000396becbSandi $MSG = array(); 1010396becbSandi $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message); 1020396becbSandi html_msgarea(); 1030396becbSandi } 104f3f0262cSandi} 105f3f0262cSandi 106f3f0262cSandi/** 10715fae107Sandi * This builds the breadcrumb trail and returns it as array 10815fae107Sandi * 10915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 110f3f0262cSandi */ 111f3f0262cSandifunction breadcrumbs(){ 112f3f0262cSandi global $ID; 113f3f0262cSandi global $ACT; 114f3f0262cSandi global $conf; 115f3f0262cSandi $crumbs = $_SESSION[$conf['title']]['bc']; 116f3f0262cSandi 117f3f0262cSandi //first visit? 118f3f0262cSandi if (!is_array($crumbs)){ 119f3f0262cSandi $crumbs = array(); 120f3f0262cSandi } 121f3f0262cSandi //we only save on show and existing wiki documents 122f3f0262cSandi if($ACT != 'show' || !@file_exists(wikiFN($ID))){ 123f3f0262cSandi $_SESSION[$conf['title']]['bc'] = $crumbs; 124f3f0262cSandi return $crumbs; 125f3f0262cSandi } 126f3f0262cSandi //remove ID from array 127f3f0262cSandi $pos = array_search($ID,$crumbs); 128f3f0262cSandi if($pos !== false && $pos !== null){ 129f3f0262cSandi array_splice($crumbs,$pos,1); 130f3f0262cSandi } 131f3f0262cSandi 132f3f0262cSandi //add to array 133f3f0262cSandi $crumbs[] =$ID; 134f3f0262cSandi //reduce size 135f3f0262cSandi while(count($crumbs) > $conf['breadcrumbs']){ 136f3f0262cSandi array_shift($crumbs); 137f3f0262cSandi } 138f3f0262cSandi //save to session 139f3f0262cSandi $_SESSION[$conf['title']]['bc'] = $crumbs; 140f3f0262cSandi return $crumbs; 141f3f0262cSandi} 142f3f0262cSandi 143f3f0262cSandi/** 14415fae107Sandi * Filter for page IDs 14515fae107Sandi * 146f3f0262cSandi * This is run on a ID before it is outputted somewhere 147f3f0262cSandi * currently used to replace the colon with something else 148f3f0262cSandi * on Windows systems and to have proper URL encoding 14915fae107Sandi * 15049c713a3Sandi * Urlencoding is ommitted when the second parameter is false 15149c713a3Sandi * 15215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 153f3f0262cSandi */ 15449c713a3Sandifunction idfilter($id,$ue=true){ 155f3f0262cSandi global $conf; 156f3f0262cSandi if ($conf['useslash'] && $conf['userewrite']){ 157f3f0262cSandi $id = strtr($id,':','/'); 158f3f0262cSandi }elseif (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' && 159f3f0262cSandi $conf['userewrite']) { 160f3f0262cSandi $id = strtr($id,':',';'); 161f3f0262cSandi } 16249c713a3Sandi if($ue){ 163f3f0262cSandi $id = urlencode($id); 164f3f0262cSandi $id = str_replace('%3A',':',$id); //keep as colon 165f3f0262cSandi $id = str_replace('%2F','/',$id); //keep as slash 16649c713a3Sandi } 167f3f0262cSandi return $id; 168f3f0262cSandi} 169f3f0262cSandi 170f3f0262cSandi/** 171*ed7b5f09Sandi * This builds a link to a wikipage 17215fae107Sandi * 17315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 174f3f0262cSandi */ 175*ed7b5f09Sandifunction wl($id='',$more='',$abs=false){ 176f3f0262cSandi global $conf; 177f3f0262cSandi $more = str_replace(',','&',$more); 178f3f0262cSandi 179f3f0262cSandi $id = idfilter($id); 180*ed7b5f09Sandi if($abs){ 181*ed7b5f09Sandi $xlink = DOKU_URL; 182*ed7b5f09Sandi }else{ 183*ed7b5f09Sandi $xlink = DOKU_BASE; 184*ed7b5f09Sandi } 185f3f0262cSandi 186f3f0262cSandi if(!$conf['userewrite']){ 187*ed7b5f09Sandi $xlink .= DOKU_SCRIPT.'?id='.$id; 188f3f0262cSandi if($more) $xlink .= '&'.$more; 189f3f0262cSandi }else{ 190f3f0262cSandi $xlink .= $id; 191f3f0262cSandi if($more) $xlink .= '?'.$more; 192f3f0262cSandi } 193f3f0262cSandi 194f3f0262cSandi return $xlink; 195f3f0262cSandi} 196f3f0262cSandi 197f3f0262cSandi/** 198f3f0262cSandi * Just builds a link to a script 19915fae107Sandi * 200*ed7b5f09Sandi * @todo maybe obsolete 20115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 202f3f0262cSandi */ 203f3f0262cSandifunction script($script='doku.php'){ 204*ed7b5f09Sandi# $link = getBaseURL(); 205*ed7b5f09Sandi# $link .= $script; 206*ed7b5f09Sandi# return $link; 207*ed7b5f09Sandi return DOKU_BASE.DOKU_SCRIPT; 208f3f0262cSandi} 209f3f0262cSandi 210f3f0262cSandi/** 211f3f0262cSandi * Return namespacepart of a wiki ID 21215fae107Sandi * 21315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 214f3f0262cSandi */ 215f3f0262cSandifunction getNS($id){ 216f3f0262cSandi if(strpos($id,':')!==false){ 217f3f0262cSandi return substr($id,0,strrpos($id,':')); 218f3f0262cSandi } 219f3f0262cSandi return false; 220f3f0262cSandi} 221f3f0262cSandi 222f3f0262cSandi/** 22315fae107Sandi * Returns the ID without the namespace 22415fae107Sandi * 22515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 226f3f0262cSandi */ 227f3f0262cSandifunction noNS($id){ 228f3f0262cSandi return preg_replace('/.*:/','',$id); 229f3f0262cSandi} 230f3f0262cSandi 231f3f0262cSandi/** 23215fae107Sandi * Spamcheck against wordlist 23315fae107Sandi * 234f3f0262cSandi * Checks the wikitext against a list of blocked expressions 235f3f0262cSandi * returns true if the text contains any bad words 23615fae107Sandi * 23715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 238f3f0262cSandi */ 239f3f0262cSandifunction checkwordblock(){ 240f3f0262cSandi global $TEXT; 241f3f0262cSandi global $conf; 242f3f0262cSandi 243f3f0262cSandi if(!$conf['usewordblock']) return false; 244f3f0262cSandi 245703f6fdeSandi $blockfile = file('conf/wordblock.conf'); 2463e2965d7Sandi //how many lines to read at once (to work around some PCRE limits) 2473e2965d7Sandi if(version_compare(phpversion(),'4.3.0','<')){ 2483e2965d7Sandi //old versions of PCRE define a maximum of parenthesises even if no 2493e2965d7Sandi //backreferences are used - the maximum is 99 2503e2965d7Sandi //this is very bad performancewise and may even be too high still 2513e2965d7Sandi $chunksize = 40; 2523e2965d7Sandi }else{ 253703f6fdeSandi //read file in chunks of 600 - this should work around the 2543e2965d7Sandi //MAX_PATTERN_SIZE in modern PCRE 2553e2965d7Sandi $chunksize = 600; 2563e2965d7Sandi } 2573e2965d7Sandi while($blocks = array_splice($blockfile,0,$chunksize)){ 258f3f0262cSandi $re = array(); 259f3f0262cSandi #build regexp from blocks 260f3f0262cSandi foreach($blocks as $block){ 261f3f0262cSandi $block = preg_replace('/#.*$/','',$block); 262f3f0262cSandi $block = trim($block); 263f3f0262cSandi if(empty($block)) continue; 264f3f0262cSandi $re[] = $block; 265f3f0262cSandi } 266f3f0262cSandi if(preg_match('#('.join('|',$re).')#si',$TEXT)) return true; 267703f6fdeSandi } 268f3f0262cSandi return false; 269f3f0262cSandi} 270f3f0262cSandi 271f3f0262cSandi/** 27215fae107Sandi * Return the IP of the client 27315fae107Sandi * 27415fae107Sandi * Honours X-Forwarded-For Proxy Headers 27515fae107Sandi * 27615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 277f3f0262cSandi */ 278f3f0262cSandifunction clientIP(){ 279f3f0262cSandi $my = $_SERVER['REMOTE_ADDR']; 280f3f0262cSandi if($_SERVER['HTTP_X_FORWARDED_FOR']){ 281f3f0262cSandi $my .= ' ('.$_SERVER['HTTP_X_FORWARDED_FOR'].')'; 282f3f0262cSandi } 283f3f0262cSandi return $my; 284f3f0262cSandi} 285f3f0262cSandi 286f3f0262cSandi/** 28715fae107Sandi * Checks if a given page is currently locked. 28815fae107Sandi * 289f3f0262cSandi * removes stale lockfiles 29015fae107Sandi * 29115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 292f3f0262cSandi */ 293f3f0262cSandifunction checklock($id){ 294f3f0262cSandi global $conf; 295f3f0262cSandi $lock = wikiFN($id).'.lock'; 296f3f0262cSandi 297f3f0262cSandi //no lockfile 298f3f0262cSandi if(!@file_exists($lock)) return false; 299f3f0262cSandi 300f3f0262cSandi //lockfile expired 301f3f0262cSandi if((time() - filemtime($lock)) > $conf['locktime']){ 302f3f0262cSandi unlink($lock); 303f3f0262cSandi return false; 304f3f0262cSandi } 305f3f0262cSandi 306f3f0262cSandi //my own lock 307f3f0262cSandi $ip = io_readFile($lock); 308f3f0262cSandi if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){ 309f3f0262cSandi return false; 310f3f0262cSandi } 311f3f0262cSandi 312f3f0262cSandi return $ip; 313f3f0262cSandi} 314f3f0262cSandi 315f3f0262cSandi/** 31615fae107Sandi * Lock a page for editing 31715fae107Sandi * 31815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 319f3f0262cSandi */ 320f3f0262cSandifunction lock($id){ 321f3f0262cSandi $lock = wikiFN($id).'.lock'; 322f3f0262cSandi if($_SERVER['REMOTE_USER']){ 323f3f0262cSandi io_saveFile($lock,$_SERVER['REMOTE_USER']); 324f3f0262cSandi }else{ 325f3f0262cSandi io_saveFile($lock,clientIP()); 326f3f0262cSandi } 327f3f0262cSandi} 328f3f0262cSandi 329f3f0262cSandi/** 33015fae107Sandi * Unlock a page if it was locked by the user 331f3f0262cSandi * 33215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 33315fae107Sandi * @return bool true if a lock was removed 334f3f0262cSandi */ 335f3f0262cSandifunction unlock($id){ 336f3f0262cSandi $lock = wikiFN($id).'.lock'; 337f3f0262cSandi if(@file_exists($lock)){ 338f3f0262cSandi $ip = io_readFile($lock); 339f3f0262cSandi if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){ 340f3f0262cSandi @unlink($lock); 341f3f0262cSandi return true; 342f3f0262cSandi } 343f3f0262cSandi } 344f3f0262cSandi return false; 345f3f0262cSandi} 346f3f0262cSandi 347f3f0262cSandi/** 34815fae107Sandi * Remove unwanted chars from ID 34915fae107Sandi * 350f3f0262cSandi * Cleans a given ID to only use allowed characters. Accented characters are 351f3f0262cSandi * converted to unaccented ones 35215fae107Sandi * 35315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 354f3f0262cSandi */ 355f3f0262cSandifunction cleanID($id){ 356f3f0262cSandi global $conf; 357f3f0262cSandi global $lang; 358f3f0262cSandi $id = trim($id); 359c41c03f3Sandi $id = utf8_strtolower($id); 360f3f0262cSandi 361f3f0262cSandi //alternative namespace seperator 362f3f0262cSandi $id = strtr($id,';',':'); 3633021e063Sandi if($conf['useslash']){ 3643021e063Sandi $id = strtr($id,'/',':'); 3653021e063Sandi }else{ 3663021e063Sandi $id = strtr($id,'/','_'); 3673021e063Sandi } 368f3f0262cSandi 3698b709e9dSandi if($conf['deaccent']) $id = utf8_deaccent($id,-1); 370c41c03f3Sandi 371099ada41Sandi //remove specials 372099ada41Sandi //$id = preg_replace('#[\x00-\x20 ¡!"§$%&()\[\]{}¿\\?`\'\#~*+=,<>\|^°@µ¹²³¼½¬]#u','_',$id); 373099ada41Sandi $id = utf8_stripspecials($id,'_','_:.-'); 374c41c03f3Sandi 375c41c03f3Sandi //clean up 376f3f0262cSandi $id = preg_replace('#__#','_',$id); 377f3f0262cSandi $id = preg_replace('#:+#',':',$id); 378f3f0262cSandi $id = trim($id,':._-'); 379f3f0262cSandi $id = preg_replace('#:[:\._\-]+#',':',$id); 380f3f0262cSandi 381f3f0262cSandi return($id); 382f3f0262cSandi} 383f3f0262cSandi 384f3f0262cSandi/** 385f3f0262cSandi * returns the full path to the datafile specified by ID and 386f3f0262cSandi * optional revision 38715fae107Sandi * 38849c713a3Sandi * The filename is URL encoded to protect Unicode chars 38949c713a3Sandi * 39015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 391f3f0262cSandi */ 392f3f0262cSandifunction wikiFN($id,$rev=''){ 393f3f0262cSandi global $conf; 394f3f0262cSandi $id = cleanID($id); 395f3f0262cSandi $id = str_replace(':','/',$id); 396f3f0262cSandi if(empty($rev)){ 39749c713a3Sandi $fn = $conf['datadir'].'/'.$id.'.txt'; 398f3f0262cSandi }else{ 399f3f0262cSandi $fn = $conf['olddir'].'/'.$id.'.'.$rev.'.txt'; 40049c713a3Sandi if($conf['usegzip'] && !@file_exists($fn)){ 40149c713a3Sandi //return gzip if enabled and plaintext doesn't exist 40249c713a3Sandi $fn .= '.gz'; 40349c713a3Sandi } 40449c713a3Sandi } 40549c713a3Sandi $fn = utf8_encodeFN($fn); 406f3f0262cSandi return $fn; 407f3f0262cSandi} 408f3f0262cSandi 409f3f0262cSandi/** 410f3f0262cSandi * Returns the full filepath to a localized textfile if local 411f3f0262cSandi * version isn't found the english one is returned 41215fae107Sandi * 41315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 414f3f0262cSandi */ 415f3f0262cSandifunction localeFN($id){ 416f3f0262cSandi global $conf; 417f3f0262cSandi $file = './lang/'.$conf['lang'].'/'.$id.'.txt'; 418f3f0262cSandi if(!@file_exists($file)){ 419f3f0262cSandi //fall back to english 420f3f0262cSandi $file = './lang/en/'.$id.'.txt'; 421f3f0262cSandi } 422f3f0262cSandi return cleanText($file); 423f3f0262cSandi} 424f3f0262cSandi 425f3f0262cSandi/** 426f3f0262cSandi * convert line ending to unix format 427f3f0262cSandi * 42815fae107Sandi * @see formText() for 2crlf conversion 42915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 430f3f0262cSandi */ 431f3f0262cSandifunction cleanText($text){ 432f3f0262cSandi $text = preg_replace("/(\015\012)|(\015)/","\012",$text); 433f3f0262cSandi return $text; 434f3f0262cSandi} 435f3f0262cSandi 436f3f0262cSandi/** 437f3f0262cSandi * Prepares text for print in Webforms by encoding special chars. 438f3f0262cSandi * It also converts line endings to Windows format which is 439f3f0262cSandi * pseudo standard for webforms. 440f3f0262cSandi * 44115fae107Sandi * @see cleanText() for 2unix conversion 44215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 443f3f0262cSandi */ 444f3f0262cSandifunction formText($text){ 445f3f0262cSandi $text = preg_replace("/\012/","\015\012",$text); 446f3f0262cSandi return htmlspecialchars($text); 447f3f0262cSandi} 448f3f0262cSandi 449f3f0262cSandi/** 45015fae107Sandi * Returns the specified local text in parsed format 45115fae107Sandi * 45215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 453f3f0262cSandi */ 454f3f0262cSandifunction parsedLocale($id){ 455f3f0262cSandi //disable section editing 456f3f0262cSandi global $parser; 457f3f0262cSandi $se = $parser['secedit']; 458f3f0262cSandi $parser['secedit'] = false; 459f3f0262cSandi //fetch parsed locale 460f3f0262cSandi $html = io_cacheParse(localeFN($id)); 461f3f0262cSandi //reset section editing 462f3f0262cSandi $parser['secedit'] = $se; 463f3f0262cSandi return $html; 464f3f0262cSandi} 465f3f0262cSandi 466f3f0262cSandi/** 46715fae107Sandi * Returns the specified local text in raw format 46815fae107Sandi * 46915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 470f3f0262cSandi */ 471f3f0262cSandifunction rawLocale($id){ 472f3f0262cSandi return io_readFile(localeFN($id)); 473f3f0262cSandi} 474f3f0262cSandi 475f3f0262cSandi 476f3f0262cSandi/** 47715fae107Sandi * Returns the parsed Wikitext for the given id and revision. 47815fae107Sandi * 47915fae107Sandi * If $excuse is true an explanation is returned if the file 48015fae107Sandi * wasn't found 48115fae107Sandi * 48215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 483f3f0262cSandi */ 484f3f0262cSandifunction parsedWiki($id,$rev='',$excuse=true){ 485f3f0262cSandi $file = wikiFN($id,$rev); 486f3f0262cSandi $ret = ''; 487f3f0262cSandi 488f3f0262cSandi //ensure $id is in global $ID (needed for parsing) 489f3f0262cSandi global $ID; 490f3f0262cSandi $ID = $id; 491f3f0262cSandi 492f3f0262cSandi if($rev){ 493f3f0262cSandi if(@file_exists($file)){ 494f3f0262cSandi $ret = parse(io_readFile($file)); 495f3f0262cSandi }elseif($excuse){ 496f3f0262cSandi $ret = parsedLocale('norev'); 497f3f0262cSandi } 498f3f0262cSandi }else{ 499f3f0262cSandi if(@file_exists($file)){ 500f3f0262cSandi $ret = io_cacheParse($file); 501f3f0262cSandi }elseif($excuse){ 502f3f0262cSandi $ret = parsedLocale('newpage'); 503f3f0262cSandi } 504f3f0262cSandi } 505f3f0262cSandi return $ret; 506f3f0262cSandi} 507f3f0262cSandi 508f3f0262cSandi/** 509f3f0262cSandi * Returns the raw WikiText 51015fae107Sandi * 51115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 512f3f0262cSandi */ 513f3f0262cSandifunction rawWiki($id,$rev=''){ 514f3f0262cSandi return io_readFile(wikiFN($id,$rev)); 515f3f0262cSandi} 516f3f0262cSandi 517f3f0262cSandi/** 51815fae107Sandi * Returns the raw Wiki Text in three slices. 51915fae107Sandi * 52015fae107Sandi * The range parameter needs to have the form "from-to" 52115fae107Sandi * and gives the range of the section. 522f3f0262cSandi * The returned order is prefix, section and suffix. 52315fae107Sandi * 52415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 525f3f0262cSandi */ 526f3f0262cSandifunction rawWikiSlices($range,$id,$rev=''){ 527f3f0262cSandi list($from,$to) = split('-',$range,2); 528f3f0262cSandi $text = io_readFile(wikiFN($id,$rev)); 529f3f0262cSandi $text = split("\n",$text); 530f3f0262cSandi if(!$from) $from = 0; 531f3f0262cSandi if(!$to) $to = count($text); 532f3f0262cSandi 533f3f0262cSandi $slices[0] = join("\n",array_slice($text,0,$from)); 534f3f0262cSandi $slices[1] = join("\n",array_slice($text,$from,$to + 1 - $from)); 535f3f0262cSandi $slices[2] = join("\n",array_slice($text,$to+1)); 536f3f0262cSandi 537f3f0262cSandi return $slices; 538f3f0262cSandi} 539f3f0262cSandi 540f3f0262cSandi/** 54115fae107Sandi * Joins wiki text slices 54215fae107Sandi * 543f3f0262cSandi * function to join the text slices with correct lineendings again. 544f3f0262cSandi * When the pretty parameter is set to true it adds additional empty 545f3f0262cSandi * lines between sections if needed (used on saving). 54615fae107Sandi * 54715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 548f3f0262cSandi */ 549f3f0262cSandifunction con($pre,$text,$suf,$pretty=false){ 550f3f0262cSandi 551f3f0262cSandi if($pretty){ 552f3f0262cSandi if($pre && substr($pre,-1) != "\n") $pre .= "\n"; 553f3f0262cSandi if($suf && substr($text,-1) != "\n") $text .= "\n"; 554f3f0262cSandi } 555f3f0262cSandi 556f3f0262cSandi if($pre) $pre .= "\n"; 557f3f0262cSandi if($suf) $text .= "\n"; 558f3f0262cSandi return $pre.$text.$suf; 559f3f0262cSandi} 560f3f0262cSandi 561f3f0262cSandi/** 56215fae107Sandi * print debug messages 56315fae107Sandi * 564f3f0262cSandi * little function to print the content of a var 56515fae107Sandi * 56615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 567f3f0262cSandi */ 568f3f0262cSandifunction dbg($msg,$hidden=false){ 569f3f0262cSandi (!$hidden) ? print '<pre class="dbg">' : print "<!--\n"; 570f3f0262cSandi print_r($msg); 571f3f0262cSandi (!$hidden) ? print '</pre>' : print "\n-->"; 572f3f0262cSandi} 573f3f0262cSandi 574f3f0262cSandi/** 575f3f0262cSandi * Add's an entry to the changelog 57615fae107Sandi * 57715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 578f3f0262cSandi */ 579652610a2Sandifunction addLogEntry($date,$id,$summary=""){ 580f3f0262cSandi global $conf; 581c1049928Sandi $id = cleanID($id);//FIXME not needed anymore? 582c1049928Sandi 583c1049928Sandi if(!@is_writable($conf['changelog'])){ 584c1049928Sandi msg($conf['changelog'].' is not writable!',-1); 585c1049928Sandi return; 586c1049928Sandi } 587c1049928Sandi 588652610a2Sandi if(!$date) $date = time(); //use current time if none supplied 589f3f0262cSandi $remote = $_SERVER['REMOTE_ADDR']; 590f3f0262cSandi $user = $_SERVER['REMOTE_USER']; 591f3f0262cSandi 592f3f0262cSandi $logline = join("\t",array($date,$remote,$id,$user,$summary))."\n"; 593f3f0262cSandi 594c1049928Sandi //FIXME: use adjusted io_saveFile instead 595f3f0262cSandi $fh = fopen($conf['changelog'],'a'); 596f3f0262cSandi if($fh){ 597f3f0262cSandi fwrite($fh,$logline); 598f3f0262cSandi fclose($fh); 599f3f0262cSandi } 600f3f0262cSandi} 601f3f0262cSandi 602f3f0262cSandi/** 603f3f0262cSandi * returns an array of recently changed files using the 604f3f0262cSandi * changelog 60515fae107Sandi * 60615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 607f3f0262cSandi */ 608f3f0262cSandifunction getRecents($num=0,$incdel=false){ 609f3f0262cSandi global $conf; 610f3f0262cSandi $recent = array(); 611f3f0262cSandi if(!$num) $num = $conf['recent']; 612f3f0262cSandi 613c1049928Sandi if(!@is_readable($conf['changelog'])){ 614c1049928Sandi msg($conf['changelog'].' is not readable',-1); 615c1049928Sandi return $recent; 616c1049928Sandi } 617c1049928Sandi 618f3f0262cSandi $loglines = file($conf['changelog']); 619f3f0262cSandi rsort($loglines); //reverse sort on timestamp 620f3f0262cSandi 621f3f0262cSandi foreach ($loglines as $line){ 622f3f0262cSandi $line = rtrim($line); //remove newline 623f3f0262cSandi if(empty($line)) continue; //skip empty lines 624f3f0262cSandi $info = split("\t",$line); //split into parts 625f3f0262cSandi //add id if not in yet and file still exists and is allowed to read 626f3f0262cSandi if(!$recent[$info[2]] && 627f3f0262cSandi (@file_exists(wikiFN($info[2])) || $incdel) && 628f3f0262cSandi (auth_quickaclcheck($info[2]) >= AUTH_READ) 629f3f0262cSandi ){ 630f3f0262cSandi $recent[$info[2]]['date'] = $info[0]; 631f3f0262cSandi $recent[$info[2]]['ip'] = $info[1]; 632f3f0262cSandi $recent[$info[2]]['user'] = $info[3]; 633f3f0262cSandi $recent[$info[2]]['sum'] = $info[4]; 634f3f0262cSandi $recent[$info[2]]['del'] = !@file_exists(wikiFN($info[2])); 635f3f0262cSandi } 636f3f0262cSandi if(count($recent) >= $num){ 637f3f0262cSandi break; //finish if enough items found 638f3f0262cSandi } 639f3f0262cSandi } 640f3f0262cSandi return $recent; 641f3f0262cSandi} 642f3f0262cSandi 643f3f0262cSandi/** 644652610a2Sandi * gets additonal informations for a certain pagerevison 645652610a2Sandi * from the changelog 646652610a2Sandi * 647652610a2Sandi * @author Andreas Gohr <andi@splitbrain.org> 648652610a2Sandi */ 649652610a2Sandifunction getRevisionInfo($id,$rev){ 650652610a2Sandi global $conf; 651c1049928Sandi $info = array(); 652c1049928Sandi if(!@is_readable($conf['changelog'])){ 653c1049928Sandi msg($conf['changelog'].' is not readable',-1); 654c1049928Sandi return $recent; 655c1049928Sandi } 656652610a2Sandi $loglines = file($conf['changelog']); 657652610a2Sandi $loglines = preg_grep("/$rev\t\d+\.\d+\.\d+\.\d+\t$id\t/",$loglines); 658652610a2Sandi rsort($loglines); //reverse sort on timestamp (shouldn't be needed) 659652610a2Sandi $line = split("\t",$loglines[0]); 660652610a2Sandi $info['date'] = $line[0]; 661652610a2Sandi $info['ip'] = $line[1]; 662652610a2Sandi $info['user'] = $line[3]; 663652610a2Sandi $info['sum'] = $line[4]; 664652610a2Sandi return $info; 665652610a2Sandi} 666652610a2Sandi 667652610a2Sandi/** 668f3f0262cSandi * Saves a wikitext by calling io_saveFile 66915fae107Sandi * 67015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 671f3f0262cSandi */ 672f3f0262cSandifunction saveWikiText($id,$text,$summary){ 673f3f0262cSandi global $conf; 674f3f0262cSandi global $lang; 675f3f0262cSandi umask($conf['umask']); 676f3f0262cSandi // ignore if no changes were made 677f3f0262cSandi if($text == rawWiki($id,'')){ 678f3f0262cSandi return; 679f3f0262cSandi } 680f3f0262cSandi 681f3f0262cSandi $file = wikiFN($id); 682f3f0262cSandi $old = saveOldRevision($id); 683f3f0262cSandi 684f3f0262cSandi if (empty($text)){ 685f3f0262cSandi // remove empty files 686f3f0262cSandi @unlink($file); 687f3f0262cSandi $del = true; 6883ce054b3Sandi //autoset summary on deletion 6893ce054b3Sandi if(empty($summary)) $summary = $lang['deleted']; 690f3f0262cSandi }else{ 691f3f0262cSandi // save file (datadir is created in io_saveFile) 692f3f0262cSandi io_saveFile($file,$text); 693f3f0262cSandi $del = false; 694f3f0262cSandi } 695f3f0262cSandi 696652610a2Sandi addLogEntry(@filemtime($file),$id,$summary); 697f3f0262cSandi notify($id,$old,$summary); 698f3f0262cSandi 699f3f0262cSandi //purge cache on add by updating the purgefile 700f3f0262cSandi if($conf['purgeonadd'] && (!$old || $del)){ 701f3f0262cSandi io_saveFile($conf['datadir'].'/.cache/purgefile',time()); 702f3f0262cSandi } 703f3f0262cSandi} 704f3f0262cSandi 705f3f0262cSandi/** 706f3f0262cSandi * moves the current version to the attic and returns its 707f3f0262cSandi * revision date 70815fae107Sandi * 70915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 710f3f0262cSandi */ 711f3f0262cSandifunction saveOldRevision($id){ 712f3f0262cSandi global $conf; 713f3f0262cSandi umask($conf['umask']); 714f3f0262cSandi $oldf = wikiFN($id); 715f3f0262cSandi if(!@file_exists($oldf)) return ''; 716f3f0262cSandi $date = filemtime($oldf); 717f3f0262cSandi $newf = wikiFN($id,$date); 718f3f0262cSandi if(substr($newf,-3)=='.gz'){ 719f3f0262cSandi io_saveFile($newf,rawWiki($id)); 720f3f0262cSandi }else{ 721f3f0262cSandi io_makeFileDir($newf); 722f3f0262cSandi copy($oldf, $newf); 723f3f0262cSandi } 724f3f0262cSandi return $date; 725f3f0262cSandi} 726f3f0262cSandi 727f3f0262cSandi/** 728f3f0262cSandi * Sends a notify mail to the wikiadmin when a page was 729f3f0262cSandi * changed 73015fae107Sandi * 73115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 732f3f0262cSandi */ 733f3f0262cSandifunction notify($id,$rev="",$summary=""){ 734f3f0262cSandi global $lang; 735f3f0262cSandi global $conf; 736f3f0262cSandi $hdrs =''; 737f3f0262cSandi if(empty($conf['notify'])) return; //notify enabled? 738f3f0262cSandi 739f3f0262cSandi $text = rawLocale('mailtext'); 740f3f0262cSandi $text = str_replace('@DATE@',date($conf['dformat']),$text); 741f3f0262cSandi $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text); 742f3f0262cSandi $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text); 743f3f0262cSandi $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text); 744*ed7b5f09Sandi $text = str_replace('@NEWPAGE@',wl($id,'',true),$text); 745*ed7b5f09Sandi $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text); 746f3f0262cSandi $text = str_replace('@SUMMARY@',$summary,$text); 7477a82afdcSandi $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text); 748f3f0262cSandi 749f3f0262cSandi if($rev){ 750f3f0262cSandi $subject = $lang['mail_changed'].' '.$id; 751*ed7b5f09Sandi $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true),$text); 752f3f0262cSandi require_once("inc/DifferenceEngine.php"); 753f3f0262cSandi $df = new Diff(split("\n",rawWiki($id,$rev)), 754f3f0262cSandi split("\n",rawWiki($id))); 755f3f0262cSandi $dformat = new UnifiedDiffFormatter(); 756f3f0262cSandi $diff = $dformat->format($df); 757f3f0262cSandi }else{ 758f3f0262cSandi $subject=$lang['mail_newpage'].' '.$id; 759f3f0262cSandi $text = str_replace('@OLDPAGE@','none',$text); 760f3f0262cSandi $diff = rawWiki($id); 761f3f0262cSandi } 762f3f0262cSandi $text = str_replace('@DIFF@',$diff,$text); 763f3f0262cSandi 76444f669e9Sandi mail_send($conf['notify'],$subject,$text,$conf['mailfrom']); 765f3f0262cSandi} 766f3f0262cSandi 76715fae107Sandi/** 76815fae107Sandi * Return a list of available page revisons 76915fae107Sandi * 77015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 77115fae107Sandi */ 772f3f0262cSandifunction getRevisions($id){ 773f3f0262cSandi $revd = dirname(wikiFN($id,'foo')); 774f3f0262cSandi $revs = array(); 775f3f0262cSandi $clid = cleanID($id); 776f3f0262cSandi if(strrpos($clid,':')) $clid = substr($clid,strrpos($clid,':')+1); //remove path 777f3f0262cSandi 778f3f0262cSandi if (is_dir($revd) && $dh = opendir($revd)) { 779f3f0262cSandi while (($file = readdir($dh)) !== false) { 780f3f0262cSandi if (is_dir($revd.'/'.$file)) continue; 781f3f0262cSandi if (preg_match('/^'.$clid.'\.(\d+)\.txt(\.gz)?$/',$file,$match)){ 782f3f0262cSandi $revs[]=$match[1]; 783f3f0262cSandi } 784f3f0262cSandi } 785f3f0262cSandi closedir($dh); 786f3f0262cSandi } 787f3f0262cSandi rsort($revs); 788f3f0262cSandi return $revs; 789f3f0262cSandi} 790f3f0262cSandi 791f3f0262cSandi/** 792f3f0262cSandi * downloads a file from the net and saves it to the given location 79315fae107Sandi * 79415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 795f3f0262cSandi */ 796f3f0262cSandifunction download($url,$file){ 797f3f0262cSandi $fp = @fopen($url,"rb"); 798f3f0262cSandi if(!$fp) return false; 799f3f0262cSandi 800f3f0262cSandi while(!feof($fp)){ 801f3f0262cSandi $cont.= fread($fp,1024); 802f3f0262cSandi } 803f3f0262cSandi fclose($fp); 804f3f0262cSandi 805f3f0262cSandi $fp2 = @fopen($file,"w"); 806f3f0262cSandi if(!$fp2) return false; 807f3f0262cSandi fwrite($fp2,$cont); 808f3f0262cSandi fclose($fp2); 809f3f0262cSandi return true; 810f3f0262cSandi} 811f3f0262cSandi 812f3f0262cSandi/** 813f3f0262cSandi * extracts the query from a google referer 81415fae107Sandi * 81515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 816f3f0262cSandi */ 817f3f0262cSandifunction getGoogleQuery(){ 818f3f0262cSandi $url = parse_url($_SERVER['HTTP_REFERER']); 819f3f0262cSandi 820f3f0262cSandi if(!preg_match("#google\.#i",$url['host'])) return ''; 821f3f0262cSandi $query = array(); 822f3f0262cSandi parse_str($url['query'],$query); 823f3f0262cSandi 824f3f0262cSandi return $query['q']; 825f3f0262cSandi} 826f3f0262cSandi 827f3f0262cSandi/** 82815fae107Sandi * Try to set correct locale 82915fae107Sandi * 830095bfd5cSandi * @deprecated No longer used 83115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 832f3f0262cSandi */ 833f3f0262cSandifunction setCorrectLocale(){ 834f3f0262cSandi global $conf; 835f3f0262cSandi global $lang; 836f3f0262cSandi 837f3f0262cSandi $enc = strtoupper($lang['encoding']); 838f3f0262cSandi foreach ($lang['locales'] as $loc){ 839f3f0262cSandi //try locale 840f3f0262cSandi if(@setlocale(LC_ALL,$loc)) return; 841f3f0262cSandi //try loceale with encoding 842f3f0262cSandi if(@setlocale(LC_ALL,"$loc.$enc")) return; 843f3f0262cSandi } 844f3f0262cSandi //still here? try to set from environment 845f3f0262cSandi @setlocale(LC_ALL,""); 846f3f0262cSandi} 847f3f0262cSandi 848f3f0262cSandi/** 849f3f0262cSandi * Return the human readable size of a file 850f3f0262cSandi * 851f3f0262cSandi * @param int $size A file size 852f3f0262cSandi * @param int $dec A number of decimal places 853f3f0262cSandi * @author Martin Benjamin <b.martin@cybernet.ch> 854f3f0262cSandi * @author Aidan Lister <aidan@php.net> 855f3f0262cSandi * @version 1.0.0 856f3f0262cSandi */ 857f31d5b73Sandifunction filesize_h($size, $dec = 1){ 858f3f0262cSandi $sizes = array('B', 'KB', 'MB', 'GB'); 859f3f0262cSandi $count = count($sizes); 860f3f0262cSandi $i = 0; 861f3f0262cSandi 862f3f0262cSandi while ($size >= 1024 && ($i < $count - 1)) { 863f3f0262cSandi $size /= 1024; 864f3f0262cSandi $i++; 865f3f0262cSandi } 866f3f0262cSandi 867f3f0262cSandi return round($size, $dec) . ' ' . $sizes[$i]; 868f3f0262cSandi} 869f3f0262cSandi 87015fae107Sandi/** 87115fae107Sandi * Run a few sanity checks 87215fae107Sandi * 87315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 87415fae107Sandi */ 875f31d5b73Sandifunction getVersion(){ 876f31d5b73Sandi //import version string 877f31d5b73Sandi if(@file_exists('VERSION')){ 878f31d5b73Sandi //official release 879f31d5b73Sandi return 'Release '.io_readfile('VERSION'); 880f31d5b73Sandi }elseif(is_dir('_darcs')){ 881f31d5b73Sandi //darcs checkout 882f31d5b73Sandi $inv = file('_darcs/inventory'); 883f31d5b73Sandi $inv = preg_grep('#andi@splitbrain\.org\*\*\d{14}#',$inv); 884f31d5b73Sandi $cur = array_pop($inv); 885f31d5b73Sandi preg_match('#\*\*(\d{4})(\d{2})(\d{2})#',$cur,$matches); 886f31d5b73Sandi return 'Darcs '.$matches[1].'-'.$matches[2].'-'.$matches[3]; 887f31d5b73Sandi }else{ 888f31d5b73Sandi return 'snapshot?'; 889f31d5b73Sandi } 890f31d5b73Sandi} 891f31d5b73Sandi 892f31d5b73Sandi/** 893f31d5b73Sandi * Run a few sanity checks 894f31d5b73Sandi * 895f31d5b73Sandi * @author Andreas Gohr <andi@splitbrain.org> 896f31d5b73Sandi */ 897f3f0262cSandifunction check(){ 898f3f0262cSandi global $conf; 899f3f0262cSandi global $INFO; 900f3f0262cSandi 901f31d5b73Sandi msg('DokuWiki version: '.getVersion(),1); 902f31d5b73Sandi 90349022a38Sandi if(version_compare(phpversion(),'4.3.0','<')){ 90449022a38Sandi msg('Your PHP version is too old ('.phpversion().' vs. 4.3.+ recommended)',-1); 90549022a38Sandi }elseif(version_compare(phpversion(),'4.3.10','<')){ 90649022a38Sandi msg('Consider upgrading PHP to 4.3.10 or higher for security reasons (your version: '.phpversion().')',0); 90749022a38Sandi }else{ 90849022a38Sandi msg('PHP version '.phpversion(),1); 90949022a38Sandi } 91049022a38Sandi 911f3f0262cSandi if(is_writable($conf['changelog'])){ 912f3f0262cSandi msg('Changelog is writable',1); 913f3f0262cSandi }else{ 914f3f0262cSandi msg('Changelog is not writable',-1); 915f3f0262cSandi } 916f3f0262cSandi 917f3f0262cSandi if(is_writable($conf['datadir'])){ 918f3f0262cSandi msg('Datadir is writable',1); 919f3f0262cSandi }else{ 920f3f0262cSandi msg('Datadir is not writable',-1); 921f3f0262cSandi } 922f3f0262cSandi 923f3f0262cSandi if(is_writable($conf['olddir'])){ 924f3f0262cSandi msg('Attic is writable',1); 925f3f0262cSandi }else{ 926f3f0262cSandi msg('Attic is not writable',-1); 927f3f0262cSandi } 928f3f0262cSandi 929f3f0262cSandi if(is_writable($conf['mediadir'])){ 930f3f0262cSandi msg('Mediadir is writable',1); 931f3f0262cSandi }else{ 932f3f0262cSandi msg('Mediadir is not writable',-1); 933f3f0262cSandi } 934f3f0262cSandi 935f3f0262cSandi if(is_writable('conf/users.auth')){ 936f3f0262cSandi msg('conf/users.auth is writable',1); 937f3f0262cSandi }else{ 938f3f0262cSandi msg('conf/users.auth is not writable',0); 939f3f0262cSandi } 94093a9e835Sandi 94193a9e835Sandi if(function_exists('mb_strpos')){ 94293a9e835Sandi if(defined('UTF8_NOMBSTRING')){ 94393a9e835Sandi msg('mb_string extension is available but will not be used',0); 94493a9e835Sandi }else{ 94593a9e835Sandi msg('mb_string extension is available and will be used',1); 94693a9e835Sandi } 94793a9e835Sandi }else{ 94893a9e835Sandi msg('mb_string extension not available - PHP only replacements will be used',0); 94993a9e835Sandi } 950f3f0262cSandi 951f3f0262cSandi msg('Your current permission for this page is '.$INFO['perm'],0); 952f3f0262cSandi 953f3f0262cSandi if(is_writable($INFO['filepath'])){ 954f3f0262cSandi msg('The current page is writable by the webserver',0); 955f3f0262cSandi }else{ 956f3f0262cSandi msg('The current page is not writable by the webserver',0); 957f3f0262cSandi } 958f3f0262cSandi 959f3f0262cSandi if($INFO['writable']){ 960f3f0262cSandi msg('The current page is writable by you',0); 961f3f0262cSandi }else{ 962f3f0262cSandi msg('The current page is not writable you',0); 963f3f0262cSandi } 964f3f0262cSandi} 965f3f0262cSandi?> 966