xref: /dokuwiki/inc/common.php (revision 9afe4dbf2e0907d016556ca89d08115ab99ab903)
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');
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
122a77f5846Sjan  $file = wikiFN($ID);
123a77f5846Sjan  if($ACT != 'show' || !@file_exists($file)){
124f3f0262cSandi    $_SESSION[$conf['title']]['bc'] = $crumbs;
125f3f0262cSandi    return $crumbs;
126f3f0262cSandi  }
127a77f5846Sjan
128a77f5846Sjan  // page names
129a77f5846Sjan  $name = noNS($ID);
130a77f5846Sjan  if ($conf['useheading']) {
131a77f5846Sjan    // get page title
132a77f5846Sjan    $title = getFirstHeading(io_readFile($file));
133a77f5846Sjan    if ($title) {
134a77f5846Sjan      $name = $title;
135a77f5846Sjan    }
136a77f5846Sjan  }
137a77f5846Sjan
138f3f0262cSandi  //remove ID from array
139a77f5846Sjan  if (isset($crumbs[$ID])) {
140a77f5846Sjan    unset($crumbs[$ID]);
141f3f0262cSandi  }
142f3f0262cSandi
143f3f0262cSandi  //add to array
144a77f5846Sjan  $crumbs[$ID] = $name;
145f3f0262cSandi  //reduce size
146f3f0262cSandi  while(count($crumbs) > $conf['breadcrumbs']){
147f3f0262cSandi    array_shift($crumbs);
148f3f0262cSandi  }
149f3f0262cSandi  //save to session
150f3f0262cSandi  $_SESSION[$conf['title']]['bc'] = $crumbs;
151f3f0262cSandi  return $crumbs;
152f3f0262cSandi}
153f3f0262cSandi
154f3f0262cSandi/**
15515fae107Sandi * Filter for page IDs
15615fae107Sandi *
157f3f0262cSandi * This is run on a ID before it is outputted somewhere
158f3f0262cSandi * currently used to replace the colon with something else
159f3f0262cSandi * on Windows systems and to have proper URL encoding
16015fae107Sandi *
16149c713a3Sandi * Urlencoding is ommitted when the second parameter is false
16249c713a3Sandi *
16315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
164f3f0262cSandi */
16549c713a3Sandifunction idfilter($id,$ue=true){
166f3f0262cSandi  global $conf;
167f3f0262cSandi  if ($conf['useslash'] && $conf['userewrite']){
168f3f0262cSandi    $id = strtr($id,':','/');
169f3f0262cSandi  }elseif (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' &&
170f3f0262cSandi      $conf['userewrite']) {
171f3f0262cSandi    $id = strtr($id,':',';');
172f3f0262cSandi  }
17349c713a3Sandi  if($ue){
174f3f0262cSandi    $id = urlencode($id);
175f3f0262cSandi    $id = str_replace('%3A',':',$id); //keep as colon
176f3f0262cSandi    $id = str_replace('%2F','/',$id); //keep as slash
17749c713a3Sandi  }
178f3f0262cSandi  return $id;
179f3f0262cSandi}
180f3f0262cSandi
181f3f0262cSandi/**
182ed7b5f09Sandi * This builds a link to a wikipage
18315fae107Sandi *
18415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
185f3f0262cSandi */
186ed7b5f09Sandifunction wl($id='',$more='',$abs=false){
187f3f0262cSandi  global $conf;
188f3f0262cSandi  $more = str_replace(',','&amp;',$more);
189f3f0262cSandi
190f3f0262cSandi  $id    = idfilter($id);
191ed7b5f09Sandi  if($abs){
192ed7b5f09Sandi    $xlink = DOKU_URL;
193ed7b5f09Sandi  }else{
194ed7b5f09Sandi    $xlink = DOKU_BASE;
195ed7b5f09Sandi  }
196f3f0262cSandi
197f3f0262cSandi  if(!$conf['userewrite']){
198ed7b5f09Sandi    $xlink .= DOKU_SCRIPT.'?id='.$id;
199f3f0262cSandi    if($more) $xlink .= '&amp;'.$more;
200f3f0262cSandi  }else{
201f3f0262cSandi    $xlink .= $id;
202f3f0262cSandi    if($more) $xlink .= '?'.$more;
203f3f0262cSandi  }
204f3f0262cSandi
205f3f0262cSandi  return $xlink;
206f3f0262cSandi}
207f3f0262cSandi
208f3f0262cSandi/**
209f3f0262cSandi * Just builds a link to a script
21015fae107Sandi *
211ed7b5f09Sandi * @todo   maybe obsolete
21215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
213f3f0262cSandi */
214f3f0262cSandifunction script($script='doku.php'){
215ed7b5f09Sandi#  $link = getBaseURL();
216ed7b5f09Sandi#  $link .= $script;
217ed7b5f09Sandi#  return $link;
218ed7b5f09Sandi  return DOKU_BASE.DOKU_SCRIPT;
219f3f0262cSandi}
220f3f0262cSandi
221f3f0262cSandi/**
222f3f0262cSandi * Return namespacepart of a wiki ID
22315fae107Sandi *
22415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
225f3f0262cSandi */
226f3f0262cSandifunction getNS($id){
227f3f0262cSandi if(strpos($id,':')!==false){
228f3f0262cSandi   return substr($id,0,strrpos($id,':'));
229f3f0262cSandi }
230f3f0262cSandi return false;
231f3f0262cSandi}
232f3f0262cSandi
233f3f0262cSandi/**
23415fae107Sandi * Returns the ID without the namespace
23515fae107Sandi *
23615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
237f3f0262cSandi */
238f3f0262cSandifunction noNS($id){
239f3f0262cSandi  return preg_replace('/.*:/','',$id);
240f3f0262cSandi}
241f3f0262cSandi
242f3f0262cSandi/**
24315fae107Sandi * Spamcheck against wordlist
24415fae107Sandi *
245f3f0262cSandi * Checks the wikitext against a list of blocked expressions
246f3f0262cSandi * returns true if the text contains any bad words
24715fae107Sandi *
24815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
249f3f0262cSandi */
250f3f0262cSandifunction checkwordblock(){
251f3f0262cSandi  global $TEXT;
252f3f0262cSandi  global $conf;
253f3f0262cSandi
254f3f0262cSandi  if(!$conf['usewordblock']) return false;
255f3f0262cSandi
256703f6fdeSandi  $blockfile = file('conf/wordblock.conf');
2573e2965d7Sandi  //how many lines to read at once (to work around some PCRE limits)
2583e2965d7Sandi  if(version_compare(phpversion(),'4.3.0','<')){
2593e2965d7Sandi    //old versions of PCRE define a maximum of parenthesises even if no
2603e2965d7Sandi    //backreferences are used - the maximum is 99
2613e2965d7Sandi    //this is very bad performancewise and may even be too high still
2623e2965d7Sandi    $chunksize = 40;
2633e2965d7Sandi  }else{
264703f6fdeSandi    //read file in chunks of 600 - this should work around the
2653e2965d7Sandi    //MAX_PATTERN_SIZE in modern PCRE
2663e2965d7Sandi    $chunksize = 600;
2673e2965d7Sandi  }
2683e2965d7Sandi  while($blocks = array_splice($blockfile,0,$chunksize)){
269f3f0262cSandi    $re = array();
270f3f0262cSandi    #build regexp from blocks
271f3f0262cSandi    foreach($blocks as $block){
272f3f0262cSandi      $block = preg_replace('/#.*$/','',$block);
273f3f0262cSandi      $block = trim($block);
274f3f0262cSandi      if(empty($block)) continue;
275f3f0262cSandi      $re[]  = $block;
276f3f0262cSandi    }
277f3f0262cSandi    if(preg_match('#('.join('|',$re).')#si',$TEXT)) return true;
278703f6fdeSandi  }
279f3f0262cSandi  return false;
280f3f0262cSandi}
281f3f0262cSandi
282f3f0262cSandi/**
28315fae107Sandi * Return the IP of the client
28415fae107Sandi *
28515fae107Sandi * Honours X-Forwarded-For Proxy Headers
28615fae107Sandi *
28715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
288f3f0262cSandi */
289f3f0262cSandifunction clientIP(){
290f3f0262cSandi  $my = $_SERVER['REMOTE_ADDR'];
291f3f0262cSandi  if($_SERVER['HTTP_X_FORWARDED_FOR']){
292f3f0262cSandi    $my .= ' ('.$_SERVER['HTTP_X_FORWARDED_FOR'].')';
293f3f0262cSandi  }
294f3f0262cSandi  return $my;
295f3f0262cSandi}
296f3f0262cSandi
297f3f0262cSandi/**
29815fae107Sandi * Checks if a given page is currently locked.
29915fae107Sandi *
300f3f0262cSandi * removes stale lockfiles
30115fae107Sandi *
30215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
303f3f0262cSandi */
304f3f0262cSandifunction checklock($id){
305f3f0262cSandi  global $conf;
306f3f0262cSandi  $lock = wikiFN($id).'.lock';
307f3f0262cSandi
308f3f0262cSandi  //no lockfile
309f3f0262cSandi  if(!@file_exists($lock)) return false;
310f3f0262cSandi
311f3f0262cSandi  //lockfile expired
312f3f0262cSandi  if((time() - filemtime($lock)) > $conf['locktime']){
313f3f0262cSandi    unlink($lock);
314f3f0262cSandi    return false;
315f3f0262cSandi  }
316f3f0262cSandi
317f3f0262cSandi  //my own lock
318f3f0262cSandi  $ip = io_readFile($lock);
319f3f0262cSandi  if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
320f3f0262cSandi    return false;
321f3f0262cSandi  }
322f3f0262cSandi
323f3f0262cSandi  return $ip;
324f3f0262cSandi}
325f3f0262cSandi
326f3f0262cSandi/**
32715fae107Sandi * Lock a page for editing
32815fae107Sandi *
32915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
330f3f0262cSandi */
331f3f0262cSandifunction lock($id){
332f3f0262cSandi  $lock = wikiFN($id).'.lock';
333f3f0262cSandi  if($_SERVER['REMOTE_USER']){
334f3f0262cSandi    io_saveFile($lock,$_SERVER['REMOTE_USER']);
335f3f0262cSandi  }else{
336f3f0262cSandi    io_saveFile($lock,clientIP());
337f3f0262cSandi  }
338f3f0262cSandi}
339f3f0262cSandi
340f3f0262cSandi/**
34115fae107Sandi * Unlock a page if it was locked by the user
342f3f0262cSandi *
34315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
34415fae107Sandi * @return bool true if a lock was removed
345f3f0262cSandi */
346f3f0262cSandifunction unlock($id){
347f3f0262cSandi  $lock = wikiFN($id).'.lock';
348f3f0262cSandi  if(@file_exists($lock)){
349f3f0262cSandi    $ip = io_readFile($lock);
350f3f0262cSandi    if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
351f3f0262cSandi      @unlink($lock);
352f3f0262cSandi      return true;
353f3f0262cSandi    }
354f3f0262cSandi  }
355f3f0262cSandi  return false;
356f3f0262cSandi}
357f3f0262cSandi
358f3f0262cSandi/**
35915fae107Sandi * Remove unwanted chars from ID
36015fae107Sandi *
361f3f0262cSandi * Cleans a given ID to only use allowed characters. Accented characters are
362f3f0262cSandi * converted to unaccented ones
36315fae107Sandi *
36415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
365f3f0262cSandi */
366f3f0262cSandifunction cleanID($id){
367f3f0262cSandi  global $conf;
368f3f0262cSandi  global $lang;
369f3f0262cSandi  $id = trim($id);
370c41c03f3Sandi  $id = utf8_strtolower($id);
371f3f0262cSandi
372f3f0262cSandi  //alternative namespace seperator
373f3f0262cSandi  $id = strtr($id,';',':');
3743021e063Sandi  if($conf['useslash']){
3753021e063Sandi    $id = strtr($id,'/',':');
3763021e063Sandi  }else{
3773021e063Sandi    $id = strtr($id,'/','_');
3783021e063Sandi  }
379f3f0262cSandi
3808b709e9dSandi  if($conf['deaccent']) $id = utf8_deaccent($id,-1);
381c41c03f3Sandi
382099ada41Sandi  //remove specials
383099ada41Sandi  $id = utf8_stripspecials($id,'_','_:.-');
384c41c03f3Sandi
385c41c03f3Sandi  //clean up
386f3f0262cSandi  $id = preg_replace('#__#','_',$id);
387f3f0262cSandi  $id = preg_replace('#:+#',':',$id);
388f3f0262cSandi  $id = trim($id,':._-');
389f3f0262cSandi  $id = preg_replace('#:[:\._\-]+#',':',$id);
390f3f0262cSandi
391f3f0262cSandi  return($id);
392f3f0262cSandi}
393f3f0262cSandi
394f3f0262cSandi/**
395f3f0262cSandi * returns the full path to the datafile specified by ID and
396f3f0262cSandi * optional revision
39715fae107Sandi *
39849c713a3Sandi * The filename is URL encoded to protect Unicode chars
39949c713a3Sandi *
40015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
401f3f0262cSandi */
402f3f0262cSandifunction wikiFN($id,$rev=''){
403f3f0262cSandi  global $conf;
404f3f0262cSandi  $id = cleanID($id);
405f3f0262cSandi  $id = str_replace(':','/',$id);
406f3f0262cSandi  if(empty($rev)){
407eca0e1f9Sandi    $fn = $conf['datadir'].'/'.utf8_encodeFN($id).'.txt';
408f3f0262cSandi  }else{
409eca0e1f9Sandi    $fn = $conf['olddir'].'/'.utf8_encodeFN($id).'.'.$rev.'.txt';
41049c713a3Sandi    if($conf['usegzip'] && !@file_exists($fn)){
41149c713a3Sandi      //return gzip if enabled and plaintext doesn't exist
41249c713a3Sandi      $fn .= '.gz';
41349c713a3Sandi    }
41449c713a3Sandi  }
415f3f0262cSandi  return $fn;
416f3f0262cSandi}
417f3f0262cSandi
418f3f0262cSandi/**
419f3f0262cSandi * Returns the full filepath to a localized textfile if local
420f3f0262cSandi * version isn't found the english one is returned
42115fae107Sandi *
42215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
423f3f0262cSandi */
424f3f0262cSandifunction localeFN($id){
425f3f0262cSandi  global $conf;
426f3f0262cSandi  $file = './lang/'.$conf['lang'].'/'.$id.'.txt';
427f3f0262cSandi  if(!@file_exists($file)){
428f3f0262cSandi    //fall back to english
429f3f0262cSandi    $file = './lang/en/'.$id.'.txt';
430f3f0262cSandi  }
431f3f0262cSandi  return cleanText($file);
432f3f0262cSandi}
433f3f0262cSandi
434f3f0262cSandi/**
435f3f0262cSandi * convert line ending to unix format
436f3f0262cSandi *
43715fae107Sandi * @see    formText() for 2crlf conversion
43815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
439f3f0262cSandi */
440f3f0262cSandifunction cleanText($text){
441f3f0262cSandi  $text = preg_replace("/(\015\012)|(\015)/","\012",$text);
442f3f0262cSandi  return $text;
443f3f0262cSandi}
444f3f0262cSandi
445f3f0262cSandi/**
446f3f0262cSandi * Prepares text for print in Webforms by encoding special chars.
447f3f0262cSandi * It also converts line endings to Windows format which is
448f3f0262cSandi * pseudo standard for webforms.
449f3f0262cSandi *
45015fae107Sandi * @see    cleanText() for 2unix conversion
45115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
452f3f0262cSandi */
453f3f0262cSandifunction formText($text){
454f3f0262cSandi  $text = preg_replace("/\012/","\015\012",$text);
455f3f0262cSandi  return htmlspecialchars($text);
456f3f0262cSandi}
457f3f0262cSandi
458f3f0262cSandi/**
45915fae107Sandi * Returns the specified local text in parsed format
46015fae107Sandi *
46115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
462f3f0262cSandi */
463f3f0262cSandifunction parsedLocale($id){
464f3f0262cSandi  //disable section editing
465f3f0262cSandi  global $parser;
466f3f0262cSandi  $se = $parser['secedit'];
467f3f0262cSandi  $parser['secedit'] = false;
468f3f0262cSandi  //fetch parsed locale
469f3f0262cSandi  $html = io_cacheParse(localeFN($id));
470f3f0262cSandi  //reset section editing
471f3f0262cSandi  $parser['secedit'] = $se;
472f3f0262cSandi  return $html;
473f3f0262cSandi}
474f3f0262cSandi
475f3f0262cSandi/**
47615fae107Sandi * Returns the specified local text in raw format
47715fae107Sandi *
47815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
479f3f0262cSandi */
480f3f0262cSandifunction rawLocale($id){
481f3f0262cSandi  return io_readFile(localeFN($id));
482f3f0262cSandi}
483f3f0262cSandi
484f3f0262cSandi
485f3f0262cSandi/**
48615fae107Sandi * Returns the parsed Wikitext for the given id and revision.
48715fae107Sandi *
48815fae107Sandi * If $excuse is true an explanation is returned if the file
48915fae107Sandi * wasn't found
49015fae107Sandi *
49115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
492f3f0262cSandi */
493f3f0262cSandifunction parsedWiki($id,$rev='',$excuse=true){
494f3f0262cSandi  $file = wikiFN($id,$rev);
495f3f0262cSandi  $ret  = '';
496f3f0262cSandi
497f3f0262cSandi  //ensure $id is in global $ID (needed for parsing)
498f3f0262cSandi  global $ID;
499f3f0262cSandi  $ID = $id;
500f3f0262cSandi
501f3f0262cSandi  if($rev){
502f3f0262cSandi    if(@file_exists($file)){
503f3f0262cSandi      $ret = parse(io_readFile($file));
504f3f0262cSandi    }elseif($excuse){
505f3f0262cSandi      $ret = parsedLocale('norev');
506f3f0262cSandi    }
507f3f0262cSandi  }else{
508f3f0262cSandi    if(@file_exists($file)){
509f3f0262cSandi      $ret = io_cacheParse($file);
510f3f0262cSandi    }elseif($excuse){
511f3f0262cSandi      $ret = parsedLocale('newpage');
512f3f0262cSandi    }
513f3f0262cSandi  }
514f3f0262cSandi  return $ret;
515f3f0262cSandi}
516f3f0262cSandi
517f3f0262cSandi/**
518f3f0262cSandi * Returns the raw WikiText
51915fae107Sandi *
52015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
521f3f0262cSandi */
522f3f0262cSandifunction rawWiki($id,$rev=''){
523f3f0262cSandi  return io_readFile(wikiFN($id,$rev));
524f3f0262cSandi}
525f3f0262cSandi
526f3f0262cSandi/**
52715fae107Sandi * Returns the raw Wiki Text in three slices.
52815fae107Sandi *
52915fae107Sandi * The range parameter needs to have the form "from-to"
53015fae107Sandi * and gives the range of the section.
531f3f0262cSandi * The returned order is prefix, section and suffix.
53215fae107Sandi *
53315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
534f3f0262cSandi */
535f3f0262cSandifunction rawWikiSlices($range,$id,$rev=''){
536f3f0262cSandi  list($from,$to) = split('-',$range,2);
537f3f0262cSandi  $text = io_readFile(wikiFN($id,$rev));
538f3f0262cSandi  $text = split("\n",$text);
539f3f0262cSandi  if(!$from) $from = 0;
540f3f0262cSandi  if(!$to)   $to   = count($text);
541f3f0262cSandi
542f3f0262cSandi  $slices[0] = join("\n",array_slice($text,0,$from));
543f3f0262cSandi  $slices[1] = join("\n",array_slice($text,$from,$to + 1  - $from));
544f3f0262cSandi  $slices[2] = join("\n",array_slice($text,$to+1));
545f3f0262cSandi
546f3f0262cSandi  return $slices;
547f3f0262cSandi}
548f3f0262cSandi
549f3f0262cSandi/**
55015fae107Sandi * Joins wiki text slices
55115fae107Sandi *
552f3f0262cSandi * function to join the text slices with correct lineendings again.
553f3f0262cSandi * When the pretty parameter is set to true it adds additional empty
554f3f0262cSandi * lines between sections if needed (used on saving).
55515fae107Sandi *
55615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
557f3f0262cSandi */
558f3f0262cSandifunction con($pre,$text,$suf,$pretty=false){
559f3f0262cSandi
560f3f0262cSandi  if($pretty){
561f3f0262cSandi    if($pre && substr($pre,-1) != "\n") $pre .= "\n";
562f3f0262cSandi    if($suf && substr($text,-1) != "\n") $text .= "\n";
563f3f0262cSandi  }
564f3f0262cSandi
565f3f0262cSandi  if($pre) $pre .= "\n";
566f3f0262cSandi  if($suf) $text .= "\n";
567f3f0262cSandi  return $pre.$text.$suf;
568f3f0262cSandi}
569f3f0262cSandi
570f3f0262cSandi/**
57115fae107Sandi * print debug messages
57215fae107Sandi *
573f3f0262cSandi * little function to print the content of a var
57415fae107Sandi *
57515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
576f3f0262cSandi */
577f3f0262cSandifunction dbg($msg,$hidden=false){
578f3f0262cSandi  (!$hidden) ? print '<pre class="dbg">' : print "<!--\n";
579f3f0262cSandi  print_r($msg);
580f3f0262cSandi  (!$hidden) ? print '</pre>' : print "\n-->";
581f3f0262cSandi}
582f3f0262cSandi
583f3f0262cSandi/**
584f3f0262cSandi * Add's an entry to the changelog
58515fae107Sandi *
58615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
587f3f0262cSandi */
588652610a2Sandifunction addLogEntry($date,$id,$summary=""){
589f3f0262cSandi  global $conf;
590c1049928Sandi  $id     = cleanID($id);//FIXME not needed anymore?
591c1049928Sandi
592c1049928Sandi  if(!@is_writable($conf['changelog'])){
593c1049928Sandi    msg($conf['changelog'].' is not writable!',-1);
594c1049928Sandi    return;
595c1049928Sandi  }
596c1049928Sandi
597652610a2Sandi  if(!$date) $date = time(); //use current time if none supplied
598f3f0262cSandi  $remote = $_SERVER['REMOTE_ADDR'];
599f3f0262cSandi  $user   = $_SERVER['REMOTE_USER'];
600f3f0262cSandi
601f3f0262cSandi  $logline = join("\t",array($date,$remote,$id,$user,$summary))."\n";
602f3f0262cSandi
603c1049928Sandi  //FIXME: use adjusted io_saveFile instead
604f3f0262cSandi  $fh = fopen($conf['changelog'],'a');
605f3f0262cSandi  if($fh){
606f3f0262cSandi    fwrite($fh,$logline);
607f3f0262cSandi    fclose($fh);
608f3f0262cSandi  }
609f3f0262cSandi}
610f3f0262cSandi
611f3f0262cSandi/**
612f3f0262cSandi * returns an array of recently changed files using the
613f3f0262cSandi * changelog
61415fae107Sandi *
61515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
616f3f0262cSandi */
617f3f0262cSandifunction getRecents($num=0,$incdel=false){
618f3f0262cSandi  global $conf;
619f3f0262cSandi  $recent = array();
620f3f0262cSandi  if(!$num) $num = $conf['recent'];
621f3f0262cSandi
622c1049928Sandi  if(!@is_readable($conf['changelog'])){
623c1049928Sandi    msg($conf['changelog'].' is not readable',-1);
624c1049928Sandi    return $recent;
625c1049928Sandi  }
626c1049928Sandi
627f3f0262cSandi  $loglines = file($conf['changelog']);
628f3f0262cSandi  rsort($loglines); //reverse sort on timestamp
629f3f0262cSandi
630f3f0262cSandi  foreach ($loglines as $line){
631f3f0262cSandi    $line = rtrim($line);        //remove newline
632f3f0262cSandi    if(empty($line)) continue;   //skip empty lines
633f3f0262cSandi    $info = split("\t",$line);   //split into parts
634f3f0262cSandi    //add id if not in yet and file still exists and is allowed to read
635f3f0262cSandi    if(!$recent[$info[2]] &&
636f3f0262cSandi       (@file_exists(wikiFN($info[2])) || $incdel) &&
637f3f0262cSandi       (auth_quickaclcheck($info[2]) >= AUTH_READ)
638f3f0262cSandi      ){
639f3f0262cSandi      $recent[$info[2]]['date'] = $info[0];
640f3f0262cSandi      $recent[$info[2]]['ip']   = $info[1];
641f3f0262cSandi      $recent[$info[2]]['user'] = $info[3];
642f3f0262cSandi      $recent[$info[2]]['sum']  = $info[4];
643f3f0262cSandi      $recent[$info[2]]['del']  = !@file_exists(wikiFN($info[2]));
644f3f0262cSandi    }
645f3f0262cSandi    if(count($recent) >= $num){
646f3f0262cSandi      break; //finish if enough items found
647f3f0262cSandi    }
648f3f0262cSandi  }
649f3f0262cSandi  return $recent;
650f3f0262cSandi}
651f3f0262cSandi
652f3f0262cSandi/**
653652610a2Sandi * gets additonal informations for a certain pagerevison
654652610a2Sandi * from the changelog
655652610a2Sandi *
656652610a2Sandi * @author Andreas Gohr <andi@splitbrain.org>
657652610a2Sandi */
658652610a2Sandifunction getRevisionInfo($id,$rev){
659652610a2Sandi  global $conf;
660258641c6Sandi
661258641c6Sandi  if(!$rev) return(null);
662258641c6Sandi
663c1049928Sandi  $info = array();
664c1049928Sandi  if(!@is_readable($conf['changelog'])){
665c1049928Sandi    msg($conf['changelog'].' is not readable',-1);
666c1049928Sandi    return $recent;
667c1049928Sandi  }
668652610a2Sandi  $loglines = file($conf['changelog']);
669652610a2Sandi  $loglines = preg_grep("/$rev\t\d+\.\d+\.\d+\.\d+\t$id\t/",$loglines);
670652610a2Sandi  rsort($loglines); //reverse sort on timestamp (shouldn't be needed)
671652610a2Sandi  $line = split("\t",$loglines[0]);
672652610a2Sandi  $info['date'] = $line[0];
673652610a2Sandi  $info['ip']   = $line[1];
674652610a2Sandi  $info['user'] = $line[3];
675652610a2Sandi  $info['sum']   = $line[4];
676652610a2Sandi  return $info;
677652610a2Sandi}
678652610a2Sandi
679652610a2Sandi/**
680f3f0262cSandi * Saves a wikitext by calling io_saveFile
68115fae107Sandi *
68215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
683f3f0262cSandi */
684f3f0262cSandifunction saveWikiText($id,$text,$summary){
685f3f0262cSandi  global $conf;
686f3f0262cSandi  global $lang;
687f3f0262cSandi  umask($conf['umask']);
688f3f0262cSandi  // ignore if no changes were made
689f3f0262cSandi  if($text == rawWiki($id,'')){
690f3f0262cSandi    return;
691f3f0262cSandi  }
692f3f0262cSandi
693f3f0262cSandi  $file = wikiFN($id);
694f3f0262cSandi  $old  = saveOldRevision($id);
695f3f0262cSandi
696f3f0262cSandi  if (empty($text)){
697f3f0262cSandi    // remove empty files
698f3f0262cSandi    @unlink($file);
699f3f0262cSandi    $del = true;
7003ce054b3Sandi    //autoset summary on deletion
7013ce054b3Sandi    if(empty($summary)) $summary = $lang['deleted'];
70253d6ccfeSandi    //remove empty namespaces
70353d6ccfeSandi    io_sweepNS($id);
704f3f0262cSandi  }else{
705f3f0262cSandi    // save file (datadir is created in io_saveFile)
706f3f0262cSandi    io_saveFile($file,$text);
707f3f0262cSandi    $del = false;
708f3f0262cSandi  }
709f3f0262cSandi
710652610a2Sandi  addLogEntry(@filemtime($file),$id,$summary);
711f3f0262cSandi  notify($id,$old,$summary);
712f3f0262cSandi
713f3f0262cSandi  //purge cache on add by updating the purgefile
714f3f0262cSandi  if($conf['purgeonadd'] && (!$old || $del)){
715*9afe4dbfSjan    io_saveFile($conf['datadir'].'/_cache/purgefile',time());
716f3f0262cSandi  }
717f3f0262cSandi}
718f3f0262cSandi
719f3f0262cSandi/**
720f3f0262cSandi * moves the current version to the attic and returns its
721f3f0262cSandi * revision date
72215fae107Sandi *
72315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
724f3f0262cSandi */
725f3f0262cSandifunction saveOldRevision($id){
726f3f0262cSandi	global $conf;
727f3f0262cSandi  umask($conf['umask']);
728f3f0262cSandi  $oldf = wikiFN($id);
729f3f0262cSandi  if(!@file_exists($oldf)) return '';
730f3f0262cSandi  $date = filemtime($oldf);
731f3f0262cSandi  $newf = wikiFN($id,$date);
732f3f0262cSandi  if(substr($newf,-3)=='.gz'){
733f3f0262cSandi    io_saveFile($newf,rawWiki($id));
734f3f0262cSandi  }else{
735f3f0262cSandi    io_makeFileDir($newf);
736f3f0262cSandi    copy($oldf, $newf);
737f3f0262cSandi  }
738f3f0262cSandi  return $date;
739f3f0262cSandi}
740f3f0262cSandi
741f3f0262cSandi/**
742f3f0262cSandi * Sends a notify mail to the wikiadmin when a page was
743f3f0262cSandi * changed
74415fae107Sandi *
74515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
746f3f0262cSandi */
747f3f0262cSandifunction notify($id,$rev="",$summary=""){
748f3f0262cSandi  global $lang;
749f3f0262cSandi  global $conf;
750f3f0262cSandi  $hdrs ='';
751f3f0262cSandi  if(empty($conf['notify'])) return; //notify enabled?
752f3f0262cSandi
753f3f0262cSandi  $text = rawLocale('mailtext');
754f3f0262cSandi  $text = str_replace('@DATE@',date($conf['dformat']),$text);
755f3f0262cSandi  $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text);
756f3f0262cSandi  $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text);
757f3f0262cSandi  $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text);
758ed7b5f09Sandi  $text = str_replace('@NEWPAGE@',wl($id,'',true),$text);
759ed7b5f09Sandi  $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text);
760f3f0262cSandi  $text = str_replace('@SUMMARY@',$summary,$text);
7617a82afdcSandi  $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text);
762f3f0262cSandi
763f3f0262cSandi  if($rev){
764f3f0262cSandi    $subject = $lang['mail_changed'].' '.$id;
765ed7b5f09Sandi    $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true),$text);
766f3f0262cSandi    require_once("inc/DifferenceEngine.php");
767f3f0262cSandi    $df  = new Diff(split("\n",rawWiki($id,$rev)),
768f3f0262cSandi                    split("\n",rawWiki($id)));
769f3f0262cSandi    $dformat = new UnifiedDiffFormatter();
770f3f0262cSandi    $diff    = $dformat->format($df);
771f3f0262cSandi  }else{
772f3f0262cSandi    $subject=$lang['mail_newpage'].' '.$id;
773f3f0262cSandi    $text = str_replace('@OLDPAGE@','none',$text);
774f3f0262cSandi    $diff = rawWiki($id);
775f3f0262cSandi  }
776f3f0262cSandi  $text = str_replace('@DIFF@',$diff,$text);
777f3f0262cSandi
77844f669e9Sandi  mail_send($conf['notify'],$subject,$text,$conf['mailfrom']);
779f3f0262cSandi}
780f3f0262cSandi
78115fae107Sandi/**
78215fae107Sandi * Return a list of available page revisons
78315fae107Sandi *
78415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
78515fae107Sandi */
786f3f0262cSandifunction getRevisions($id){
787f3f0262cSandi  $revd = dirname(wikiFN($id,'foo'));
788f3f0262cSandi  $revs = array();
789f3f0262cSandi  $clid = cleanID($id);
790f3f0262cSandi  if(strrpos($clid,':')) $clid = substr($clid,strrpos($clid,':')+1); //remove path
791f3f0262cSandi
792f3f0262cSandi  if (is_dir($revd) && $dh = opendir($revd)) {
793f3f0262cSandi    while (($file = readdir($dh)) !== false) {
794f3f0262cSandi      if (is_dir($revd.'/'.$file)) continue;
795f3f0262cSandi      if (preg_match('/^'.$clid.'\.(\d+)\.txt(\.gz)?$/',$file,$match)){
796f3f0262cSandi        $revs[]=$match[1];
797f3f0262cSandi      }
798f3f0262cSandi    }
799f3f0262cSandi    closedir($dh);
800f3f0262cSandi  }
801f3f0262cSandi  rsort($revs);
802f3f0262cSandi  return $revs;
803f3f0262cSandi}
804f3f0262cSandi
805f3f0262cSandi/**
806f3f0262cSandi * downloads a file from the net and saves it to the given location
80715fae107Sandi *
80815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
809f3f0262cSandi */
810f3f0262cSandifunction download($url,$file){
811f3f0262cSandi  $fp = @fopen($url,"rb");
812f3f0262cSandi  if(!$fp) return false;
813f3f0262cSandi
814f3f0262cSandi  while(!feof($fp)){
815f3f0262cSandi    $cont.= fread($fp,1024);
816f3f0262cSandi  }
817f3f0262cSandi  fclose($fp);
818f3f0262cSandi
819f3f0262cSandi  $fp2 = @fopen($file,"w");
820f3f0262cSandi  if(!$fp2) return false;
821f3f0262cSandi  fwrite($fp2,$cont);
822f3f0262cSandi  fclose($fp2);
823f3f0262cSandi  return true;
824f3f0262cSandi}
825f3f0262cSandi
826f3f0262cSandi/**
827f3f0262cSandi * extracts the query from a google referer
82815fae107Sandi *
8296b13307fSandi * @todo   should be more generic and support yahoo et al
83015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
831f3f0262cSandi */
832f3f0262cSandifunction getGoogleQuery(){
833f3f0262cSandi  $url = parse_url($_SERVER['HTTP_REFERER']);
834f3f0262cSandi
835f3f0262cSandi  if(!preg_match("#google\.#i",$url['host'])) return '';
836f3f0262cSandi  $query = array();
837f3f0262cSandi  parse_str($url['query'],$query);
838f3f0262cSandi
839f3f0262cSandi  return $query['q'];
840f3f0262cSandi}
841f3f0262cSandi
842f3f0262cSandi/**
84315fae107Sandi * Try to set correct locale
84415fae107Sandi *
845095bfd5cSandi * @deprecated No longer used
84615fae107Sandi * @author     Andreas Gohr <andi@splitbrain.org>
847f3f0262cSandi */
848f3f0262cSandifunction setCorrectLocale(){
849f3f0262cSandi  global $conf;
850f3f0262cSandi  global $lang;
851f3f0262cSandi
852f3f0262cSandi  $enc = strtoupper($lang['encoding']);
853f3f0262cSandi  foreach ($lang['locales'] as $loc){
854f3f0262cSandi    //try locale
855f3f0262cSandi    if(@setlocale(LC_ALL,$loc)) return;
856f3f0262cSandi    //try loceale with encoding
857f3f0262cSandi    if(@setlocale(LC_ALL,"$loc.$enc")) return;
858f3f0262cSandi  }
859f3f0262cSandi  //still here? try to set from environment
860f3f0262cSandi  @setlocale(LC_ALL,"");
861f3f0262cSandi}
862f3f0262cSandi
863f3f0262cSandi/**
864f3f0262cSandi * Return the human readable size of a file
865f3f0262cSandi *
866f3f0262cSandi * @param       int    $size   A file size
867f3f0262cSandi * @param       int    $dec    A number of decimal places
868f3f0262cSandi * @author      Martin Benjamin <b.martin@cybernet.ch>
869f3f0262cSandi * @author      Aidan Lister <aidan@php.net>
870f3f0262cSandi * @version     1.0.0
871f3f0262cSandi */
872f31d5b73Sandifunction filesize_h($size, $dec = 1){
873f3f0262cSandi  $sizes = array('B', 'KB', 'MB', 'GB');
874f3f0262cSandi  $count = count($sizes);
875f3f0262cSandi  $i = 0;
876f3f0262cSandi
877f3f0262cSandi  while ($size >= 1024 && ($i < $count - 1)) {
878f3f0262cSandi    $size /= 1024;
879f3f0262cSandi    $i++;
880f3f0262cSandi  }
881f3f0262cSandi
882f3f0262cSandi  return round($size, $dec) . ' ' . $sizes[$i];
883f3f0262cSandi}
884f3f0262cSandi
88515fae107Sandi/**
88615fae107Sandi * Run a few sanity checks
88715fae107Sandi *
88815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
88915fae107Sandi */
890f31d5b73Sandifunction getVersion(){
891f31d5b73Sandi  //import version string
892f31d5b73Sandi  if(@file_exists('VERSION')){
893f31d5b73Sandi    //official release
894f31d5b73Sandi    return 'Release '.io_readfile('VERSION');
895f31d5b73Sandi  }elseif(is_dir('_darcs')){
896f31d5b73Sandi    //darcs checkout
897f31d5b73Sandi    $inv = file('_darcs/inventory');
898f31d5b73Sandi    $inv = preg_grep('#andi@splitbrain\.org\*\*\d{14}#',$inv);
899f31d5b73Sandi    $cur = array_pop($inv);
900f31d5b73Sandi    preg_match('#\*\*(\d{4})(\d{2})(\d{2})#',$cur,$matches);
901f31d5b73Sandi    return 'Darcs '.$matches[1].'-'.$matches[2].'-'.$matches[3];
902f31d5b73Sandi  }else{
903f31d5b73Sandi    return 'snapshot?';
904f31d5b73Sandi  }
905f31d5b73Sandi}
906f31d5b73Sandi
907f31d5b73Sandi/**
908f31d5b73Sandi * Run a few sanity checks
909f31d5b73Sandi *
910f31d5b73Sandi * @author Andreas Gohr <andi@splitbrain.org>
911f31d5b73Sandi */
912f3f0262cSandifunction check(){
913f3f0262cSandi  global $conf;
914f3f0262cSandi  global $INFO;
915f3f0262cSandi
916f31d5b73Sandi  msg('DokuWiki version: '.getVersion(),1);
917f31d5b73Sandi
91849022a38Sandi  if(version_compare(phpversion(),'4.3.0','<')){
91949022a38Sandi    msg('Your PHP version is too old ('.phpversion().' vs. 4.3.+ recommended)',-1);
92049022a38Sandi  }elseif(version_compare(phpversion(),'4.3.10','<')){
92149022a38Sandi    msg('Consider upgrading PHP to 4.3.10 or higher for security reasons (your version: '.phpversion().')',0);
92249022a38Sandi  }else{
92349022a38Sandi    msg('PHP version '.phpversion(),1);
92449022a38Sandi  }
92549022a38Sandi
926f3f0262cSandi  if(is_writable($conf['changelog'])){
927f3f0262cSandi    msg('Changelog is writable',1);
928f3f0262cSandi  }else{
929f3f0262cSandi    msg('Changelog is not writable',-1);
930f3f0262cSandi  }
931f3f0262cSandi
932f3f0262cSandi  if(is_writable($conf['datadir'])){
933f3f0262cSandi    msg('Datadir is writable',1);
934f3f0262cSandi  }else{
935f3f0262cSandi    msg('Datadir is not writable',-1);
936f3f0262cSandi  }
937f3f0262cSandi
938f3f0262cSandi  if(is_writable($conf['olddir'])){
939f3f0262cSandi    msg('Attic is writable',1);
940f3f0262cSandi  }else{
941f3f0262cSandi    msg('Attic is not writable',-1);
942f3f0262cSandi  }
943f3f0262cSandi
944f3f0262cSandi  if(is_writable($conf['mediadir'])){
945f3f0262cSandi    msg('Mediadir is writable',1);
946f3f0262cSandi  }else{
947f3f0262cSandi    msg('Mediadir is not writable',-1);
948f3f0262cSandi  }
949f3f0262cSandi
950f3f0262cSandi  if(is_writable('conf/users.auth')){
951f3f0262cSandi    msg('conf/users.auth is writable',1);
952f3f0262cSandi  }else{
953f3f0262cSandi    msg('conf/users.auth is not writable',0);
954f3f0262cSandi  }
95593a9e835Sandi
95693a9e835Sandi  if(function_exists('mb_strpos')){
95793a9e835Sandi    if(defined('UTF8_NOMBSTRING')){
95893a9e835Sandi      msg('mb_string extension is available but will not be used',0);
95993a9e835Sandi    }else{
96093a9e835Sandi      msg('mb_string extension is available and will be used',1);
96193a9e835Sandi    }
96293a9e835Sandi  }else{
96393a9e835Sandi    msg('mb_string extension not available - PHP only replacements will be used',0);
96493a9e835Sandi  }
965f3f0262cSandi
966f3f0262cSandi  msg('Your current permission for this page is '.$INFO['perm'],0);
967f3f0262cSandi
968f3f0262cSandi  if(is_writable($INFO['filepath'])){
969f3f0262cSandi    msg('The current page is writable by the webserver',0);
970f3f0262cSandi  }else{
971f3f0262cSandi    msg('The current page is not writable by the webserver',0);
972f3f0262cSandi  }
973f3f0262cSandi
974f3f0262cSandi  if($INFO['writable']){
975f3f0262cSandi    msg('The current page is writable by you',0);
976f3f0262cSandi  }else{
977f3f0262cSandi    msg('The current page is not writable you',0);
978f3f0262cSandi  }
979f3f0262cSandi}
980f3f0262cSandi?>
981