xref: /dokuwiki/inc/common.php (revision c9321d914faa3e35ad2bfc961b414f870691656b)
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
9ed7b5f09Sandiif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/');
10e7cb32dcSAndreas Gohrrequire_once(DOKU_CONF.'dokuwiki.php');
11ed7b5f09Sandirequire_once(DOKU_INC.'inc/io.php');
127d559c7fSBen Coburnrequire_once(DOKU_INC.'inc/changelog.php');
13ed7b5f09Sandirequire_once(DOKU_INC.'inc/utf8.php');
14ed7b5f09Sandirequire_once(DOKU_INC.'inc/mail.php');
15c112d578Sandirequire_once(DOKU_INC.'inc/parserutils.php');
16c29dc6e4SAndreas Gohrrequire_once(DOKU_INC.'inc/infoutils.php');
17f3f0262cSandi
18f3f0262cSandi/**
19b6912aeaSAndreas Gohr * These constants are used with the recents function
20b6912aeaSAndreas Gohr */
21b6912aeaSAndreas Gohrdefine('RECENTS_SKIP_DELETED',2);
22b6912aeaSAndreas Gohrdefine('RECENTS_SKIP_MINORS',4);
23b6912aeaSAndreas Gohrdefine('RECENTS_SKIP_SUBSPACES',8);
24b6912aeaSAndreas Gohr
25b6912aeaSAndreas Gohr/**
26d5197206Schris * Wrapper around htmlspecialchars()
27d5197206Schris *
28d5197206Schris * @author Andreas Gohr <andi@splitbrain.org>
29d5197206Schris * @see    htmlspecialchars()
30d5197206Schris */
31d5197206Schrisfunction hsc($string){
32d5197206Schris  return htmlspecialchars($string, ENT_QUOTES, 'UTF-8');
33d5197206Schris}
34d5197206Schris
35d5197206Schris/**
36d5197206Schris * print a newline terminated string
37d5197206Schris *
38d5197206Schris * You can give an indention as optional parameter
39d5197206Schris *
40d5197206Schris * @author Andreas Gohr <andi@splitbrain.org>
41d5197206Schris */
42d5197206Schrisfunction ptln($string,$intend=0){
4397c91fb2Shenning.noren  echo str_repeat(' ', $intend)."$string\n";
4402b0b681SAndreas Gohr}
4502b0b681SAndreas Gohr
4602b0b681SAndreas Gohr/**
4702b0b681SAndreas Gohr * strips control characters (<32) from the given string
4802b0b681SAndreas Gohr *
4902b0b681SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
5002b0b681SAndreas Gohr */
5102b0b681SAndreas Gohrfunction stripctl($string){
5202b0b681SAndreas Gohr  return preg_replace('/[\x00-\x1F]+/s','',$string);
53d5197206Schris}
54d5197206Schris
55d5197206Schris/**
5615fae107Sandi * Return info about the current document as associative
57f3f0262cSandi * array.
5815fae107Sandi *
5915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
60f3f0262cSandi */
61f3f0262cSandifunction pageinfo(){
62f3f0262cSandi  global $ID;
63f3f0262cSandi  global $REV;
64f3f0262cSandi  global $USERINFO;
65f3f0262cSandi  global $conf;
66f3f0262cSandi
676afe8dcaSchris  // include ID & REV not redundant, as some parts of DokuWiki may temporarily change $ID, e.g. p_wiki_xhtml
686afe8dcaSchris  // FIXME ... perhaps it would be better to ensure the temporary changes weren't necessary
696afe8dcaSchris  $info['id'] = $ID;
706afe8dcaSchris  $info['rev'] = $REV;
716afe8dcaSchris
72f3f0262cSandi  if($_SERVER['REMOTE_USER']){
73f3f0262cSandi    $info['userinfo']   = $USERINFO;
74f3f0262cSandi    $info['perm']       = auth_quickaclcheck($ID);
751380fc45SAndreas Gohr    $info['subscribed'] = is_subscribed($ID,$_SERVER['REMOTE_USER']);
76ee4c4a1bSAndreas Gohr    $info['client']     = $_SERVER['REMOTE_USER'];
7717ee7f66SAndreas Gohr
78f8cc712eSAndreas Gohr    // set info about manager/admin status
79f8cc712eSAndreas Gohr    $info['isadmin']   = false;
80f8cc712eSAndreas Gohr    $info['ismanager'] = false;
81f8cc712eSAndreas Gohr    if($info['perm'] == AUTH_ADMIN){
82f8cc712eSAndreas Gohr      $info['isadmin']   = true;
83f8cc712eSAndreas Gohr      $info['ismanager'] = true;
84f8cc712eSAndreas Gohr    }elseif(auth_ismanager()){
85f8cc712eSAndreas Gohr      $info['ismanager'] = true;
86f8cc712eSAndreas Gohr    }
87f8cc712eSAndreas Gohr
8817ee7f66SAndreas Gohr    // if some outside auth were used only REMOTE_USER is set
8917ee7f66SAndreas Gohr    if(!$info['userinfo']['name']){
9017ee7f66SAndreas Gohr      $info['userinfo']['name'] = $_SERVER['REMOTE_USER'];
9117ee7f66SAndreas Gohr    }
92ee4c4a1bSAndreas Gohr
93f3f0262cSandi  }else{
94f3f0262cSandi    $info['perm']       = auth_aclcheck($ID,'',null);
951380fc45SAndreas Gohr    $info['subscribed'] = false;
96ee4c4a1bSAndreas Gohr    $info['client']     = clientIP(true);
97f3f0262cSandi  }
98f3f0262cSandi
99f3f0262cSandi  $info['namespace'] = getNS($ID);
100f3f0262cSandi  $info['locked']    = checklock($ID);
1012ca9d91cSBen Coburn  $info['filepath']  = realpath(wikiFN($ID));
1022ca9d91cSBen Coburn  $info['exists']    = @file_exists($info['filepath']);
1032ca9d91cSBen Coburn  if($REV){
1042ca9d91cSBen Coburn    //check if current revision was meant
1052ca9d91cSBen Coburn    if($info['exists'] && (@filemtime($info['filepath'])==$REV)){
1062ca9d91cSBen Coburn      $REV = '';
1072ca9d91cSBen Coburn    }else{
1082ca9d91cSBen Coburn      //really use old revision
109f3f0262cSandi      $info['filepath'] = realpath(wikiFN($ID,$REV));
110f3f0262cSandi      $info['exists']   = @file_exists($info['filepath']);
111f3f0262cSandi    }
112f3f0262cSandi  }
113c112d578Sandi  $info['rev'] = $REV;
114f3f0262cSandi  if($info['exists']){
115f3f0262cSandi    $info['writable'] = (is_writable($info['filepath']) &&
116f3f0262cSandi                         ($info['perm'] >= AUTH_EDIT));
117f3f0262cSandi  }else{
118f3f0262cSandi    $info['writable'] = ($info['perm'] >= AUTH_CREATE);
119f3f0262cSandi  }
120f3f0262cSandi  $info['editable']  = ($info['writable'] && empty($info['lock']));
121f3f0262cSandi  $info['lastmod']   = @filemtime($info['filepath']);
122f3f0262cSandi
12371726d78SBen Coburn  //load page meta data
12471726d78SBen Coburn  $info['meta'] = p_get_metadata($ID);
12571726d78SBen Coburn
126652610a2Sandi  //who's the editor
127652610a2Sandi  if($REV){
12871726d78SBen Coburn    $revinfo = getRevisionInfo($ID, $REV, 1024);
129652610a2Sandi  }else{
130aa27cf05SAndreas Gohr    if (is_array($info['meta']['last_change'])) {
131aa27cf05SAndreas Gohr       $revinfo = $info['meta']['last_change'];
132aa27cf05SAndreas Gohr    } else {
133cd00a034SBen Coburn      $revinfo = getRevisionInfo($ID, $info['lastmod'], 1024);
134cd00a034SBen Coburn      // cache most recent changelog line in metadata if missing and still valid
135cd00a034SBen Coburn      if ($revinfo!==false) {
136cd00a034SBen Coburn        $info['meta']['last_change'] = $revinfo;
137cd00a034SBen Coburn        p_set_metadata($ID, array('last_change' => $revinfo));
138cd00a034SBen Coburn      }
139cd00a034SBen Coburn    }
140cd00a034SBen Coburn  }
141cd00a034SBen Coburn  //and check for an external edit
142cd00a034SBen Coburn  if($revinfo!==false && $revinfo['date']!=$info['lastmod']){
143cd00a034SBen Coburn    // cached changelog line no longer valid
144cd00a034SBen Coburn    $revinfo = false;
145cd00a034SBen Coburn    $info['meta']['last_change'] = $revinfo;
146cd00a034SBen Coburn    p_set_metadata($ID, array('last_change' => $revinfo));
147652610a2Sandi  }
148bb4866bdSchris
149652610a2Sandi  $info['ip']     = $revinfo['ip'];
150652610a2Sandi  $info['user']   = $revinfo['user'];
151652610a2Sandi  $info['sum']    = $revinfo['sum'];
15271726d78SBen Coburn  // See also $INFO['meta']['last_change'] which is the most recent log line for page $ID.
153ebf1501fSBen Coburn  // Use $INFO['meta']['last_change']['type']===DOKU_CHANGE_TYPE_MINOR_EDIT in place of $info['minor'].
15459f257aeSchris
15588f522e9Sandi  if($revinfo['user']){
15688f522e9Sandi    $info['editor'] = $revinfo['user'];
15788f522e9Sandi  }else{
15888f522e9Sandi    $info['editor'] = $revinfo['ip'];
15988f522e9Sandi  }
160652610a2Sandi
161ee4c4a1bSAndreas Gohr  // draft
162ee4c4a1bSAndreas Gohr  $draft = getCacheName($info['client'].$ID,'.draft');
163ee4c4a1bSAndreas Gohr  if(@file_exists($draft)){
164ee4c4a1bSAndreas Gohr    if(@filemtime($draft) < @filemtime(wikiFN($ID))){
165ee4c4a1bSAndreas Gohr      // remove stale draft
166ee4c4a1bSAndreas Gohr      @unlink($draft);
167ee4c4a1bSAndreas Gohr    }else{
168ee4c4a1bSAndreas Gohr      $info['draft'] = $draft;
169ee4c4a1bSAndreas Gohr    }
170ee4c4a1bSAndreas Gohr  }
171ee4c4a1bSAndreas Gohr
172f3f0262cSandi  return $info;
173f3f0262cSandi}
174f3f0262cSandi
175f3f0262cSandi/**
1762684e50aSAndreas Gohr * Build an string of URL parameters
1772684e50aSAndreas Gohr *
1782684e50aSAndreas Gohr * @author Andreas Gohr
1792684e50aSAndreas Gohr */
180b174aeaeSchrisfunction buildURLparams($params, $sep='&amp;'){
1812684e50aSAndreas Gohr  $url = '';
1822684e50aSAndreas Gohr  $amp = false;
1832684e50aSAndreas Gohr  foreach($params as $key => $val){
184b174aeaeSchris    if($amp) $url .= $sep;
1852684e50aSAndreas Gohr
1862684e50aSAndreas Gohr    $url .= $key.'=';
187b6c6979fSAndreas Gohr    $url .= rawurlencode($val);
1882684e50aSAndreas Gohr    $amp = true;
1892684e50aSAndreas Gohr  }
1902684e50aSAndreas Gohr  return $url;
1912684e50aSAndreas Gohr}
1922684e50aSAndreas Gohr
1932684e50aSAndreas Gohr/**
1942684e50aSAndreas Gohr * Build an string of html tag attributes
1952684e50aSAndreas Gohr *
1967bff22c0SAndreas Gohr * Skips keys starting with '_', values get HTML encoded
1977bff22c0SAndreas Gohr *
1982684e50aSAndreas Gohr * @author Andreas Gohr
1992684e50aSAndreas Gohr */
2002684e50aSAndreas Gohrfunction buildAttributes($params){
2012684e50aSAndreas Gohr  $url = '';
2022684e50aSAndreas Gohr  foreach($params as $key => $val){
2037bff22c0SAndreas Gohr    if($key{0} == '_') continue;
2047bff22c0SAndreas Gohr
2052684e50aSAndreas Gohr    $url .= $key.'="';
2062684e50aSAndreas Gohr    $url .= htmlspecialchars ($val);
2072684e50aSAndreas Gohr    $url .= '" ';
2082684e50aSAndreas Gohr  }
2092684e50aSAndreas Gohr  return $url;
2102684e50aSAndreas Gohr}
2112684e50aSAndreas Gohr
2122684e50aSAndreas Gohr
2132684e50aSAndreas Gohr/**
21415fae107Sandi * This builds the breadcrumb trail and returns it as array
21515fae107Sandi *
21615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
217f3f0262cSandi */
218f3f0262cSandifunction breadcrumbs(){
2198746e727Sandi  // we prepare the breadcrumbs early for quick session closing
2208746e727Sandi  static $crumbs = null;
2218746e727Sandi  if($crumbs != null) return $crumbs;
2228746e727Sandi
223f3f0262cSandi  global $ID;
224f3f0262cSandi  global $ACT;
225f3f0262cSandi  global $conf;
226e71ce681SAndreas Gohr  $crumbs = $_SESSION[DOKU_COOKIE]['bc'];
227f3f0262cSandi
228f3f0262cSandi  //first visit?
229f3f0262cSandi  if (!is_array($crumbs)){
230f3f0262cSandi    $crumbs = array();
231f3f0262cSandi  }
232f3f0262cSandi  //we only save on show and existing wiki documents
233a77f5846Sjan  $file = wikiFN($ID);
234a77f5846Sjan  if($ACT != 'show' || !@file_exists($file)){
235e71ce681SAndreas Gohr    $_SESSION[DOKU_COOKIE]['bc'] = $crumbs;
236f3f0262cSandi    return $crumbs;
237f3f0262cSandi  }
238a77f5846Sjan
239a77f5846Sjan  // page names
2401a84a0f3SAnika Henke  $name = noNSorNS($ID);
241a77f5846Sjan  if ($conf['useheading']) {
242a77f5846Sjan    // get page title
243bb0a59d4Sjan    $title = p_get_first_heading($ID);
244a77f5846Sjan    if ($title) {
245a77f5846Sjan      $name = $title;
246a77f5846Sjan    }
247a77f5846Sjan  }
248a77f5846Sjan
249f3f0262cSandi  //remove ID from array
250a77f5846Sjan  if (isset($crumbs[$ID])) {
251a77f5846Sjan    unset($crumbs[$ID]);
252f3f0262cSandi  }
253f3f0262cSandi
254f3f0262cSandi  //add to array
255a77f5846Sjan  $crumbs[$ID] = $name;
256f3f0262cSandi  //reduce size
257f3f0262cSandi  while(count($crumbs) > $conf['breadcrumbs']){
258f3f0262cSandi    array_shift($crumbs);
259f3f0262cSandi  }
260f3f0262cSandi  //save to session
261e71ce681SAndreas Gohr  $_SESSION[DOKU_COOKIE]['bc'] = $crumbs;
262f3f0262cSandi  return $crumbs;
263f3f0262cSandi}
264f3f0262cSandi
265f3f0262cSandi/**
26615fae107Sandi * Filter for page IDs
26715fae107Sandi *
268f3f0262cSandi * This is run on a ID before it is outputted somewhere
269f3f0262cSandi * currently used to replace the colon with something else
270f3f0262cSandi * on Windows systems and to have proper URL encoding
27115fae107Sandi *
27249c713a3Sandi * Urlencoding is ommitted when the second parameter is false
27349c713a3Sandi *
27415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
275f3f0262cSandi */
27649c713a3Sandifunction idfilter($id,$ue=true){
277f3f0262cSandi  global $conf;
278f3f0262cSandi  if ($conf['useslash'] && $conf['userewrite']){
279f3f0262cSandi    $id = strtr($id,':','/');
280f3f0262cSandi  }elseif (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' &&
281f3f0262cSandi      $conf['userewrite']) {
282f3f0262cSandi    $id = strtr($id,':',';');
283f3f0262cSandi  }
28449c713a3Sandi  if($ue){
285b6c6979fSAndreas Gohr    $id = rawurlencode($id);
286f3f0262cSandi    $id = str_replace('%3A',':',$id); //keep as colon
287f3f0262cSandi    $id = str_replace('%2F','/',$id); //keep as slash
28849c713a3Sandi  }
289f3f0262cSandi  return $id;
290f3f0262cSandi}
291f3f0262cSandi
292f3f0262cSandi/**
293ed7b5f09Sandi * This builds a link to a wikipage
29415fae107Sandi *
2956c7843b5Sandi * It handles URL rewriting and adds additional parameter if
2966c7843b5Sandi * given in $more
2976c7843b5Sandi *
29815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
299f3f0262cSandi */
300b174aeaeSchrisfunction wl($id='',$more='',$abs=false,$sep='&amp;'){
301f3f0262cSandi  global $conf;
3026de3759aSAndreas Gohr  if(is_array($more)){
303b174aeaeSchris    $more = buildURLparams($more,$sep);
3046de3759aSAndreas Gohr  }else{
305b174aeaeSchris    $more = str_replace(',',$sep,$more);
3066de3759aSAndreas Gohr  }
307f3f0262cSandi
308f3f0262cSandi  $id    = idfilter($id);
309ed7b5f09Sandi  if($abs){
310ed7b5f09Sandi    $xlink = DOKU_URL;
311ed7b5f09Sandi  }else{
312ed7b5f09Sandi    $xlink = DOKU_BASE;
313ed7b5f09Sandi  }
314f3f0262cSandi
3156c7843b5Sandi  if($conf['userewrite'] == 2){
3166c7843b5Sandi    $xlink .= DOKU_SCRIPT.'/'.$id;
3176c7843b5Sandi    if($more) $xlink .= '?'.$more;
3186c7843b5Sandi  }elseif($conf['userewrite']){
319f3f0262cSandi    $xlink .= $id;
320f3f0262cSandi    if($more) $xlink .= '?'.$more;
3216c7843b5Sandi  }else{
3226c7843b5Sandi    $xlink .= DOKU_SCRIPT.'?id='.$id;
323b174aeaeSchris    if($more) $xlink .= $sep.$more;
324f3f0262cSandi  }
325f3f0262cSandi
326f3f0262cSandi  return $xlink;
327f3f0262cSandi}
328f3f0262cSandi
329f3f0262cSandi/**
330f5c2808fSBen Coburn * This builds a link to an alternate page format
331f5c2808fSBen Coburn *
332f5c2808fSBen Coburn * Handles URL rewriting if enabled. Follows the style of wl().
333f5c2808fSBen Coburn *
334f5c2808fSBen Coburn * @author Ben Coburn <btcoburn@silicodon.net>
335f5c2808fSBen Coburn */
336f5c2808fSBen Coburnfunction exportlink($id='',$format='raw',$more='',$abs=false,$sep='&amp;'){
337f5c2808fSBen Coburn  global $conf;
338f5c2808fSBen Coburn  if(is_array($more)){
339f5c2808fSBen Coburn    $more = buildURLparams($more,$sep);
340f5c2808fSBen Coburn  }else{
341f5c2808fSBen Coburn    $more = str_replace(',',$sep,$more);
342f5c2808fSBen Coburn  }
343f5c2808fSBen Coburn
344f5c2808fSBen Coburn  $format = rawurlencode($format);
345f5c2808fSBen Coburn  $id = idfilter($id);
346f5c2808fSBen Coburn  if($abs){
347f5c2808fSBen Coburn    $xlink = DOKU_URL;
348f5c2808fSBen Coburn  }else{
349f5c2808fSBen Coburn    $xlink = DOKU_BASE;
350f5c2808fSBen Coburn  }
351f5c2808fSBen Coburn
352f5c2808fSBen Coburn  if($conf['userewrite'] == 2){
353f5c2808fSBen Coburn    $xlink .= DOKU_SCRIPT.'/'.$id.'?do=export_'.$format;
354f5c2808fSBen Coburn    if($more) $xlink .= $sep.$more;
355f5c2808fSBen Coburn  }elseif($conf['userewrite'] == 1){
356f5c2808fSBen Coburn    $xlink .= '_export/'.$format.'/'.$id;
357f5c2808fSBen Coburn    if($more) $xlink .= '?'.$more;
358f5c2808fSBen Coburn  }else{
359f5c2808fSBen Coburn    $xlink .= DOKU_SCRIPT.'?do=export_'.$format.$sep.'id='.$id;
360f5c2808fSBen Coburn    if($more) $xlink .= $sep.$more;
361f5c2808fSBen Coburn  }
362f5c2808fSBen Coburn
363f5c2808fSBen Coburn  return $xlink;
364f5c2808fSBen Coburn}
365f5c2808fSBen Coburn
366f5c2808fSBen Coburn/**
3676de3759aSAndreas Gohr * Build a link to a media file
3686de3759aSAndreas Gohr *
3696de3759aSAndreas Gohr * Will return a link to the detail page if $direct is false
3706de3759aSAndreas Gohr */
371b174aeaeSchrisfunction ml($id='',$more='',$direct=true,$sep='&amp;'){
3726de3759aSAndreas Gohr  global $conf;
3736de3759aSAndreas Gohr  if(is_array($more)){
374b174aeaeSchris    $more = buildURLparams($more,$sep);
3756de3759aSAndreas Gohr  }else{
376b174aeaeSchris    $more = str_replace(',',$sep,$more);
3776de3759aSAndreas Gohr  }
3786de3759aSAndreas Gohr
3796de3759aSAndreas Gohr  $xlink = DOKU_BASE;
3806de3759aSAndreas Gohr
3816de3759aSAndreas Gohr  // external URLs are always direct without rewriting
3826de3759aSAndreas Gohr  if(preg_match('#^(https?|ftp)://#i',$id)){
3836de3759aSAndreas Gohr    $xlink .= 'lib/exe/fetch.php';
3846de3759aSAndreas Gohr    if($more){
3856de3759aSAndreas Gohr      $xlink .= '?'.$more;
386b174aeaeSchris      $xlink .= $sep.'media='.rawurlencode($id);
3876de3759aSAndreas Gohr    }else{
388b6c6979fSAndreas Gohr      $xlink .= '?media='.rawurlencode($id);
3896de3759aSAndreas Gohr    }
3906de3759aSAndreas Gohr    return $xlink;
3916de3759aSAndreas Gohr  }
3926de3759aSAndreas Gohr
3936de3759aSAndreas Gohr  $id = idfilter($id);
3946de3759aSAndreas Gohr
3956de3759aSAndreas Gohr  // decide on scriptname
3966de3759aSAndreas Gohr  if($direct){
3976de3759aSAndreas Gohr    if($conf['userewrite'] == 1){
3986de3759aSAndreas Gohr      $script = '_media';
3996de3759aSAndreas Gohr    }else{
4006de3759aSAndreas Gohr      $script = 'lib/exe/fetch.php';
4016de3759aSAndreas Gohr    }
4026de3759aSAndreas Gohr  }else{
4036de3759aSAndreas Gohr    if($conf['userewrite'] == 1){
4046de3759aSAndreas Gohr      $script = '_detail';
4056de3759aSAndreas Gohr    }else{
4066de3759aSAndreas Gohr      $script = 'lib/exe/detail.php';
4076de3759aSAndreas Gohr    }
4086de3759aSAndreas Gohr  }
4096de3759aSAndreas Gohr
4106de3759aSAndreas Gohr  // build URL based on rewrite mode
4116de3759aSAndreas Gohr   if($conf['userewrite']){
4126de3759aSAndreas Gohr     $xlink .= $script.'/'.$id;
4136de3759aSAndreas Gohr     if($more) $xlink .= '?'.$more;
4146de3759aSAndreas Gohr   }else{
4156de3759aSAndreas Gohr     if($more){
416a99d3236SEsther Brunner       $xlink .= $script.'?'.$more;
417b174aeaeSchris       $xlink .= $sep.'media='.$id;
4186de3759aSAndreas Gohr     }else{
419a99d3236SEsther Brunner       $xlink .= $script.'?media='.$id;
4206de3759aSAndreas Gohr     }
4216de3759aSAndreas Gohr   }
4226de3759aSAndreas Gohr
4236de3759aSAndreas Gohr  return $xlink;
4246de3759aSAndreas Gohr}
4256de3759aSAndreas Gohr
4266de3759aSAndreas Gohr
4276de3759aSAndreas Gohr
4286de3759aSAndreas Gohr/**
429f3f0262cSandi * Just builds a link to a script
43015fae107Sandi *
431ed7b5f09Sandi * @todo   maybe obsolete
43215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
433f3f0262cSandi */
434f3f0262cSandifunction script($script='doku.php'){
435ed7b5f09Sandi#  $link = getBaseURL();
436ed7b5f09Sandi#  $link .= $script;
437ed7b5f09Sandi#  return $link;
438ed7b5f09Sandi  return DOKU_BASE.DOKU_SCRIPT;
439f3f0262cSandi}
440f3f0262cSandi
441f3f0262cSandi/**
44215fae107Sandi * Spamcheck against wordlist
44315fae107Sandi *
444f3f0262cSandi * Checks the wikitext against a list of blocked expressions
445f3f0262cSandi * returns true if the text contains any bad words
44615fae107Sandi *
44715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
448f3f0262cSandi */
449f3f0262cSandifunction checkwordblock(){
450f3f0262cSandi  global $TEXT;
451f3f0262cSandi  global $conf;
452f3f0262cSandi
453f3f0262cSandi  if(!$conf['usewordblock']) return false;
454f3f0262cSandi
455041d1964SAndreas Gohr  // we prepare the text a tiny bit to prevent spammers circumventing URL checks
456041d1964SAndreas Gohr  $text = preg_replace('!(\b)(www\.[\w.:?\-;,]+?\.[\w.:?\-;,]+?[\w/\#~:.?+=&%@\!\-.:?\-;,]+?)([.:?\-;,]*[^\w/\#~:.?+=&%@\!\-.:?\-;,])!i','\1http://\2 \2\3',$TEXT);
457041d1964SAndreas Gohr
458b9ac8716Schris  $wordblocks = getWordblocks();
4593e2965d7Sandi  //how many lines to read at once (to work around some PCRE limits)
4603e2965d7Sandi  if(version_compare(phpversion(),'4.3.0','<')){
4613e2965d7Sandi    //old versions of PCRE define a maximum of parenthesises even if no
4623e2965d7Sandi    //backreferences are used - the maximum is 99
4633e2965d7Sandi    //this is very bad performancewise and may even be too high still
4643e2965d7Sandi    $chunksize = 40;
4653e2965d7Sandi  }else{
466a51d08efSAndreas Gohr    //read file in chunks of 200 - this should work around the
4673e2965d7Sandi    //MAX_PATTERN_SIZE in modern PCRE
468a51d08efSAndreas Gohr    $chunksize = 200;
4693e2965d7Sandi  }
470b9ac8716Schris  while($blocks = array_splice($wordblocks,0,$chunksize)){
471f3f0262cSandi    $re = array();
472f3f0262cSandi    #build regexp from blocks
473f3f0262cSandi    foreach($blocks as $block){
474f3f0262cSandi      $block = preg_replace('/#.*$/','',$block);
475f3f0262cSandi      $block = trim($block);
476f3f0262cSandi      if(empty($block)) continue;
477f3f0262cSandi      $re[]  = $block;
478f3f0262cSandi    }
4796e983f2aSAndreas Gohr    if(preg_match('#('.join('|',$re).')#si',$text)) {
480b9ac8716Schris      return true;
481b9ac8716Schris    }
482703f6fdeSandi  }
483f3f0262cSandi  return false;
484f3f0262cSandi}
485f3f0262cSandi
486f3f0262cSandi/**
48715fae107Sandi * Return the IP of the client
48815fae107Sandi *
4896d8affe6SAndreas Gohr * Honours X-Forwarded-For and X-Real-IP Proxy Headers
49015fae107Sandi *
4916d8affe6SAndreas Gohr * It returns a comma separated list of IPs if the above mentioned
4926d8affe6SAndreas Gohr * headers are set. If the single parameter is set, it tries to return
4936d8affe6SAndreas Gohr * a routable public address, prefering the ones suplied in the X
4946d8affe6SAndreas Gohr * headers
4956d8affe6SAndreas Gohr *
4966d8affe6SAndreas Gohr * @param  boolean $single If set only a single IP is returned
49715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
498f3f0262cSandi */
4996d8affe6SAndreas Gohrfunction clientIP($single=false){
5006d8affe6SAndreas Gohr  $ip = array();
5016d8affe6SAndreas Gohr  $ip[] = $_SERVER['REMOTE_ADDR'];
502bb4866bdSchris  if(!empty($_SERVER['HTTP_X_FORWARDED_FOR']))
5036d8affe6SAndreas Gohr    $ip = array_merge($ip,explode(',',$_SERVER['HTTP_X_FORWARDED_FOR']));
504bb4866bdSchris  if(!empty($_SERVER['HTTP_X_REAL_IP']))
5056d8affe6SAndreas Gohr    $ip = array_merge($ip,explode(',',$_SERVER['HTTP_X_REAL_IP']));
5066d8affe6SAndreas Gohr
5076d8affe6SAndreas Gohr  // remove any non-IP stuff
5086d8affe6SAndreas Gohr  $cnt = count($ip);
5094ff28443Schris  $match = array();
5106d8affe6SAndreas Gohr  for($i=0; $i<$cnt; $i++){
5114ff28443Schris    if(preg_match('/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/',$ip[$i],$match)) {
5124ff28443Schris      $ip[$i] = $match[0];
5134ff28443Schris    } else {
5144ff28443Schris      $ip[$i] = '';
5154ff28443Schris    }
5166d8affe6SAndreas Gohr    if(empty($ip[$i])) unset($ip[$i]);
517f3f0262cSandi  }
5186d8affe6SAndreas Gohr  $ip = array_values(array_unique($ip));
5196d8affe6SAndreas Gohr  if(!$ip[0]) $ip[0] = '0.0.0.0'; // for some strange reason we don't have a IP
5206d8affe6SAndreas Gohr
5216d8affe6SAndreas Gohr  if(!$single) return join(',',$ip);
5226d8affe6SAndreas Gohr
5236d8affe6SAndreas Gohr  // decide which IP to use, trying to avoid local addresses
5246d8affe6SAndreas Gohr  $ip = array_reverse($ip);
5256d8affe6SAndreas Gohr  foreach($ip as $i){
5266d8affe6SAndreas Gohr    if(preg_match('/^(127\.|10\.|192\.168\.|172\.((1[6-9])|(2[0-9])|(3[0-1]))\.)/',$i)){
5276d8affe6SAndreas Gohr      continue;
5286d8affe6SAndreas Gohr    }else{
5296d8affe6SAndreas Gohr      return $i;
5306d8affe6SAndreas Gohr    }
5316d8affe6SAndreas Gohr  }
5326d8affe6SAndreas Gohr  // still here? just use the first (last) address
5336d8affe6SAndreas Gohr  return $ip[0];
534f3f0262cSandi}
535f3f0262cSandi
536f3f0262cSandi/**
53715fae107Sandi * Checks if a given page is currently locked.
53815fae107Sandi *
539f3f0262cSandi * removes stale lockfiles
54015fae107Sandi *
54115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
542f3f0262cSandi */
543f3f0262cSandifunction checklock($id){
544f3f0262cSandi  global $conf;
545c9b4bd1eSBen Coburn  $lock = wikiLockFN($id);
546f3f0262cSandi
547f3f0262cSandi  //no lockfile
548f3f0262cSandi  if(!@file_exists($lock)) return false;
549f3f0262cSandi
550f3f0262cSandi  //lockfile expired
551f3f0262cSandi  if((time() - filemtime($lock)) > $conf['locktime']){
552d8186216SBen Coburn    @unlink($lock);
553f3f0262cSandi    return false;
554f3f0262cSandi  }
555f3f0262cSandi
556f3f0262cSandi  //my own lock
557f3f0262cSandi  $ip = io_readFile($lock);
558f3f0262cSandi  if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
559f3f0262cSandi    return false;
560f3f0262cSandi  }
561f3f0262cSandi
562f3f0262cSandi  return $ip;
563f3f0262cSandi}
564f3f0262cSandi
565f3f0262cSandi/**
56615fae107Sandi * Lock a page for editing
56715fae107Sandi *
56815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
569f3f0262cSandi */
570f3f0262cSandifunction lock($id){
571c9b4bd1eSBen Coburn  $lock = wikiLockFN($id);
572f3f0262cSandi  if($_SERVER['REMOTE_USER']){
573f3f0262cSandi    io_saveFile($lock,$_SERVER['REMOTE_USER']);
574f3f0262cSandi  }else{
575f3f0262cSandi    io_saveFile($lock,clientIP());
576f3f0262cSandi  }
577f3f0262cSandi}
578f3f0262cSandi
579f3f0262cSandi/**
58015fae107Sandi * Unlock a page if it was locked by the user
581f3f0262cSandi *
58215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
58315fae107Sandi * @return bool true if a lock was removed
584f3f0262cSandi */
585f3f0262cSandifunction unlock($id){
586c9b4bd1eSBen Coburn  $lock = wikiLockFN($id);
587f3f0262cSandi  if(@file_exists($lock)){
588f3f0262cSandi    $ip = io_readFile($lock);
589f3f0262cSandi    if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){
590f3f0262cSandi      @unlink($lock);
591f3f0262cSandi      return true;
592f3f0262cSandi    }
593f3f0262cSandi  }
594f3f0262cSandi  return false;
595f3f0262cSandi}
596f3f0262cSandi
597f3f0262cSandi/**
598f3f0262cSandi * convert line ending to unix format
599f3f0262cSandi *
60015fae107Sandi * @see    formText() for 2crlf conversion
60115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
602f3f0262cSandi */
603f3f0262cSandifunction cleanText($text){
604f3f0262cSandi  $text = preg_replace("/(\015\012)|(\015)/","\012",$text);
605f3f0262cSandi  return $text;
606f3f0262cSandi}
607f3f0262cSandi
608f3f0262cSandi/**
609f3f0262cSandi * Prepares text for print in Webforms by encoding special chars.
610f3f0262cSandi * It also converts line endings to Windows format which is
611f3f0262cSandi * pseudo standard for webforms.
612f3f0262cSandi *
61315fae107Sandi * @see    cleanText() for 2unix conversion
61415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
615f3f0262cSandi */
616f3f0262cSandifunction formText($text){
6175b7d45a5SAndreas Gohr  $text = str_replace("\012","\015\012",$text);
618f3f0262cSandi  return htmlspecialchars($text);
619f3f0262cSandi}
620f3f0262cSandi
621f3f0262cSandi/**
62215fae107Sandi * Returns the specified local text in raw format
62315fae107Sandi *
62415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
625f3f0262cSandi */
626f3f0262cSandifunction rawLocale($id){
627f3f0262cSandi  return io_readFile(localeFN($id));
628f3f0262cSandi}
629f3f0262cSandi
630f3f0262cSandi/**
631f3f0262cSandi * Returns the raw WikiText
63215fae107Sandi *
63315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
634f3f0262cSandi */
635f3f0262cSandifunction rawWiki($id,$rev=''){
636cc7d0c94SBen Coburn  return io_readWikiPage(wikiFN($id, $rev), $id, $rev);
637f3f0262cSandi}
638f3f0262cSandi
639f3f0262cSandi/**
6407146cee2SAndreas Gohr * Returns the pagetemplate contents for the ID's namespace
6417146cee2SAndreas Gohr *
6427146cee2SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
6437146cee2SAndreas Gohr */
644b7d5a5f0SAndreas Gohrfunction pageTemplate($data){
645b7d5a5f0SAndreas Gohr  $id = $data[0];
646a15ce62dSEsther Brunner  global $conf;
647a15ce62dSEsther Brunner  global $INFO;
648a15ce62dSEsther Brunner  $tpl = io_readFile(dirname(wikiFN($id)).'/_template.txt');
649a15ce62dSEsther Brunner  $tpl = str_replace('@ID@',$id,$tpl);
650a15ce62dSEsther Brunner  $tpl = str_replace('@NS@',getNS($id),$tpl);
651a15ce62dSEsther Brunner  $tpl = str_replace('@PAGE@',strtr(noNS($id),'_',' '),$tpl);
652a15ce62dSEsther Brunner  $tpl = str_replace('@USER@',$_SERVER['REMOTE_USER'],$tpl);
653a15ce62dSEsther Brunner  $tpl = str_replace('@NAME@',$INFO['userinfo']['name'],$tpl);
654a15ce62dSEsther Brunner  $tpl = str_replace('@MAIL@',$INFO['userinfo']['mail'],$tpl);
655a15ce62dSEsther Brunner  $tpl = str_replace('@DATE@',date($conf['dformat']),$tpl);
656a15ce62dSEsther Brunner  return $tpl;
6577146cee2SAndreas Gohr}
6587146cee2SAndreas Gohr
6597146cee2SAndreas Gohr
6607146cee2SAndreas Gohr/**
66115fae107Sandi * Returns the raw Wiki Text in three slices.
66215fae107Sandi *
66315fae107Sandi * The range parameter needs to have the form "from-to"
66415cfe303Sandi * and gives the range of the section in bytes - no
66515cfe303Sandi * UTF-8 awareness is needed.
666f3f0262cSandi * The returned order is prefix, section and suffix.
66715fae107Sandi *
66815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
669f3f0262cSandi */
670f3f0262cSandifunction rawWikiSlices($range,$id,$rev=''){
671f3f0262cSandi  list($from,$to) = split('-',$range,2);
672cc7d0c94SBen Coburn  $text = io_readWikiPage(wikiFN($id, $rev), $id, $rev);
673f3f0262cSandi  if(!$from) $from = 0;
674c3d8e19bSandi  if(!$to)   $to   = strlen($text)+1;
675f3f0262cSandi
67615cfe303Sandi  $slices[0] = substr($text,0,$from-1);
67715cfe303Sandi  $slices[1] = substr($text,$from-1,$to-$from);
67815cfe303Sandi  $slices[2] = substr($text,$to);
679f3f0262cSandi
680f3f0262cSandi  return $slices;
681f3f0262cSandi}
682f3f0262cSandi
683f3f0262cSandi/**
68415fae107Sandi * Joins wiki text slices
68515fae107Sandi *
686f3f0262cSandi * function to join the text slices with correct lineendings again.
687f3f0262cSandi * When the pretty parameter is set to true it adds additional empty
688f3f0262cSandi * lines between sections if needed (used on saving).
68915fae107Sandi *
69015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
691f3f0262cSandi */
692f3f0262cSandifunction con($pre,$text,$suf,$pretty=false){
693f3f0262cSandi
694f3f0262cSandi  if($pretty){
695f3f0262cSandi    if($pre && substr($pre,-1) != "\n") $pre .= "\n";
696f3f0262cSandi    if($suf && substr($text,-1) != "\n") $text .= "\n";
697f3f0262cSandi  }
698f3f0262cSandi
699f3f0262cSandi  if($pre) $pre .= "\n";
700f3f0262cSandi  if($suf) $text .= "\n";
701f3f0262cSandi  return $pre.$text.$suf;
702f3f0262cSandi}
703f3f0262cSandi
704f3f0262cSandi/**
705a701424fSBen Coburn * Saves a wikitext by calling io_writeWikiPage.
706a701424fSBen Coburn * Also directs changelog and attic updates.
70715fae107Sandi *
70815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
70971726d78SBen Coburn * @author Ben Coburn <btcoburn@silicodon.net>
710f3f0262cSandi */
711b6912aeaSAndreas Gohrfunction saveWikiText($id,$text,$summary,$minor=false){
712a701424fSBen Coburn  /* Note to developers:
713a701424fSBen Coburn     This code is subtle and delicate. Test the behavior of
714a701424fSBen Coburn     the attic and changelog with dokuwiki and external edits
715a701424fSBen Coburn     after any changes. External edits change the wiki page
716a701424fSBen Coburn     directly without using php or dokuwiki.
717a701424fSBen Coburn  */
718f3f0262cSandi  global $conf;
719f3f0262cSandi  global $lang;
72071726d78SBen Coburn  global $REV;
721f3f0262cSandi  // ignore if no changes were made
722f3f0262cSandi  if($text == rawWiki($id,'')){
723f3f0262cSandi    return;
724f3f0262cSandi  }
725f3f0262cSandi
726f3f0262cSandi  $file = wikiFN($id);
727a701424fSBen Coburn  $old = @filemtime($file); // from page
72871726d78SBen Coburn  $wasRemoved = empty($text);
729d8186216SBen Coburn  $wasCreated = !@file_exists($file);
73071726d78SBen Coburn  $wasReverted = ($REV==true);
731e45b34cdSBen Coburn  $newRev = false;
732a701424fSBen Coburn  $oldRev = getRevisions($id, -1, 1, 1024); // from changelog
733a701424fSBen Coburn  $oldRev = (int)(empty($oldRev)?0:$oldRev[0]);
734a701424fSBen Coburn  if(!@file_exists(wikiFN($id, $old)) && @file_exists($file) && $old>=$oldRev) {
73546844156SBen Coburn    // add old revision to the attic if missing
73646844156SBen Coburn    saveOldRevision($id);
73746844156SBen Coburn    // add a changelog entry if this edit came from outside dokuwiki
738a701424fSBen Coburn    if ($old>$oldRev) {
739ebf1501fSBen Coburn      addLogEntry($old, $id, DOKU_CHANGE_TYPE_EDIT, $lang['external_edit'], '', array('ExternalEdit'=>true));
74046844156SBen Coburn      // remove soon to be stale instructions
74146844156SBen Coburn      $cache = new cache_instructions($id, $file);
74246844156SBen Coburn      $cache->removeCache();
74346844156SBen Coburn    }
74446844156SBen Coburn  }
745f3f0262cSandi
74671726d78SBen Coburn  if ($wasRemoved){
747e45b34cdSBen Coburn    // pre-save deleted revision
748e45b34cdSBen Coburn    @touch($file);
74946844156SBen Coburn    clearstatcache();
750e45b34cdSBen Coburn    $newRev = saveOldRevision($id);
751e1f3d9e1SEsther Brunner    // remove empty file
752f3f0262cSandi    @unlink($file);
75371726d78SBen Coburn    // remove old meta info...
754e1f3d9e1SEsther Brunner    $mfiles = metaFiles($id);
75571726d78SBen Coburn    $changelog = metaFN($id, '.changes');
756e1f3d9e1SEsther Brunner    foreach ($mfiles as $mfile) {
75771726d78SBen Coburn      // but keep per-page changelog to preserve page history
758d8186216SBen Coburn      if (@file_exists($mfile) && $mfile!==$changelog) { @unlink($mfile); }
759b158d625SSteven Danz    }
760f3f0262cSandi    $del = true;
7613ce054b3Sandi    // autoset summary on deletion
7623ce054b3Sandi    if(empty($summary)) $summary = $lang['deleted'];
76353d6ccfeSandi    // remove empty namespaces
764cc7d0c94SBen Coburn    io_sweepNS($id, 'datadir');
765cc7d0c94SBen Coburn    io_sweepNS($id, 'mediadir');
766f3f0262cSandi  }else{
767cc7d0c94SBen Coburn    // save file (namespace dir is created in io_writeWikiPage)
768cc7d0c94SBen Coburn    io_writeWikiPage($file, $text, $id);
76946844156SBen Coburn    // pre-save the revision, to keep the attic in sync
77046844156SBen Coburn    $newRev = saveOldRevision($id);
771f3f0262cSandi    $del = false;
772f3f0262cSandi  }
773f3f0262cSandi
77471726d78SBen Coburn  // select changelog line type
77571726d78SBen Coburn  $extra = '';
776ebf1501fSBen Coburn  $type = DOKU_CHANGE_TYPE_EDIT;
77771726d78SBen Coburn  if ($wasReverted) {
778ebf1501fSBen Coburn    $type = DOKU_CHANGE_TYPE_REVERT;
77971726d78SBen Coburn    $extra = $REV;
78071726d78SBen Coburn  }
781ebf1501fSBen Coburn  else if ($wasCreated) { $type = DOKU_CHANGE_TYPE_CREATE; }
782ebf1501fSBen Coburn  else if ($wasRemoved) { $type = DOKU_CHANGE_TYPE_DELETE; }
783ebf1501fSBen Coburn  else if ($minor && $conf['useacl'] && $_SERVER['REMOTE_USER']) { $type = DOKU_CHANGE_TYPE_MINOR_EDIT; } //minor edits only for logged in users
78471726d78SBen Coburn
785e45b34cdSBen Coburn  addLogEntry($newRev, $id, $type, $summary, $extra);
78626a0801fSAndreas Gohr  // send notify mails
78790033e9dSAndreas Gohr  notify($id,'admin',$old,$summary,$minor);
78890033e9dSAndreas Gohr  notify($id,'subscribers',$old,$summary,$minor);
789f3f0262cSandi
790ce6b63d9Schris  // update the purgefile (timestamp of the last time anything within the wiki was changed)
79198407a7aSandi  io_saveFile($conf['cachedir'].'/purgefile',time());
792f3f0262cSandi}
793f3f0262cSandi
794f3f0262cSandi/**
795f3f0262cSandi * moves the current version to the attic and returns its
796f3f0262cSandi * revision date
79715fae107Sandi *
79815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
799f3f0262cSandi */
800f3f0262cSandifunction saveOldRevision($id){
801f3f0262cSandi  global $conf;
802f3f0262cSandi  $oldf = wikiFN($id);
803f3f0262cSandi  if(!@file_exists($oldf)) return '';
804f3f0262cSandi  $date = filemtime($oldf);
805f3f0262cSandi  $newf = wikiFN($id,$date);
806cc7d0c94SBen Coburn  io_writeWikiPage($newf, rawWiki($id), $id, $date);
807f3f0262cSandi  return $date;
808f3f0262cSandi}
809f3f0262cSandi
810f3f0262cSandi/**
81126a0801fSAndreas Gohr * Sends a notify mail on page change
81226a0801fSAndreas Gohr *
81326a0801fSAndreas Gohr * @param  string  $id       The changed page
81426a0801fSAndreas Gohr * @param  string  $who      Who to notify (admin|subscribers)
81526a0801fSAndreas Gohr * @param  int     $rev      Old page revision
81626a0801fSAndreas Gohr * @param  string  $summary  What changed
81790033e9dSAndreas Gohr * @param  boolean $minor    Is this a minor edit?
81802a498e7Schris * @param  array   $replace  Additional string substitutions, @KEY@ to be replaced by value
81915fae107Sandi *
82015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
821f3f0262cSandi */
82202a498e7Schrisfunction notify($id,$who,$rev='',$summary='',$minor=false,$replace=array()){
823f3f0262cSandi  global $lang;
824f3f0262cSandi  global $conf;
82530d7d718SMike Frysinger  global $INFO;
826b158d625SSteven Danz
82726a0801fSAndreas Gohr  // decide if there is something to do
82826a0801fSAndreas Gohr  if($who == 'admin'){
82926a0801fSAndreas Gohr    if(empty($conf['notify'])) return; //notify enabled?
830f3f0262cSandi    $text = rawLocale('mailtext');
83126a0801fSAndreas Gohr    $to   = $conf['notify'];
83226a0801fSAndreas Gohr    $bcc  = '';
83326a0801fSAndreas Gohr  }elseif($who == 'subscribers'){
83426a0801fSAndreas Gohr    if(!$conf['subscribers']) return; //subscribers enabled?
83590033e9dSAndreas Gohr    if($conf['useacl'] && $_SERVER['REMOTE_USER'] && $minor) return; //skip minors
83626a0801fSAndreas Gohr    $bcc  = subscriber_addresslist($id);
83726a0801fSAndreas Gohr    if(empty($bcc)) return;
83826a0801fSAndreas Gohr    $to   = '';
83926a0801fSAndreas Gohr    $text = rawLocale('subscribermail');
840a06e4bdbSSebastian Harl  }elseif($who == 'register'){
841a06e4bdbSSebastian Harl    if(empty($conf['registernotify'])) return;
842a06e4bdbSSebastian Harl    $text = rawLocale('registermail');
843a06e4bdbSSebastian Harl    $to   = $conf['registernotify'];
844a06e4bdbSSebastian Harl    $bcc  = '';
84526a0801fSAndreas Gohr  }else{
84626a0801fSAndreas Gohr    return; //just to be safe
84726a0801fSAndreas Gohr  }
84826a0801fSAndreas Gohr
849f3f0262cSandi  $text = str_replace('@DATE@',date($conf['dformat']),$text);
850f3f0262cSandi  $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text);
851f3f0262cSandi  $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text);
852f3f0262cSandi  $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text);
853*c9321d91SAndreas Gohr  $text = str_replace('@NEWPAGE@',wl($id,'',true,'&'),$text);
85426a0801fSAndreas Gohr  $text = str_replace('@PAGE@',$id,$text);
85526a0801fSAndreas Gohr  $text = str_replace('@TITLE@',$conf['title'],$text);
856ed7b5f09Sandi  $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text);
857f3f0262cSandi  $text = str_replace('@SUMMARY@',$summary,$text);
8587a82afdcSandi  $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text);
859f3f0262cSandi
86002a498e7Schris  foreach ($replace as $key => $substitution) {
86102a498e7Schris    $text = str_replace('@'.strtoupper($key).'@',$substitution, $text);
86202a498e7Schris  }
86302a498e7Schris
864a06e4bdbSSebastian Harl  if($who == 'register'){
865a06e4bdbSSebastian Harl    $subject = $lang['mail_new_user'].' '.$summary;
866a06e4bdbSSebastian Harl  }elseif($rev){
867f3f0262cSandi    $subject = $lang['mail_changed'].' '.$id;
868*c9321d91SAndreas Gohr    $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true,'&'),$text);
869ccdfa6c0SAndreas Gohr    require_once(DOKU_INC.'inc/DifferenceEngine.php');
870f3f0262cSandi    $df  = new Diff(split("\n",rawWiki($id,$rev)),
871f3f0262cSandi                    split("\n",rawWiki($id)));
872f3f0262cSandi    $dformat = new UnifiedDiffFormatter();
873f3f0262cSandi    $diff    = $dformat->format($df);
874f3f0262cSandi  }else{
875f3f0262cSandi    $subject=$lang['mail_newpage'].' '.$id;
876f3f0262cSandi    $text = str_replace('@OLDPAGE@','none',$text);
877f3f0262cSandi    $diff = rawWiki($id);
878f3f0262cSandi  }
879f3f0262cSandi  $text = str_replace('@DIFF@',$diff,$text);
880241f3a36Sandi  $subject = '['.$conf['title'].'] '.$subject;
881f3f0262cSandi
88230d7d718SMike Frysinger  $from = $conf['mailfrom'];
88330d7d718SMike Frysinger  $from = str_replace('@USER@',$_SERVER['REMOTE_USER'],$from);
88430d7d718SMike Frysinger  $from = str_replace('@NAME@',$INFO['userinfo']['name'],$from);
88530d7d718SMike Frysinger  $from = str_replace('@MAIL@',$INFO['userinfo']['mail'],$from);
88630d7d718SMike Frysinger
88730d7d718SMike Frysinger  mail_send($to,$subject,$text,$from,'',$bcc);
888f3f0262cSandi}
889f3f0262cSandi
89015fae107Sandi/**
891f3f0262cSandi * extracts the query from a google referer
89215fae107Sandi *
8936b13307fSandi * @todo   should be more generic and support yahoo et al
89415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
895f3f0262cSandi */
896f3f0262cSandifunction getGoogleQuery(){
897f3f0262cSandi  $url = parse_url($_SERVER['HTTP_REFERER']);
8985c3f206fSandi  if(!$url) return '';
899f3f0262cSandi
900f3f0262cSandi  if(!preg_match("#google\.#i",$url['host'])) return '';
901f3f0262cSandi  $query = array();
902f3f0262cSandi  parse_str($url['query'],$query);
903f3f0262cSandi
904f3f0262cSandi  return $query['q'];
905f3f0262cSandi}
906f3f0262cSandi
907f3f0262cSandi/**
90815fae107Sandi * Try to set correct locale
90915fae107Sandi *
910095bfd5cSandi * @deprecated No longer used
91115fae107Sandi * @author     Andreas Gohr <andi@splitbrain.org>
912f3f0262cSandi */
913f3f0262cSandifunction setCorrectLocale(){
914f3f0262cSandi  global $conf;
915f3f0262cSandi  global $lang;
916f3f0262cSandi
917f3f0262cSandi  $enc = strtoupper($lang['encoding']);
918f3f0262cSandi  foreach ($lang['locales'] as $loc){
919f3f0262cSandi    //try locale
920f3f0262cSandi    if(@setlocale(LC_ALL,$loc)) return;
921f3f0262cSandi    //try loceale with encoding
922f3f0262cSandi    if(@setlocale(LC_ALL,"$loc.$enc")) return;
923f3f0262cSandi  }
924f3f0262cSandi  //still here? try to set from environment
925f3f0262cSandi  @setlocale(LC_ALL,"");
926f3f0262cSandi}
927f3f0262cSandi
928f3f0262cSandi/**
929f3f0262cSandi * Return the human readable size of a file
930f3f0262cSandi *
931f3f0262cSandi * @param       int    $size   A file size
932f3f0262cSandi * @param       int    $dec    A number of decimal places
933f3f0262cSandi * @author      Martin Benjamin <b.martin@cybernet.ch>
934f3f0262cSandi * @author      Aidan Lister <aidan@php.net>
935f3f0262cSandi * @version     1.0.0
936f3f0262cSandi */
937f31d5b73Sandifunction filesize_h($size, $dec = 1){
938f3f0262cSandi  $sizes = array('B', 'KB', 'MB', 'GB');
939f3f0262cSandi  $count = count($sizes);
940f3f0262cSandi  $i = 0;
941f3f0262cSandi
942f3f0262cSandi  while ($size >= 1024 && ($i < $count - 1)) {
943f3f0262cSandi    $size /= 1024;
944f3f0262cSandi    $i++;
945f3f0262cSandi  }
946f3f0262cSandi
947f3f0262cSandi  return round($size, $dec) . ' ' . $sizes[$i];
948f3f0262cSandi}
949f3f0262cSandi
95015fae107Sandi/**
95100a7b5adSEsther Brunner * return an obfuscated email address in line with $conf['mailguard'] setting
95200a7b5adSEsther Brunner *
95300a7b5adSEsther Brunner * @author Harry Fuecks <hfuecks@gmail.com>
95400a7b5adSEsther Brunner * @author Christopher Smith <chris@jalakai.co.uk>
95500a7b5adSEsther Brunner */
95600a7b5adSEsther Brunnerfunction obfuscate($email) {
95700a7b5adSEsther Brunner  global $conf;
95800a7b5adSEsther Brunner
95900a7b5adSEsther Brunner  switch ($conf['mailguard']) {
96000a7b5adSEsther Brunner    case 'visible' :
96100a7b5adSEsther Brunner      $obfuscate = array('@' => ' [at] ', '.' => ' [dot] ', '-' => ' [dash] ');
96200a7b5adSEsther Brunner      return strtr($email, $obfuscate);
96300a7b5adSEsther Brunner
96400a7b5adSEsther Brunner    case 'hex' :
96500a7b5adSEsther Brunner      $encode = '';
96600a7b5adSEsther Brunner      for ($x=0; $x < strlen($email); $x++) $encode .= '&#x' . bin2hex($email{$x}).';';
96700a7b5adSEsther Brunner      return $encode;
96800a7b5adSEsther Brunner
96900a7b5adSEsther Brunner    case 'none' :
97000a7b5adSEsther Brunner    default :
97100a7b5adSEsther Brunner      return $email;
97200a7b5adSEsther Brunner  }
97300a7b5adSEsther Brunner}
97400a7b5adSEsther Brunner
97500a7b5adSEsther Brunner/**
976b158d625SSteven Danz * Let us know if a user is tracking a page
977b158d625SSteven Danz *
9781380fc45SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
979b158d625SSteven Danz */
9801380fc45SAndreas Gohrfunction is_subscribed($id,$uid){
9811380fc45SAndreas Gohr  $file=metaFN($id,'.mlist');
9821380fc45SAndreas Gohr  if (@file_exists($file)) {
983b158d625SSteven Danz    $mlist = file($file);
9841380fc45SAndreas Gohr    $pos = array_search($uid."\n",$mlist);
9851380fc45SAndreas Gohr    return is_int($pos);
986b158d625SSteven Danz  }
9871380fc45SAndreas Gohr
988b158d625SSteven Danz  return false;
989b158d625SSteven Danz}
990340756e4Sandi
991f9eb5648Ssteven-danz/**
992f9eb5648Ssteven-danz * Return a string with the email addresses of all the
993f9eb5648Ssteven-danz * users subscribed to a page
994f9eb5648Ssteven-danz *
99526a0801fSAndreas Gohr * @author Steven Danz <steven-danz@kc.rr.com>
996f9eb5648Ssteven-danz */
997f9eb5648Ssteven-danzfunction subscriber_addresslist($id){
998f9eb5648Ssteven-danz  global $conf;
999cd52f92dSchris  global $auth;
1000f9eb5648Ssteven-danz
1001f9eb5648Ssteven-danz  $emails = '';
1002f9eb5648Ssteven-danz
100326a0801fSAndreas Gohr  if (!$conf['subscribers']) return;
100426a0801fSAndreas Gohr
1005f9eb5648Ssteven-danz  $mlist = array();
1006f9eb5648Ssteven-danz  $file=metaFN($id,'.mlist');
1007d8186216SBen Coburn  if (@file_exists($file)) {
1008f9eb5648Ssteven-danz    $mlist = file($file);
1009f9eb5648Ssteven-danz  }
1010f9eb5648Ssteven-danz  if(count($mlist) > 0) {
1011f9eb5648Ssteven-danz    foreach ($mlist as $who) {
1012f9eb5648Ssteven-danz      $who = rtrim($who);
1013cd52f92dSchris      $info = $auth->getUserData($who);
1014f9eb5648Ssteven-danz      $level = auth_aclcheck($id,$who,$info['grps']);
1015f9eb5648Ssteven-danz      if ($level >= AUTH_READ) {
1016f9eb5648Ssteven-danz        if (strcasecmp($info['mail'],$conf['notify']) != 0) {
1017f9eb5648Ssteven-danz          if (empty($emails)) {
1018f9eb5648Ssteven-danz            $emails = $info['mail'];
1019f9eb5648Ssteven-danz          } else {
1020f9eb5648Ssteven-danz            $emails = "$emails,".$info['mail'];
1021f9eb5648Ssteven-danz          }
1022f9eb5648Ssteven-danz        }
1023f9eb5648Ssteven-danz      }
1024f9eb5648Ssteven-danz    }
1025f9eb5648Ssteven-danz  }
1026f9eb5648Ssteven-danz
1027f9eb5648Ssteven-danz  return $emails;
1028f9eb5648Ssteven-danz}
1029f9eb5648Ssteven-danz
103089541d4bSAndreas Gohr/**
103189541d4bSAndreas Gohr * Removes quoting backslashes
103289541d4bSAndreas Gohr *
103389541d4bSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
103489541d4bSAndreas Gohr */
103589541d4bSAndreas Gohrfunction unslash($string,$char="'"){
103689541d4bSAndreas Gohr  return str_replace('\\'.$char,$char,$string);
103789541d4bSAndreas Gohr}
103889541d4bSAndreas Gohr
1039340756e4Sandi//Setup VIM: ex: et ts=2 enc=utf-8 :
1040