xref: /dokuwiki/inc/common.php (revision c112d57805d6740c6e0a200fd5fd3f2717f86c69)
1ed7b5f09Sandi<?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
9ed7b5f09Sandi  if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/');
10ed7b5f09Sandi  require_once(DOKU_INC.'conf/dokuwiki.php');
11ed7b5f09Sandi  require_once(DOKU_INC.'inc/io.php');
12ed7b5f09Sandi  require_once(DOKU_INC.'inc/utf8.php');
13ed7b5f09Sandi  require_once(DOKU_INC.'inc/mail.php');
14*c112d578Sandi  require_once(DOKU_INC.'inc/parserutils.php');
15f3f0262cSandi
16f3f0262cSandi/**
1715fae107Sandi * Return info about the current document as associative
18f3f0262cSandi * array.
1915fae107Sandi *
2015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
21f3f0262cSandi */
22f3f0262cSandifunction pageinfo(){
23f3f0262cSandi  global $ID;
24f3f0262cSandi  global $REV;
25f3f0262cSandi  global $USERINFO;
26f3f0262cSandi  global $conf;
27f3f0262cSandi
28f3f0262cSandi  if($_SERVER['REMOTE_USER']){
29f3f0262cSandi    $info['user']     = $_SERVER['REMOTE_USER'];
30f3f0262cSandi    $info['userinfo'] = $USERINFO;
31f3f0262cSandi    $info['perm']     = auth_quickaclcheck($ID);
32f3f0262cSandi  }else{
33f3f0262cSandi    $info['user']     = '';
34f3f0262cSandi    $info['perm']     = auth_aclcheck($ID,'',null);
35f3f0262cSandi  }
36f3f0262cSandi
37f3f0262cSandi  $info['namespace'] = getNS($ID);
38f3f0262cSandi  $info['locked']    = checklock($ID);
39f3f0262cSandi  $info['filepath']  = realpath(wikiFN($ID,$REV));
40f3f0262cSandi  $info['exists']    = @file_exists($info['filepath']);
41f3f0262cSandi  if($REV && !$info['exists']){
42f3f0262cSandi    //check if current revision was meant
43f3f0262cSandi    $cur = wikiFN($ID);
44f3f0262cSandi    if(@file_exists($cur) && (@filemtime($cur) == $REV)){
45f3f0262cSandi      $info['filepath'] = realpath($cur);
46f3f0262cSandi      $info['exists']   = true;
47f3f0262cSandi      $REV = '';
48f3f0262cSandi    }
49f3f0262cSandi  }
50*c112d578Sandi  $info['rev'] = $REV;
51f3f0262cSandi  if($info['exists']){
52f3f0262cSandi    $info['writable'] = (is_writable($info['filepath']) &&
53f3f0262cSandi                         ($info['perm'] >= AUTH_EDIT));
54f3f0262cSandi  }else{
55f3f0262cSandi    $info['writable'] = ($info['perm'] >= AUTH_CREATE);
56f3f0262cSandi  }
57f3f0262cSandi  $info['editable']  = ($info['writable'] && empty($info['lock']));
58f3f0262cSandi  $info['lastmod']   = @filemtime($info['filepath']);
59f3f0262cSandi
60652610a2Sandi  //who's the editor
61652610a2Sandi  if($REV){
62652610a2Sandi    $revinfo = getRevisionInfo($ID,$REV);
63652610a2Sandi  }else{
64652610a2Sandi    $revinfo = getRevisionInfo($ID,$info['lastmod']);
65652610a2Sandi  }
66652610a2Sandi  $info['ip']     = $revinfo['ip'];
67652610a2Sandi  $info['user']   = $revinfo['user'];
68652610a2Sandi  $info['sum']    = $revinfo['sum'];
69652610a2Sandi  $info['editor'] = $revinfo['ip'];
70652610a2Sandi  if($revinfo['user']) $info['editor'].= ' ('.$revinfo['user'].')';
71652610a2Sandi
72f3f0262cSandi  return $info;
73f3f0262cSandi}
74f3f0262cSandi
75f3f0262cSandi/**
760396becbSandi * print a message
770396becbSandi *
780396becbSandi * If HTTP headers were not sent yet the message is added
790396becbSandi * to the global message array else it's printed directly
800396becbSandi * using html_msgarea()
810396becbSandi *
82f3f0262cSandi *
83f3f0262cSandi * Levels can be:
84f3f0262cSandi *
85f3f0262cSandi * -1 error
86f3f0262cSandi *  0 info
87f3f0262cSandi *  1 success
8815fae107Sandi *
8915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
900396becbSandi * @see    html_msgarea
91f3f0262cSandi */
92f3f0262cSandifunction msg($message,$lvl=0){
93f3f0262cSandi  global $MSG;
94f3f0262cSandi  $errors[-1] = 'error';
95f3f0262cSandi  $errors[0]  = 'info';
96f3f0262cSandi  $errors[1]  = 'success';
97f3f0262cSandi
98cc20ad51Sandi  if(!headers_sent()){
99f3f0262cSandi    if(!isset($MSG)) $MSG = array();
100f3f0262cSandi    $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message);
1010396becbSandi  }else{
1020396becbSandi    $MSG = array();
1030396becbSandi    $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message);
1040396becbSandi    html_msgarea();
1050396becbSandi  }
106f3f0262cSandi}
107f3f0262cSandi
108f3f0262cSandi/**
10915fae107Sandi * This builds the breadcrumb trail and returns it as array
11015fae107Sandi *
11115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
112f3f0262cSandi */
113f3f0262cSandifunction breadcrumbs(){
114f3f0262cSandi  global $ID;
115f3f0262cSandi  global $ACT;
116f3f0262cSandi  global $conf;
117f3f0262cSandi  $crumbs = $_SESSION[$conf['title']]['bc'];
118f3f0262cSandi
119f3f0262cSandi  //first visit?
120f3f0262cSandi  if (!is_array($crumbs)){
121f3f0262cSandi    $crumbs = array();
122f3f0262cSandi  }
123f3f0262cSandi  //we only save on show and existing wiki documents
124a77f5846Sjan  $file = wikiFN($ID);
125a77f5846Sjan  if($ACT != 'show' || !@file_exists($file)){
126f3f0262cSandi    $_SESSION[$conf['title']]['bc'] = $crumbs;
127f3f0262cSandi    return $crumbs;
128f3f0262cSandi  }
129a77f5846Sjan
130a77f5846Sjan  // page names
131a77f5846Sjan  $name = noNS($ID);
132a77f5846Sjan  if ($conf['useheading']) {
133a77f5846Sjan    // get page title
134a77f5846Sjan    $title = getFirstHeading(io_readFile($file));
135a77f5846Sjan    if ($title) {
136a77f5846Sjan      $name = $title;
137a77f5846Sjan    }
138a77f5846Sjan  }
139a77f5846Sjan
140f3f0262cSandi  //remove ID from array
141a77f5846Sjan  if (isset($crumbs[$ID])) {
142a77f5846Sjan    unset($crumbs[$ID]);
143f3f0262cSandi  }
144f3f0262cSandi
145f3f0262cSandi  //add to array
146a77f5846Sjan  $crumbs[$ID] = $name;
147f3f0262cSandi  //reduce size
148f3f0262cSandi  while(count($crumbs) > $conf['breadcrumbs']){
149f3f0262cSandi    array_shift($crumbs);
150f3f0262cSandi  }
151f3f0262cSandi  //save to session
152f3f0262cSandi  $_SESSION[$conf['title']]['bc'] = $crumbs;
153f3f0262cSandi  return $crumbs;
154f3f0262cSandi}
155f3f0262cSandi
156f3f0262cSandi/**
15715fae107Sandi * Filter for page IDs
15815fae107Sandi *
159f3f0262cSandi * This is run on a ID before it is outputted somewhere
160f3f0262cSandi * currently used to replace the colon with something else
161f3f0262cSandi * on Windows systems and to have proper URL encoding
16215fae107Sandi *
16349c713a3Sandi * Urlencoding is ommitted when the second parameter is false
16449c713a3Sandi *
16515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
166f3f0262cSandi */
16749c713a3Sandifunction idfilter($id,$ue=true){
168f3f0262cSandi  global $conf;
169f3f0262cSandi  if ($conf['useslash'] && $conf['userewrite']){
170f3f0262cSandi    $id = strtr($id,':','/');
171f3f0262cSandi  }elseif (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' &&
172f3f0262cSandi      $conf['userewrite']) {
173f3f0262cSandi    $id = strtr($id,':',';');
174f3f0262cSandi  }
17549c713a3Sandi  if($ue){
176f3f0262cSandi    $id = urlencode($id);
177f3f0262cSandi    $id = str_replace('%3A',':',$id); //keep as colon
178f3f0262cSandi    $id = str_replace('%2F','/',$id); //keep as slash
17949c713a3Sandi  }
180f3f0262cSandi  return $id;
181f3f0262cSandi}
182f3f0262cSandi
183f3f0262cSandi/**
184ed7b5f09Sandi * This builds a link to a wikipage
18515fae107Sandi *
18615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
187f3f0262cSandi */
188ed7b5f09Sandifunction wl($id='',$more='',$abs=false){
189f3f0262cSandi  global $conf;
190f3f0262cSandi  $more = str_replace(',','&amp;',$more);
191f3f0262cSandi
192f3f0262cSandi  $id    = idfilter($id);
193ed7b5f09Sandi  if($abs){
194ed7b5f09Sandi    $xlink = DOKU_URL;
195ed7b5f09Sandi  }else{
196ed7b5f09Sandi    $xlink = DOKU_BASE;
197ed7b5f09Sandi  }
198f3f0262cSandi
199f3f0262cSandi  if(!$conf['userewrite']){
200ed7b5f09Sandi    $xlink .= DOKU_SCRIPT.'?id='.$id;
201f3f0262cSandi    if($more) $xlink .= '&amp;'.$more;
202f3f0262cSandi  }else{
203f3f0262cSandi    $xlink .= $id;
204f3f0262cSandi    if($more) $xlink .= '?'.$more;
205f3f0262cSandi  }
206f3f0262cSandi
207f3f0262cSandi  return $xlink;
208f3f0262cSandi}
209f3f0262cSandi
210f3f0262cSandi/**
211f3f0262cSandi * Just builds a link to a script
21215fae107Sandi *
213ed7b5f09Sandi * @todo   maybe obsolete
21415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
215f3f0262cSandi */
216f3f0262cSandifunction script($script='doku.php'){
217ed7b5f09Sandi#  $link = getBaseURL();
218ed7b5f09Sandi#  $link .= $script;
219ed7b5f09Sandi#  return $link;
220ed7b5f09Sandi  return DOKU_BASE.DOKU_SCRIPT;
221f3f0262cSandi}
222f3f0262cSandi
223f3f0262cSandi/**
22415fae107Sandi * Spamcheck against wordlist
22515fae107Sandi *
226f3f0262cSandi * Checks the wikitext against a list of blocked expressions
227f3f0262cSandi * returns true if the text contains any bad words
22815fae107Sandi *
22915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
230f3f0262cSandi */
231f3f0262cSandifunction checkwordblock(){
232f3f0262cSandi  global $TEXT;
233f3f0262cSandi  global $conf;
234f3f0262cSandi
235f3f0262cSandi  if(!$conf['usewordblock']) return false;
236f3f0262cSandi
237703f6fdeSandi  $blockfile = file('conf/wordblock.conf');
2383e2965d7Sandi  //how many lines to read at once (to work around some PCRE limits)
2393e2965d7Sandi  if(version_compare(phpversion(),'4.3.0','<')){
2403e2965d7Sandi    //old versions of PCRE define a maximum of parenthesises even if no
2413e2965d7Sandi    //backreferences are used - the maximum is 99
2423e2965d7Sandi    //this is very bad performancewise and may even be too high still
2433e2965d7Sandi    $chunksize = 40;
2443e2965d7Sandi  }else{
245703f6fdeSandi    //read file in chunks of 600 - this should work around the
2463e2965d7Sandi    //MAX_PATTERN_SIZE in modern PCRE
2473e2965d7Sandi    $chunksize = 600;
2483e2965d7Sandi  }
2493e2965d7Sandi  while($blocks = array_splice($blockfile,0,$chunksize)){
250f3f0262cSandi    $re = array();
251f3f0262cSandi    #build regexp from blocks
252f3f0262cSandi    foreach($blocks as $block){
253f3f0262cSandi      $block = preg_replace('/#.*$/','',$block);
254f3f0262cSandi      $block = trim($block);
255f3f0262cSandi      if(empty($block)) continue;
256f3f0262cSandi      $re[]  = $block;
257f3f0262cSandi    }
258f3f0262cSandi    if(preg_match('#('.join('|',$re).')#si',$TEXT)) return true;
259703f6fdeSandi  }
260f3f0262cSandi  return false;
261f3f0262cSandi}
262f3f0262cSandi
263f3f0262cSandi/**
26415fae107Sandi * Return the IP of the client
26515fae107Sandi *
26615fae107Sandi * Honours X-Forwarded-For Proxy Headers
26715fae107Sandi *
26815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
269f3f0262cSandi */
270f3f0262cSandifunction clientIP(){
271f3f0262cSandi  $my = $_SERVER['REMOTE_ADDR'];
272f3f0262cSandi  if($_SERVER['HTTP_X_FORWARDED_FOR']){
273f3f0262cSandi    $my .= ' ('.$_SERVER['HTTP_X_FORWARDED_FOR'].')';
274f3f0262cSandi  }
275f3f0262cSandi  return $my;
276f3f0262cSandi}
277f3f0262cSandi
278f3f0262cSandi/**
27915fae107Sandi * Checks if a given page is currently locked.
28015fae107Sandi *
281f3f0262cSandi * removes stale lockfiles
28215fae107Sandi *
28315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
284f3f0262cSandi */
285f3f0262cSandifunction checklock($id){
286f3f0262cSandi  global $conf;
287f3f0262cSandi  $lock = wikiFN($id).'.lock';
288f3f0262cSandi
289f3f0262cSandi  //no lockfile
290f3f0262cSandi  if(!@file_exists($lock)) return false;
291f3f0262cSandi
292f3f0262cSandi  //lockfile expired
293f3f0262cSandi  if((time() - filemtime($lock)) > $conf['locktime']){
294f3f0262cSandi    unlink($lock);
295f3f0262cSandi    return false;
296f3f0262cSandi  }
297f3f0262cSandi
298f3f0262cSandi  //my own lock
299f3f0262cSandi  $ip = io_readFile($lock);
300f3f0262cSandi  if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
301f3f0262cSandi    return false;
302f3f0262cSandi  }
303f3f0262cSandi
304f3f0262cSandi  return $ip;
305f3f0262cSandi}
306f3f0262cSandi
307f3f0262cSandi/**
30815fae107Sandi * Lock a page for editing
30915fae107Sandi *
31015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
311f3f0262cSandi */
312f3f0262cSandifunction lock($id){
313f3f0262cSandi  $lock = wikiFN($id).'.lock';
314f3f0262cSandi  if($_SERVER['REMOTE_USER']){
315f3f0262cSandi    io_saveFile($lock,$_SERVER['REMOTE_USER']);
316f3f0262cSandi  }else{
317f3f0262cSandi    io_saveFile($lock,clientIP());
318f3f0262cSandi  }
319f3f0262cSandi}
320f3f0262cSandi
321f3f0262cSandi/**
32215fae107Sandi * Unlock a page if it was locked by the user
323f3f0262cSandi *
32415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
32515fae107Sandi * @return bool true if a lock was removed
326f3f0262cSandi */
327f3f0262cSandifunction unlock($id){
328f3f0262cSandi  $lock = wikiFN($id).'.lock';
329f3f0262cSandi  if(@file_exists($lock)){
330f3f0262cSandi    $ip = io_readFile($lock);
331f3f0262cSandi    if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
332f3f0262cSandi      @unlink($lock);
333f3f0262cSandi      return true;
334f3f0262cSandi    }
335f3f0262cSandi  }
336f3f0262cSandi  return false;
337f3f0262cSandi}
338f3f0262cSandi
339f3f0262cSandi/**
340f3f0262cSandi * convert line ending to unix format
341f3f0262cSandi *
34215fae107Sandi * @see    formText() for 2crlf conversion
34315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
344f3f0262cSandi */
345f3f0262cSandifunction cleanText($text){
346f3f0262cSandi  $text = preg_replace("/(\015\012)|(\015)/","\012",$text);
347f3f0262cSandi  return $text;
348f3f0262cSandi}
349f3f0262cSandi
350f3f0262cSandi/**
351f3f0262cSandi * Prepares text for print in Webforms by encoding special chars.
352f3f0262cSandi * It also converts line endings to Windows format which is
353f3f0262cSandi * pseudo standard for webforms.
354f3f0262cSandi *
35515fae107Sandi * @see    cleanText() for 2unix conversion
35615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
357f3f0262cSandi */
358f3f0262cSandifunction formText($text){
359f3f0262cSandi  $text = preg_replace("/\012/","\015\012",$text);
360f3f0262cSandi  return htmlspecialchars($text);
361f3f0262cSandi}
362f3f0262cSandi
363f3f0262cSandi/**
36415fae107Sandi * Returns the specified local text in raw format
36515fae107Sandi *
36615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
367f3f0262cSandi */
368f3f0262cSandifunction rawLocale($id){
369f3f0262cSandi  return io_readFile(localeFN($id));
370f3f0262cSandi}
371f3f0262cSandi
372f3f0262cSandi/**
373f3f0262cSandi * Returns the raw WikiText
37415fae107Sandi *
37515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
376f3f0262cSandi */
377f3f0262cSandifunction rawWiki($id,$rev=''){
378f3f0262cSandi  return io_readFile(wikiFN($id,$rev));
379f3f0262cSandi}
380f3f0262cSandi
381f3f0262cSandi/**
38215fae107Sandi * Returns the raw Wiki Text in three slices.
38315fae107Sandi *
38415fae107Sandi * The range parameter needs to have the form "from-to"
38515fae107Sandi * and gives the range of the section.
386f3f0262cSandi * The returned order is prefix, section and suffix.
38715fae107Sandi *
38815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
389f3f0262cSandi */
390f3f0262cSandifunction rawWikiSlices($range,$id,$rev=''){
391f3f0262cSandi  list($from,$to) = split('-',$range,2);
392f3f0262cSandi  $text = io_readFile(wikiFN($id,$rev));
393f3f0262cSandi  $text = split("\n",$text);
394f3f0262cSandi  if(!$from) $from = 0;
395f3f0262cSandi  if(!$to)   $to   = count($text);
396f3f0262cSandi
397f3f0262cSandi  $slices[0] = join("\n",array_slice($text,0,$from));
398f3f0262cSandi  $slices[1] = join("\n",array_slice($text,$from,$to + 1  - $from));
399f3f0262cSandi  $slices[2] = join("\n",array_slice($text,$to+1));
400f3f0262cSandi
401f3f0262cSandi  return $slices;
402f3f0262cSandi}
403f3f0262cSandi
404f3f0262cSandi/**
40515fae107Sandi * Joins wiki text slices
40615fae107Sandi *
407f3f0262cSandi * function to join the text slices with correct lineendings again.
408f3f0262cSandi * When the pretty parameter is set to true it adds additional empty
409f3f0262cSandi * lines between sections if needed (used on saving).
41015fae107Sandi *
41115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
412f3f0262cSandi */
413f3f0262cSandifunction con($pre,$text,$suf,$pretty=false){
414f3f0262cSandi
415f3f0262cSandi  if($pretty){
416f3f0262cSandi    if($pre && substr($pre,-1) != "\n") $pre .= "\n";
417f3f0262cSandi    if($suf && substr($text,-1) != "\n") $text .= "\n";
418f3f0262cSandi  }
419f3f0262cSandi
420f3f0262cSandi  if($pre) $pre .= "\n";
421f3f0262cSandi  if($suf) $text .= "\n";
422f3f0262cSandi  return $pre.$text.$suf;
423f3f0262cSandi}
424f3f0262cSandi
425f3f0262cSandi/**
42615fae107Sandi * print debug messages
42715fae107Sandi *
428f3f0262cSandi * little function to print the content of a var
42915fae107Sandi *
43015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
431f3f0262cSandi */
432f3f0262cSandifunction dbg($msg,$hidden=false){
433f3f0262cSandi  (!$hidden) ? print '<pre class="dbg">' : print "<!--\n";
434f3f0262cSandi  print_r($msg);
435f3f0262cSandi  (!$hidden) ? print '</pre>' : print "\n-->";
436f3f0262cSandi}
437f3f0262cSandi
438f3f0262cSandi/**
439f3f0262cSandi * Add's an entry to the changelog
44015fae107Sandi *
44115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
442f3f0262cSandi */
443652610a2Sandifunction addLogEntry($date,$id,$summary=""){
444f3f0262cSandi  global $conf;
445c1049928Sandi  $id     = cleanID($id);//FIXME not needed anymore?
446c1049928Sandi
447c1049928Sandi  if(!@is_writable($conf['changelog'])){
448c1049928Sandi    msg($conf['changelog'].' is not writable!',-1);
449c1049928Sandi    return;
450c1049928Sandi  }
451c1049928Sandi
452652610a2Sandi  if(!$date) $date = time(); //use current time if none supplied
453f3f0262cSandi  $remote = $_SERVER['REMOTE_ADDR'];
454f3f0262cSandi  $user   = $_SERVER['REMOTE_USER'];
455f3f0262cSandi
456f3f0262cSandi  $logline = join("\t",array($date,$remote,$id,$user,$summary))."\n";
457f3f0262cSandi
458c1049928Sandi  //FIXME: use adjusted io_saveFile instead
459f3f0262cSandi  $fh = fopen($conf['changelog'],'a');
460f3f0262cSandi  if($fh){
461f3f0262cSandi    fwrite($fh,$logline);
462f3f0262cSandi    fclose($fh);
463f3f0262cSandi  }
464f3f0262cSandi}
465f3f0262cSandi
466f3f0262cSandi/**
467f3f0262cSandi * returns an array of recently changed files using the
468f3f0262cSandi * changelog
46915fae107Sandi *
47015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
471f3f0262cSandi */
472f3f0262cSandifunction getRecents($num=0,$incdel=false){
473f3f0262cSandi  global $conf;
474f3f0262cSandi  $recent = array();
475f3f0262cSandi  if(!$num) $num = $conf['recent'];
476f3f0262cSandi
477c1049928Sandi  if(!@is_readable($conf['changelog'])){
478c1049928Sandi    msg($conf['changelog'].' is not readable',-1);
479c1049928Sandi    return $recent;
480c1049928Sandi  }
481c1049928Sandi
482f3f0262cSandi  $loglines = file($conf['changelog']);
483f3f0262cSandi  rsort($loglines); //reverse sort on timestamp
484f3f0262cSandi
485f3f0262cSandi  foreach ($loglines as $line){
486f3f0262cSandi    $line = rtrim($line);        //remove newline
487f3f0262cSandi    if(empty($line)) continue;   //skip empty lines
488f3f0262cSandi    $info = split("\t",$line);   //split into parts
489f3f0262cSandi    //add id if not in yet and file still exists and is allowed to read
490f3f0262cSandi    if(!$recent[$info[2]] &&
491f3f0262cSandi       (@file_exists(wikiFN($info[2])) || $incdel) &&
492f3f0262cSandi       (auth_quickaclcheck($info[2]) >= AUTH_READ)
493f3f0262cSandi      ){
494f3f0262cSandi      $recent[$info[2]]['date'] = $info[0];
495f3f0262cSandi      $recent[$info[2]]['ip']   = $info[1];
496f3f0262cSandi      $recent[$info[2]]['user'] = $info[3];
497f3f0262cSandi      $recent[$info[2]]['sum']  = $info[4];
498f3f0262cSandi      $recent[$info[2]]['del']  = !@file_exists(wikiFN($info[2]));
499f3f0262cSandi    }
500f3f0262cSandi    if(count($recent) >= $num){
501f3f0262cSandi      break; //finish if enough items found
502f3f0262cSandi    }
503f3f0262cSandi  }
504f3f0262cSandi  return $recent;
505f3f0262cSandi}
506f3f0262cSandi
507f3f0262cSandi/**
508652610a2Sandi * gets additonal informations for a certain pagerevison
509652610a2Sandi * from the changelog
510652610a2Sandi *
511652610a2Sandi * @author Andreas Gohr <andi@splitbrain.org>
512652610a2Sandi */
513652610a2Sandifunction getRevisionInfo($id,$rev){
514652610a2Sandi  global $conf;
515258641c6Sandi
516258641c6Sandi  if(!$rev) return(null);
517258641c6Sandi
518c1049928Sandi  $info = array();
519c1049928Sandi  if(!@is_readable($conf['changelog'])){
520c1049928Sandi    msg($conf['changelog'].' is not readable',-1);
521c1049928Sandi    return $recent;
522c1049928Sandi  }
523652610a2Sandi  $loglines = file($conf['changelog']);
524652610a2Sandi  $loglines = preg_grep("/$rev\t\d+\.\d+\.\d+\.\d+\t$id\t/",$loglines);
525652610a2Sandi  rsort($loglines); //reverse sort on timestamp (shouldn't be needed)
526652610a2Sandi  $line = split("\t",$loglines[0]);
527652610a2Sandi  $info['date'] = $line[0];
528652610a2Sandi  $info['ip']   = $line[1];
529652610a2Sandi  $info['user'] = $line[3];
530652610a2Sandi  $info['sum']   = $line[4];
531652610a2Sandi  return $info;
532652610a2Sandi}
533652610a2Sandi
534652610a2Sandi/**
535f3f0262cSandi * Saves a wikitext by calling io_saveFile
53615fae107Sandi *
53715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
538f3f0262cSandi */
539f3f0262cSandifunction saveWikiText($id,$text,$summary){
540f3f0262cSandi  global $conf;
541f3f0262cSandi  global $lang;
542f3f0262cSandi  umask($conf['umask']);
543f3f0262cSandi  // ignore if no changes were made
544f3f0262cSandi  if($text == rawWiki($id,'')){
545f3f0262cSandi    return;
546f3f0262cSandi  }
547f3f0262cSandi
548f3f0262cSandi  $file = wikiFN($id);
549f3f0262cSandi  $old  = saveOldRevision($id);
550f3f0262cSandi
551f3f0262cSandi  if (empty($text)){
552f3f0262cSandi    // remove empty files
553f3f0262cSandi    @unlink($file);
554f3f0262cSandi    $del = true;
5553ce054b3Sandi    //autoset summary on deletion
5563ce054b3Sandi    if(empty($summary)) $summary = $lang['deleted'];
55753d6ccfeSandi    //remove empty namespaces
55853d6ccfeSandi    io_sweepNS($id);
559f3f0262cSandi  }else{
560f3f0262cSandi    // save file (datadir is created in io_saveFile)
561f3f0262cSandi    io_saveFile($file,$text);
562f3f0262cSandi    $del = false;
563f3f0262cSandi  }
564f3f0262cSandi
565652610a2Sandi  addLogEntry(@filemtime($file),$id,$summary);
566f3f0262cSandi  notify($id,$old,$summary);
567f3f0262cSandi
568f3f0262cSandi  //purge cache on add by updating the purgefile
569f3f0262cSandi  if($conf['purgeonadd'] && (!$old || $del)){
5709afe4dbfSjan    io_saveFile($conf['datadir'].'/_cache/purgefile',time());
571f3f0262cSandi  }
572f3f0262cSandi}
573f3f0262cSandi
574f3f0262cSandi/**
575f3f0262cSandi * moves the current version to the attic and returns its
576f3f0262cSandi * revision date
57715fae107Sandi *
57815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
579f3f0262cSandi */
580f3f0262cSandifunction saveOldRevision($id){
581f3f0262cSandi	global $conf;
582f3f0262cSandi  umask($conf['umask']);
583f3f0262cSandi  $oldf = wikiFN($id);
584f3f0262cSandi  if(!@file_exists($oldf)) return '';
585f3f0262cSandi  $date = filemtime($oldf);
586f3f0262cSandi  $newf = wikiFN($id,$date);
587f3f0262cSandi  if(substr($newf,-3)=='.gz'){
588f3f0262cSandi    io_saveFile($newf,rawWiki($id));
589f3f0262cSandi  }else{
590f3f0262cSandi    io_makeFileDir($newf);
591f3f0262cSandi    copy($oldf, $newf);
592f3f0262cSandi  }
593f3f0262cSandi  return $date;
594f3f0262cSandi}
595f3f0262cSandi
596f3f0262cSandi/**
597f3f0262cSandi * Sends a notify mail to the wikiadmin when a page was
598f3f0262cSandi * changed
59915fae107Sandi *
60015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
601f3f0262cSandi */
602f3f0262cSandifunction notify($id,$rev="",$summary=""){
603f3f0262cSandi  global $lang;
604f3f0262cSandi  global $conf;
605f3f0262cSandi  $hdrs ='';
606f3f0262cSandi  if(empty($conf['notify'])) return; //notify enabled?
607f3f0262cSandi
608f3f0262cSandi  $text = rawLocale('mailtext');
609f3f0262cSandi  $text = str_replace('@DATE@',date($conf['dformat']),$text);
610f3f0262cSandi  $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text);
611f3f0262cSandi  $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text);
612f3f0262cSandi  $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text);
613ed7b5f09Sandi  $text = str_replace('@NEWPAGE@',wl($id,'',true),$text);
614ed7b5f09Sandi  $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text);
615f3f0262cSandi  $text = str_replace('@SUMMARY@',$summary,$text);
6167a82afdcSandi  $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text);
617f3f0262cSandi
618f3f0262cSandi  if($rev){
619f3f0262cSandi    $subject = $lang['mail_changed'].' '.$id;
620ed7b5f09Sandi    $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true),$text);
621f3f0262cSandi    require_once("inc/DifferenceEngine.php");
622f3f0262cSandi    $df  = new Diff(split("\n",rawWiki($id,$rev)),
623f3f0262cSandi                    split("\n",rawWiki($id)));
624f3f0262cSandi    $dformat = new UnifiedDiffFormatter();
625f3f0262cSandi    $diff    = $dformat->format($df);
626f3f0262cSandi  }else{
627f3f0262cSandi    $subject=$lang['mail_newpage'].' '.$id;
628f3f0262cSandi    $text = str_replace('@OLDPAGE@','none',$text);
629f3f0262cSandi    $diff = rawWiki($id);
630f3f0262cSandi  }
631f3f0262cSandi  $text = str_replace('@DIFF@',$diff,$text);
632f3f0262cSandi
63344f669e9Sandi  mail_send($conf['notify'],$subject,$text,$conf['mailfrom']);
634f3f0262cSandi}
635f3f0262cSandi
63615fae107Sandi/**
63715fae107Sandi * Return a list of available page revisons
63815fae107Sandi *
63915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
64015fae107Sandi */
641f3f0262cSandifunction getRevisions($id){
642f3f0262cSandi  $revd = dirname(wikiFN($id,'foo'));
643f3f0262cSandi  $revs = array();
644f3f0262cSandi  $clid = cleanID($id);
645f3f0262cSandi  if(strrpos($clid,':')) $clid = substr($clid,strrpos($clid,':')+1); //remove path
646f3f0262cSandi
647f3f0262cSandi  if (is_dir($revd) && $dh = opendir($revd)) {
648f3f0262cSandi    while (($file = readdir($dh)) !== false) {
649f3f0262cSandi      if (is_dir($revd.'/'.$file)) continue;
650f3f0262cSandi      if (preg_match('/^'.$clid.'\.(\d+)\.txt(\.gz)?$/',$file,$match)){
651f3f0262cSandi        $revs[]=$match[1];
652f3f0262cSandi      }
653f3f0262cSandi    }
654f3f0262cSandi    closedir($dh);
655f3f0262cSandi  }
656f3f0262cSandi  rsort($revs);
657f3f0262cSandi  return $revs;
658f3f0262cSandi}
659f3f0262cSandi
660f3f0262cSandi/**
661f3f0262cSandi * extracts the query from a google referer
66215fae107Sandi *
6636b13307fSandi * @todo   should be more generic and support yahoo et al
66415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
665f3f0262cSandi */
666f3f0262cSandifunction getGoogleQuery(){
667f3f0262cSandi  $url = parse_url($_SERVER['HTTP_REFERER']);
668f3f0262cSandi
669f3f0262cSandi  if(!preg_match("#google\.#i",$url['host'])) return '';
670f3f0262cSandi  $query = array();
671f3f0262cSandi  parse_str($url['query'],$query);
672f3f0262cSandi
673f3f0262cSandi  return $query['q'];
674f3f0262cSandi}
675f3f0262cSandi
676f3f0262cSandi/**
67715fae107Sandi * Try to set correct locale
67815fae107Sandi *
679095bfd5cSandi * @deprecated No longer used
68015fae107Sandi * @author     Andreas Gohr <andi@splitbrain.org>
681f3f0262cSandi */
682f3f0262cSandifunction setCorrectLocale(){
683f3f0262cSandi  global $conf;
684f3f0262cSandi  global $lang;
685f3f0262cSandi
686f3f0262cSandi  $enc = strtoupper($lang['encoding']);
687f3f0262cSandi  foreach ($lang['locales'] as $loc){
688f3f0262cSandi    //try locale
689f3f0262cSandi    if(@setlocale(LC_ALL,$loc)) return;
690f3f0262cSandi    //try loceale with encoding
691f3f0262cSandi    if(@setlocale(LC_ALL,"$loc.$enc")) return;
692f3f0262cSandi  }
693f3f0262cSandi  //still here? try to set from environment
694f3f0262cSandi  @setlocale(LC_ALL,"");
695f3f0262cSandi}
696f3f0262cSandi
697f3f0262cSandi/**
698f3f0262cSandi * Return the human readable size of a file
699f3f0262cSandi *
700f3f0262cSandi * @param       int    $size   A file size
701f3f0262cSandi * @param       int    $dec    A number of decimal places
702f3f0262cSandi * @author      Martin Benjamin <b.martin@cybernet.ch>
703f3f0262cSandi * @author      Aidan Lister <aidan@php.net>
704f3f0262cSandi * @version     1.0.0
705f3f0262cSandi */
706f31d5b73Sandifunction filesize_h($size, $dec = 1){
707f3f0262cSandi  $sizes = array('B', 'KB', 'MB', 'GB');
708f3f0262cSandi  $count = count($sizes);
709f3f0262cSandi  $i = 0;
710f3f0262cSandi
711f3f0262cSandi  while ($size >= 1024 && ($i < $count - 1)) {
712f3f0262cSandi    $size /= 1024;
713f3f0262cSandi    $i++;
714f3f0262cSandi  }
715f3f0262cSandi
716f3f0262cSandi  return round($size, $dec) . ' ' . $sizes[$i];
717f3f0262cSandi}
718f3f0262cSandi
71915fae107Sandi/**
72015fae107Sandi * Run a few sanity checks
72115fae107Sandi *
72215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
72315fae107Sandi */
724f31d5b73Sandifunction getVersion(){
725f31d5b73Sandi  //import version string
726f31d5b73Sandi  if(@file_exists('VERSION')){
727f31d5b73Sandi    //official release
728f31d5b73Sandi    return 'Release '.io_readfile('VERSION');
729f31d5b73Sandi  }elseif(is_dir('_darcs')){
730f31d5b73Sandi    //darcs checkout
731f31d5b73Sandi    $inv = file('_darcs/inventory');
732f31d5b73Sandi    $inv = preg_grep('#andi@splitbrain\.org\*\*\d{14}#',$inv);
733f31d5b73Sandi    $cur = array_pop($inv);
734f31d5b73Sandi    preg_match('#\*\*(\d{4})(\d{2})(\d{2})#',$cur,$matches);
735f31d5b73Sandi    return 'Darcs '.$matches[1].'-'.$matches[2].'-'.$matches[3];
736f31d5b73Sandi  }else{
737f31d5b73Sandi    return 'snapshot?';
738f31d5b73Sandi  }
739f31d5b73Sandi}
740f31d5b73Sandi
741f31d5b73Sandi/**
742f31d5b73Sandi * Run a few sanity checks
743f31d5b73Sandi *
744f31d5b73Sandi * @author Andreas Gohr <andi@splitbrain.org>
745f31d5b73Sandi */
746f3f0262cSandifunction check(){
747f3f0262cSandi  global $conf;
748f3f0262cSandi  global $INFO;
749f3f0262cSandi
750f31d5b73Sandi  msg('DokuWiki version: '.getVersion(),1);
751f31d5b73Sandi
75249022a38Sandi  if(version_compare(phpversion(),'4.3.0','<')){
75349022a38Sandi    msg('Your PHP version is too old ('.phpversion().' vs. 4.3.+ recommended)',-1);
75449022a38Sandi  }elseif(version_compare(phpversion(),'4.3.10','<')){
75549022a38Sandi    msg('Consider upgrading PHP to 4.3.10 or higher for security reasons (your version: '.phpversion().')',0);
75649022a38Sandi  }else{
75749022a38Sandi    msg('PHP version '.phpversion(),1);
75849022a38Sandi  }
75949022a38Sandi
760f3f0262cSandi  if(is_writable($conf['changelog'])){
761f3f0262cSandi    msg('Changelog is writable',1);
762f3f0262cSandi  }else{
763f3f0262cSandi    msg('Changelog is not writable',-1);
764f3f0262cSandi  }
765f3f0262cSandi
766f3f0262cSandi  if(is_writable($conf['datadir'])){
767f3f0262cSandi    msg('Datadir is writable',1);
768f3f0262cSandi  }else{
769f3f0262cSandi    msg('Datadir is not writable',-1);
770f3f0262cSandi  }
771f3f0262cSandi
772f3f0262cSandi  if(is_writable($conf['olddir'])){
773f3f0262cSandi    msg('Attic is writable',1);
774f3f0262cSandi  }else{
775f3f0262cSandi    msg('Attic is not writable',-1);
776f3f0262cSandi  }
777f3f0262cSandi
778f3f0262cSandi  if(is_writable($conf['mediadir'])){
779f3f0262cSandi    msg('Mediadir is writable',1);
780f3f0262cSandi  }else{
781f3f0262cSandi    msg('Mediadir is not writable',-1);
782f3f0262cSandi  }
783f3f0262cSandi
784f3f0262cSandi  if(is_writable('conf/users.auth')){
785f3f0262cSandi    msg('conf/users.auth is writable',1);
786f3f0262cSandi  }else{
787f3f0262cSandi    msg('conf/users.auth is not writable',0);
788f3f0262cSandi  }
78993a9e835Sandi
79093a9e835Sandi  if(function_exists('mb_strpos')){
79193a9e835Sandi    if(defined('UTF8_NOMBSTRING')){
79293a9e835Sandi      msg('mb_string extension is available but will not be used',0);
79393a9e835Sandi    }else{
79493a9e835Sandi      msg('mb_string extension is available and will be used',1);
79593a9e835Sandi    }
79693a9e835Sandi  }else{
79793a9e835Sandi    msg('mb_string extension not available - PHP only replacements will be used',0);
79893a9e835Sandi  }
799f3f0262cSandi
800f3f0262cSandi  msg('Your current permission for this page is '.$INFO['perm'],0);
801f3f0262cSandi
802f3f0262cSandi  if(is_writable($INFO['filepath'])){
803f3f0262cSandi    msg('The current page is writable by the webserver',0);
804f3f0262cSandi  }else{
805f3f0262cSandi    msg('The current page is not writable by the webserver',0);
806f3f0262cSandi  }
807f3f0262cSandi
808f3f0262cSandi  if($INFO['writable']){
809f3f0262cSandi    msg('The current page is writable by you',0);
810f3f0262cSandi  }else{
811f3f0262cSandi    msg('The current page is not writable you',0);
812f3f0262cSandi  }
813f3f0262cSandi}
814340756e4Sandi
815340756e4Sandi
816340756e4Sandi//Setup VIM: ex: et ts=2 enc=utf-8 :
817