xref: /dokuwiki/inc/common.php (revision cd00a03445c6b5dcdaba4631150c3c7e1370f526)
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){
43d5197206Schris  for($i=0; $i<$intend; $i++) print ' ';
4402b0b681SAndreas Gohr  echo "$string\n";
4502b0b681SAndreas Gohr}
4602b0b681SAndreas Gohr
4702b0b681SAndreas Gohr/**
4802b0b681SAndreas Gohr * strips control characters (<32) from the given string
4902b0b681SAndreas Gohr *
5002b0b681SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
5102b0b681SAndreas Gohr */
5202b0b681SAndreas Gohrfunction stripctl($string){
5302b0b681SAndreas Gohr  return preg_replace('/[\x00-\x1F]+/s','',$string);
54d5197206Schris}
55d5197206Schris
56d5197206Schris/**
5715fae107Sandi * Return info about the current document as associative
58f3f0262cSandi * array.
5915fae107Sandi *
6015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
61f3f0262cSandi */
62f3f0262cSandifunction pageinfo(){
63f3f0262cSandi  global $ID;
64f3f0262cSandi  global $REV;
65f3f0262cSandi  global $USERINFO;
66f3f0262cSandi  global $conf;
67f3f0262cSandi
686afe8dcaSchris  // include ID & REV not redundant, as some parts of DokuWiki may temporarily change $ID, e.g. p_wiki_xhtml
696afe8dcaSchris  // FIXME ... perhaps it would be better to ensure the temporary changes weren't necessary
706afe8dcaSchris  $info['id'] = $ID;
716afe8dcaSchris  $info['rev'] = $REV;
726afe8dcaSchris
73f3f0262cSandi  if($_SERVER['REMOTE_USER']){
74f3f0262cSandi    $info['userinfo']   = $USERINFO;
75f3f0262cSandi    $info['perm']       = auth_quickaclcheck($ID);
761380fc45SAndreas Gohr    $info['subscribed'] = is_subscribed($ID,$_SERVER['REMOTE_USER']);
77ee4c4a1bSAndreas Gohr    $info['client']     = $_SERVER['REMOTE_USER'];
7817ee7f66SAndreas Gohr
79f8cc712eSAndreas Gohr    // set info about manager/admin status
80f8cc712eSAndreas Gohr    $info['isadmin']   = false;
81f8cc712eSAndreas Gohr    $info['ismanager'] = false;
82f8cc712eSAndreas Gohr    if($info['perm'] == AUTH_ADMIN){
83f8cc712eSAndreas Gohr      $info['isadmin']   = true;
84f8cc712eSAndreas Gohr      $info['ismanager'] = true;
85f8cc712eSAndreas Gohr    }elseif(auth_ismanager()){
86f8cc712eSAndreas Gohr      $info['ismanager'] = true;
87f8cc712eSAndreas Gohr    }
88f8cc712eSAndreas Gohr
8917ee7f66SAndreas Gohr    // if some outside auth were used only REMOTE_USER is set
9017ee7f66SAndreas Gohr    if(!$info['userinfo']['name']){
9117ee7f66SAndreas Gohr      $info['userinfo']['name'] = $_SERVER['REMOTE_USER'];
9217ee7f66SAndreas Gohr    }
93ee4c4a1bSAndreas Gohr
94f3f0262cSandi  }else{
95f3f0262cSandi    $info['perm']       = auth_aclcheck($ID,'',null);
961380fc45SAndreas Gohr    $info['subscribed'] = false;
97ee4c4a1bSAndreas Gohr    $info['client']     = clientIP(true);
98f3f0262cSandi  }
99f3f0262cSandi
100f3f0262cSandi  $info['namespace'] = getNS($ID);
101f3f0262cSandi  $info['locked']    = checklock($ID);
1022ca9d91cSBen Coburn  $info['filepath']  = realpath(wikiFN($ID));
1032ca9d91cSBen Coburn  $info['exists']    = @file_exists($info['filepath']);
1042ca9d91cSBen Coburn  if($REV){
1052ca9d91cSBen Coburn    //check if current revision was meant
1062ca9d91cSBen Coburn    if($info['exists'] && (@filemtime($info['filepath'])==$REV)){
1072ca9d91cSBen Coburn      $REV = '';
1082ca9d91cSBen Coburn    }else{
1092ca9d91cSBen Coburn      //really use old revision
110f3f0262cSandi      $info['filepath'] = realpath(wikiFN($ID,$REV));
111f3f0262cSandi      $info['exists']   = @file_exists($info['filepath']);
112f3f0262cSandi    }
113f3f0262cSandi  }
114c112d578Sandi  $info['rev'] = $REV;
115f3f0262cSandi  if($info['exists']){
116f3f0262cSandi    $info['writable'] = (is_writable($info['filepath']) &&
117f3f0262cSandi                         ($info['perm'] >= AUTH_EDIT));
118f3f0262cSandi  }else{
119f3f0262cSandi    $info['writable'] = ($info['perm'] >= AUTH_CREATE);
120f3f0262cSandi  }
121f3f0262cSandi  $info['editable']  = ($info['writable'] && empty($info['lock']));
122f3f0262cSandi  $info['lastmod']   = @filemtime($info['filepath']);
123f3f0262cSandi
12471726d78SBen Coburn  //load page meta data
12571726d78SBen Coburn  $info['meta'] = p_get_metadata($ID);
12671726d78SBen Coburn
127652610a2Sandi  //who's the editor
128652610a2Sandi  if($REV){
12971726d78SBen Coburn    $revinfo = getRevisionInfo($ID, $REV, 1024);
130652610a2Sandi  }else{
131*cd00a034SBen Coburn    if (isset($info['meta']['last_change'])) { $revinfo = $info['meta']['last_change']; }
132*cd00a034SBen Coburn    else {
133*cd00a034SBen Coburn      $revinfo = getRevisionInfo($ID, $info['lastmod'], 1024);
134*cd00a034SBen Coburn      // cache most recent changelog line in metadata if missing and still valid
135*cd00a034SBen Coburn      if ($revinfo!==false) {
136*cd00a034SBen Coburn        $info['meta']['last_change'] = $revinfo;
137*cd00a034SBen Coburn        p_set_metadata($ID, array('last_change' => $revinfo));
138*cd00a034SBen Coburn      }
139*cd00a034SBen Coburn    }
140*cd00a034SBen Coburn  }
141*cd00a034SBen Coburn  //and check for an external edit
142*cd00a034SBen Coburn  if($revinfo!==false && $revinfo['date']!=$info['lastmod']){
143*cd00a034SBen Coburn    // cached changelog line no longer valid
144*cd00a034SBen Coburn    $revinfo = false;
145*cd00a034SBen Coburn    $info['meta']['last_change'] = $revinfo;
146*cd00a034SBen 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.
15371726d78SBen Coburn  // Use $INFO['meta']['last_change']['type']==='e' 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
240a77f5846Sjan  $name = noNS($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    }
479041d1964SAndreas Gohr    if(preg_match('#('.join('|',$re).')#si',$text, $match=array())) {
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){
617f3f0262cSandi  $text = preg_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) {
73946844156SBen Coburn      addLogEntry($old, $id);
74046844156SBen Coburn      // send notify mails
74146844156SBen Coburn      notify($id,'admin',$oldRev,'',false);
74246844156SBen Coburn      notify($id,'subscribers',$oldRev,'',false);
74346844156SBen Coburn      // remove soon to be stale instructions
74446844156SBen Coburn      $cache = new cache_instructions($id, $file);
74546844156SBen Coburn      $cache->removeCache();
74646844156SBen Coburn    }
74746844156SBen Coburn  }
748f3f0262cSandi
74971726d78SBen Coburn  if ($wasRemoved){
750e45b34cdSBen Coburn    // pre-save deleted revision
751e45b34cdSBen Coburn    @touch($file);
75246844156SBen Coburn    clearstatcache();
753e45b34cdSBen Coburn    $newRev = saveOldRevision($id);
754e1f3d9e1SEsther Brunner    // remove empty file
755f3f0262cSandi    @unlink($file);
75671726d78SBen Coburn    // remove old meta info...
757e1f3d9e1SEsther Brunner    $mfiles = metaFiles($id);
75871726d78SBen Coburn    $changelog = metaFN($id, '.changes');
759e1f3d9e1SEsther Brunner    foreach ($mfiles as $mfile) {
76071726d78SBen Coburn      // but keep per-page changelog to preserve page history
761d8186216SBen Coburn      if (@file_exists($mfile) && $mfile!==$changelog) { @unlink($mfile); }
762b158d625SSteven Danz    }
763f3f0262cSandi    $del = true;
7643ce054b3Sandi    // autoset summary on deletion
7653ce054b3Sandi    if(empty($summary)) $summary = $lang['deleted'];
76653d6ccfeSandi    // remove empty namespaces
767cc7d0c94SBen Coburn    io_sweepNS($id, 'datadir');
768cc7d0c94SBen Coburn    io_sweepNS($id, 'mediadir');
769f3f0262cSandi  }else{
770cc7d0c94SBen Coburn    // save file (namespace dir is created in io_writeWikiPage)
771cc7d0c94SBen Coburn    io_writeWikiPage($file, $text, $id);
77246844156SBen Coburn    // pre-save the revision, to keep the attic in sync
77346844156SBen Coburn    $newRev = saveOldRevision($id);
774f3f0262cSandi    $del = false;
775f3f0262cSandi  }
776f3f0262cSandi
77771726d78SBen Coburn  // select changelog line type
77871726d78SBen Coburn  $extra = '';
77971726d78SBen Coburn  $type = 'E';
78071726d78SBen Coburn  if ($wasReverted) {
78171726d78SBen Coburn    $type = 'R';
78271726d78SBen Coburn    $extra = $REV;
78371726d78SBen Coburn  }
78471726d78SBen Coburn  else if ($wasCreated) { $type = 'C'; }
78571726d78SBen Coburn  else if ($wasRemoved) { $type = 'D'; }
78671726d78SBen Coburn  else if ($minor && $conf['useacl'] && $_SERVER['REMOTE_USER']) { $type = 'e'; } //minor edits only for logged in users
78771726d78SBen Coburn
788e45b34cdSBen Coburn  addLogEntry($newRev, $id, $type, $summary, $extra);
78926a0801fSAndreas Gohr  // send notify mails
79090033e9dSAndreas Gohr  notify($id,'admin',$old,$summary,$minor);
79190033e9dSAndreas Gohr  notify($id,'subscribers',$old,$summary,$minor);
792f3f0262cSandi
793ce6b63d9Schris  // update the purgefile (timestamp of the last time anything within the wiki was changed)
79498407a7aSandi  io_saveFile($conf['cachedir'].'/purgefile',time());
795f3f0262cSandi}
796f3f0262cSandi
797f3f0262cSandi/**
798f3f0262cSandi * moves the current version to the attic and returns its
799f3f0262cSandi * revision date
80015fae107Sandi *
80115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
802f3f0262cSandi */
803f3f0262cSandifunction saveOldRevision($id){
804f3f0262cSandi  global $conf;
805f3f0262cSandi  $oldf = wikiFN($id);
806f3f0262cSandi  if(!@file_exists($oldf)) return '';
807f3f0262cSandi  $date = filemtime($oldf);
808f3f0262cSandi  $newf = wikiFN($id,$date);
809cc7d0c94SBen Coburn  io_writeWikiPage($newf, rawWiki($id), $id, $date);
810f3f0262cSandi  return $date;
811f3f0262cSandi}
812f3f0262cSandi
813f3f0262cSandi/**
81426a0801fSAndreas Gohr * Sends a notify mail on page change
81526a0801fSAndreas Gohr *
81626a0801fSAndreas Gohr * @param  string  $id       The changed page
81726a0801fSAndreas Gohr * @param  string  $who      Who to notify (admin|subscribers)
81826a0801fSAndreas Gohr * @param  int     $rev      Old page revision
81926a0801fSAndreas Gohr * @param  string  $summary  What changed
82090033e9dSAndreas Gohr * @param  boolean $minor    Is this a minor edit?
82102a498e7Schris * @param  array   $replace  Additional string substitutions, @KEY@ to be replaced by value
82215fae107Sandi *
82315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
824f3f0262cSandi */
82502a498e7Schrisfunction notify($id,$who,$rev='',$summary='',$minor=false,$replace=array()){
826f3f0262cSandi  global $lang;
827f3f0262cSandi  global $conf;
82830d7d718SMike Frysinger  global $INFO;
829b158d625SSteven Danz
83026a0801fSAndreas Gohr  // decide if there is something to do
83126a0801fSAndreas Gohr  if($who == 'admin'){
83226a0801fSAndreas Gohr    if(empty($conf['notify'])) return; //notify enabled?
833f3f0262cSandi    $text = rawLocale('mailtext');
83426a0801fSAndreas Gohr    $to   = $conf['notify'];
83526a0801fSAndreas Gohr    $bcc  = '';
83626a0801fSAndreas Gohr  }elseif($who == 'subscribers'){
83726a0801fSAndreas Gohr    if(!$conf['subscribers']) return; //subscribers enabled?
83890033e9dSAndreas Gohr    if($conf['useacl'] && $_SERVER['REMOTE_USER'] && $minor) return; //skip minors
83926a0801fSAndreas Gohr    $bcc  = subscriber_addresslist($id);
84026a0801fSAndreas Gohr    if(empty($bcc)) return;
84126a0801fSAndreas Gohr    $to   = '';
84226a0801fSAndreas Gohr    $text = rawLocale('subscribermail');
843a06e4bdbSSebastian Harl  }elseif($who == 'register'){
844a06e4bdbSSebastian Harl    if(empty($conf['registernotify'])) return;
845a06e4bdbSSebastian Harl    $text = rawLocale('registermail');
846a06e4bdbSSebastian Harl    $to   = $conf['registernotify'];
847a06e4bdbSSebastian Harl    $bcc  = '';
84826a0801fSAndreas Gohr  }else{
84926a0801fSAndreas Gohr    return; //just to be safe
85026a0801fSAndreas Gohr  }
85126a0801fSAndreas Gohr
852f3f0262cSandi  $text = str_replace('@DATE@',date($conf['dformat']),$text);
853f3f0262cSandi  $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text);
854f3f0262cSandi  $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text);
855f3f0262cSandi  $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text);
856ed7b5f09Sandi  $text = str_replace('@NEWPAGE@',wl($id,'',true),$text);
85726a0801fSAndreas Gohr  $text = str_replace('@PAGE@',$id,$text);
85826a0801fSAndreas Gohr  $text = str_replace('@TITLE@',$conf['title'],$text);
859ed7b5f09Sandi  $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text);
860f3f0262cSandi  $text = str_replace('@SUMMARY@',$summary,$text);
8617a82afdcSandi  $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text);
862f3f0262cSandi
86302a498e7Schris  foreach ($replace as $key => $substitution) {
86402a498e7Schris    $text = str_replace('@'.strtoupper($key).'@',$substitution, $text);
86502a498e7Schris  }
86602a498e7Schris
867a06e4bdbSSebastian Harl  if($who == 'register'){
868a06e4bdbSSebastian Harl    $subject = $lang['mail_new_user'].' '.$summary;
869a06e4bdbSSebastian Harl  }elseif($rev){
870f3f0262cSandi    $subject = $lang['mail_changed'].' '.$id;
871ed7b5f09Sandi    $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true),$text);
872ccdfa6c0SAndreas Gohr    require_once(DOKU_INC.'inc/DifferenceEngine.php');
873f3f0262cSandi    $df  = new Diff(split("\n",rawWiki($id,$rev)),
874f3f0262cSandi                    split("\n",rawWiki($id)));
875f3f0262cSandi    $dformat = new UnifiedDiffFormatter();
876f3f0262cSandi    $diff    = $dformat->format($df);
877f3f0262cSandi  }else{
878f3f0262cSandi    $subject=$lang['mail_newpage'].' '.$id;
879f3f0262cSandi    $text = str_replace('@OLDPAGE@','none',$text);
880f3f0262cSandi    $diff = rawWiki($id);
881f3f0262cSandi  }
882f3f0262cSandi  $text = str_replace('@DIFF@',$diff,$text);
883241f3a36Sandi  $subject = '['.$conf['title'].'] '.$subject;
884f3f0262cSandi
88530d7d718SMike Frysinger  $from = $conf['mailfrom'];
88630d7d718SMike Frysinger  $from = str_replace('@USER@',$_SERVER['REMOTE_USER'],$from);
88730d7d718SMike Frysinger  $from = str_replace('@NAME@',$INFO['userinfo']['name'],$from);
88830d7d718SMike Frysinger  $from = str_replace('@MAIL@',$INFO['userinfo']['mail'],$from);
88930d7d718SMike Frysinger
89030d7d718SMike Frysinger  mail_send($to,$subject,$text,$from,'',$bcc);
891f3f0262cSandi}
892f3f0262cSandi
89315fae107Sandi/**
894f3f0262cSandi * extracts the query from a google referer
89515fae107Sandi *
8966b13307fSandi * @todo   should be more generic and support yahoo et al
89715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org>
898f3f0262cSandi */
899f3f0262cSandifunction getGoogleQuery(){
900f3f0262cSandi  $url = parse_url($_SERVER['HTTP_REFERER']);
9015c3f206fSandi  if(!$url) return '';
902f3f0262cSandi
903f3f0262cSandi  if(!preg_match("#google\.#i",$url['host'])) return '';
904f3f0262cSandi  $query = array();
905f3f0262cSandi  parse_str($url['query'],$query);
906f3f0262cSandi
907f3f0262cSandi  return $query['q'];
908f3f0262cSandi}
909f3f0262cSandi
910f3f0262cSandi/**
91115fae107Sandi * Try to set correct locale
91215fae107Sandi *
913095bfd5cSandi * @deprecated No longer used
91415fae107Sandi * @author     Andreas Gohr <andi@splitbrain.org>
915f3f0262cSandi */
916f3f0262cSandifunction setCorrectLocale(){
917f3f0262cSandi  global $conf;
918f3f0262cSandi  global $lang;
919f3f0262cSandi
920f3f0262cSandi  $enc = strtoupper($lang['encoding']);
921f3f0262cSandi  foreach ($lang['locales'] as $loc){
922f3f0262cSandi    //try locale
923f3f0262cSandi    if(@setlocale(LC_ALL,$loc)) return;
924f3f0262cSandi    //try loceale with encoding
925f3f0262cSandi    if(@setlocale(LC_ALL,"$loc.$enc")) return;
926f3f0262cSandi  }
927f3f0262cSandi  //still here? try to set from environment
928f3f0262cSandi  @setlocale(LC_ALL,"");
929f3f0262cSandi}
930f3f0262cSandi
931f3f0262cSandi/**
932f3f0262cSandi * Return the human readable size of a file
933f3f0262cSandi *
934f3f0262cSandi * @param       int    $size   A file size
935f3f0262cSandi * @param       int    $dec    A number of decimal places
936f3f0262cSandi * @author      Martin Benjamin <b.martin@cybernet.ch>
937f3f0262cSandi * @author      Aidan Lister <aidan@php.net>
938f3f0262cSandi * @version     1.0.0
939f3f0262cSandi */
940f31d5b73Sandifunction filesize_h($size, $dec = 1){
941f3f0262cSandi  $sizes = array('B', 'KB', 'MB', 'GB');
942f3f0262cSandi  $count = count($sizes);
943f3f0262cSandi  $i = 0;
944f3f0262cSandi
945f3f0262cSandi  while ($size >= 1024 && ($i < $count - 1)) {
946f3f0262cSandi    $size /= 1024;
947f3f0262cSandi    $i++;
948f3f0262cSandi  }
949f3f0262cSandi
950f3f0262cSandi  return round($size, $dec) . ' ' . $sizes[$i];
951f3f0262cSandi}
952f3f0262cSandi
95315fae107Sandi/**
95400a7b5adSEsther Brunner * return an obfuscated email address in line with $conf['mailguard'] setting
95500a7b5adSEsther Brunner *
95600a7b5adSEsther Brunner * @author Harry Fuecks <hfuecks@gmail.com>
95700a7b5adSEsther Brunner * @author Christopher Smith <chris@jalakai.co.uk>
95800a7b5adSEsther Brunner */
95900a7b5adSEsther Brunnerfunction obfuscate($email) {
96000a7b5adSEsther Brunner  global $conf;
96100a7b5adSEsther Brunner
96200a7b5adSEsther Brunner  switch ($conf['mailguard']) {
96300a7b5adSEsther Brunner    case 'visible' :
96400a7b5adSEsther Brunner      $obfuscate = array('@' => ' [at] ', '.' => ' [dot] ', '-' => ' [dash] ');
96500a7b5adSEsther Brunner      return strtr($email, $obfuscate);
96600a7b5adSEsther Brunner
96700a7b5adSEsther Brunner    case 'hex' :
96800a7b5adSEsther Brunner      $encode = '';
96900a7b5adSEsther Brunner      for ($x=0; $x < strlen($email); $x++) $encode .= '&#x' . bin2hex($email{$x}).';';
97000a7b5adSEsther Brunner      return $encode;
97100a7b5adSEsther Brunner
97200a7b5adSEsther Brunner    case 'none' :
97300a7b5adSEsther Brunner    default :
97400a7b5adSEsther Brunner      return $email;
97500a7b5adSEsther Brunner  }
97600a7b5adSEsther Brunner}
97700a7b5adSEsther Brunner
97800a7b5adSEsther Brunner/**
979b158d625SSteven Danz * Let us know if a user is tracking a page
980b158d625SSteven Danz *
9811380fc45SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
982b158d625SSteven Danz */
9831380fc45SAndreas Gohrfunction is_subscribed($id,$uid){
9841380fc45SAndreas Gohr  $file=metaFN($id,'.mlist');
9851380fc45SAndreas Gohr  if (@file_exists($file)) {
986b158d625SSteven Danz    $mlist = file($file);
9871380fc45SAndreas Gohr    $pos = array_search($uid."\n",$mlist);
9881380fc45SAndreas Gohr    return is_int($pos);
989b158d625SSteven Danz  }
9901380fc45SAndreas Gohr
991b158d625SSteven Danz  return false;
992b158d625SSteven Danz}
993340756e4Sandi
994f9eb5648Ssteven-danz/**
995f9eb5648Ssteven-danz * Return a string with the email addresses of all the
996f9eb5648Ssteven-danz * users subscribed to a page
997f9eb5648Ssteven-danz *
99826a0801fSAndreas Gohr * @author Steven Danz <steven-danz@kc.rr.com>
999f9eb5648Ssteven-danz */
1000f9eb5648Ssteven-danzfunction subscriber_addresslist($id){
1001f9eb5648Ssteven-danz  global $conf;
1002cd52f92dSchris  global $auth;
1003f9eb5648Ssteven-danz
1004f9eb5648Ssteven-danz  $emails = '';
1005f9eb5648Ssteven-danz
100626a0801fSAndreas Gohr  if (!$conf['subscribers']) return;
100726a0801fSAndreas Gohr
1008f9eb5648Ssteven-danz  $mlist = array();
1009f9eb5648Ssteven-danz  $file=metaFN($id,'.mlist');
1010d8186216SBen Coburn  if (@file_exists($file)) {
1011f9eb5648Ssteven-danz    $mlist = file($file);
1012f9eb5648Ssteven-danz  }
1013f9eb5648Ssteven-danz  if(count($mlist) > 0) {
1014f9eb5648Ssteven-danz    foreach ($mlist as $who) {
1015f9eb5648Ssteven-danz      $who = rtrim($who);
1016cd52f92dSchris      $info = $auth->getUserData($who);
1017f9eb5648Ssteven-danz      $level = auth_aclcheck($id,$who,$info['grps']);
1018f9eb5648Ssteven-danz      if ($level >= AUTH_READ) {
1019f9eb5648Ssteven-danz        if (strcasecmp($info['mail'],$conf['notify']) != 0) {
1020f9eb5648Ssteven-danz          if (empty($emails)) {
1021f9eb5648Ssteven-danz            $emails = $info['mail'];
1022f9eb5648Ssteven-danz          } else {
1023f9eb5648Ssteven-danz            $emails = "$emails,".$info['mail'];
1024f9eb5648Ssteven-danz          }
1025f9eb5648Ssteven-danz        }
1026f9eb5648Ssteven-danz      }
1027f9eb5648Ssteven-danz    }
1028f9eb5648Ssteven-danz  }
1029f9eb5648Ssteven-danz
1030f9eb5648Ssteven-danz  return $emails;
1031f9eb5648Ssteven-danz}
1032f9eb5648Ssteven-danz
103389541d4bSAndreas Gohr/**
103489541d4bSAndreas Gohr * Removes quoting backslashes
103589541d4bSAndreas Gohr *
103689541d4bSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org>
103789541d4bSAndreas Gohr */
103889541d4bSAndreas Gohrfunction unslash($string,$char="'"){
103989541d4bSAndreas Gohr  return str_replace('\\'.$char,$char,$string);
104089541d4bSAndreas Gohr}
104189541d4bSAndreas Gohr
1042340756e4Sandi//Setup VIM: ex: et ts=2 enc=utf-8 :
1043