xref: /dokuwiki/inc/parserutils.php (revision c90b2fb183974c22359b17366599543007f0eea3)
1c112d578Sandi<?php
2c112d578Sandi/**
3c112d578Sandi * Utilities for collecting data from config files
4c112d578Sandi *
5c112d578Sandi * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
6c112d578Sandi * @author     Harry Fuecks <hfuecks@gmail.com>
7c112d578Sandi * @author     Andreas Gohr <andi@splitbrain.org>
8c112d578Sandi */
9c112d578Sandi
10c112d578Sandi  if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/');
11c112d578Sandi
12c112d578Sandi  require_once(DOKU_INC.'inc/confutils.php');
13c112d578Sandi  require_once(DOKU_INC.'inc/pageutils.php');
14ee20e7d1Sandi  require_once(DOKU_INC.'inc/pluginutils.php');
15c112d578Sandi
16c112d578Sandi/**
17c112d578Sandi * Returns the parsed Wikitext in XHTML for the given id and revision.
18c112d578Sandi *
19c112d578Sandi * If $excuse is true an explanation is returned if the file
20c112d578Sandi * wasn't found
21c112d578Sandi *
22c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
23c112d578Sandi */
24c112d578Sandifunction p_wiki_xhtml($id, $rev='', $excuse=true){
25c112d578Sandi  $file = wikiFN($id,$rev);
26c112d578Sandi  $ret  = '';
27c112d578Sandi
28c112d578Sandi  //ensure $id is in global $ID (needed for parsing)
291e76272cSandi  global $ID;
301e76272cSandi  $ID = $id;
31c112d578Sandi
32c112d578Sandi  if($rev){
33c112d578Sandi    if(@file_exists($file)){
349dc2c2afSandi      $ret = p_render('xhtml',p_get_instructions(io_readfile($file)),$info); //no caching on old revisions
35c112d578Sandi    }elseif($excuse){
36c112d578Sandi      $ret = p_locale_xhtml('norev');
37c112d578Sandi    }
38c112d578Sandi  }else{
39c112d578Sandi    if(@file_exists($file)){
40c112d578Sandi      $ret = p_cached_xhtml($file);
41c112d578Sandi    }elseif($excuse){
42c112d578Sandi      $ret = p_locale_xhtml('newpage');
43c112d578Sandi    }
44c112d578Sandi  }
45c112d578Sandi
46c112d578Sandi  return $ret;
47c112d578Sandi}
48c112d578Sandi
49c112d578Sandi/**
50c112d578Sandi * Returns the specified local text in parsed format
51c112d578Sandi *
52c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
53c112d578Sandi */
54c112d578Sandifunction p_locale_xhtml($id){
55c112d578Sandi  //fetch parsed locale
56c112d578Sandi  $html = p_cached_xhtml(localeFN($id));
57c112d578Sandi  return $html;
58c112d578Sandi}
59c112d578Sandi
60c112d578Sandi/**
61c112d578Sandi * Returns the given file parsed to XHTML
62c112d578Sandi *
63c112d578Sandi * Uses and creates a cachefile
64c112d578Sandi *
65c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
669dc2c2afSandi * @todo   rewrite to use mode instead of hardcoded XHTML
67c112d578Sandi */
68c112d578Sandifunction p_cached_xhtml($file){
69c112d578Sandi  global $conf;
7098407a7aSandi  $cache  = getCacheName($file.$_SERVER['HTTP_HOST'].$_SERVER['SERVER_PORT'],'.xhtml');
71c7532350SDrew Amato  $purge  = $conf['cachedir'].'/purgefile';
72c112d578Sandi
73c112d578Sandi  // check if cache can be used
74c112d578Sandi  $cachetime = @filemtime($cache); // 0 if not exists
75c112d578Sandi
76c112d578Sandi  if( @file_exists($file)                                             // does the source exist
77c112d578Sandi      && $cachetime > @filemtime($file)                               // cache is fresh
78c112d578Sandi      && ((time() - $cachetime) < $conf['cachetime'])                 // and is cachefile young enough
79c112d578Sandi      && !isset($_REQUEST['purge'])                                   // no purge param was set
801094c798Sandi      && ($cachetime > @filemtime($purge))                            // and newer than the purgefile
81e7cb32dcSAndreas Gohr      && ($cachetime > @filemtime(DOKU_CONF.'dokuwiki.php'))      // newer than the config file
82e7cb32dcSAndreas Gohr      && ($cachetime > @filemtime(DOKU_CONF.'local.php'))         // newer than the local config file
83c112d578Sandi      && ($cachetime > @filemtime(DOKU_INC.'inc/parser/xhtml.php'))   // newer than the renderer
84c112d578Sandi      && ($cachetime > @filemtime(DOKU_INC.'inc/parser/parser.php'))  // newer than the parser
85c112d578Sandi      && ($cachetime > @filemtime(DOKU_INC.'inc/parser/handler.php')))// newer than the handler
86c112d578Sandi  {
87c112d578Sandi    //well then use the cache
88c112d578Sandi    $parsed = io_readfile($cache);
89c112d578Sandi    $parsed .= "\n<!-- cachefile $cache used -->\n";
90c112d578Sandi  }else{
919dc2c2afSandi    $parsed = p_render('xhtml', p_cached_instructions($file),$info); //try to use cached instructions
92c112d578Sandi
939dc2c2afSandi    if($info['cache']){
94c112d578Sandi      io_saveFile($cache,$parsed); //save cachefile
95c112d578Sandi      $parsed .= "\n<!-- no cachefile used, but created -->\n";
96c112d578Sandi    }else{
97c112d578Sandi      @unlink($cache); //try to delete cachefile
98c112d578Sandi      $parsed .= "\n<!-- no cachefile used, caching forbidden -->\n";
99c112d578Sandi    }
100c112d578Sandi  }
101c112d578Sandi
102c112d578Sandi  return $parsed;
103c112d578Sandi}
104c112d578Sandi
105c112d578Sandi/**
106c112d578Sandi * Returns the render instructions for a file
107c112d578Sandi *
108c112d578Sandi * Uses and creates a serialized cache file
109c112d578Sandi *
110c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
111c112d578Sandi */
11237e34a5eSandifunction p_cached_instructions($file,$cacheonly=false){
113c112d578Sandi  global $conf;
11498407a7aSandi  $cache  = getCacheName($file.$_SERVER['HTTP_HOST'].$_SERVER['SERVER_PORT'],'.i');
115c112d578Sandi
116c112d578Sandi  // check if cache can be used
117c112d578Sandi  $cachetime = @filemtime($cache); // 0 if not exists
118c112d578Sandi
11937e34a5eSandi  // cache forced?
12037e34a5eSandi  if($cacheonly){
12137e34a5eSandi    if($cachetime){
12237e34a5eSandi      return unserialize(io_readfile($cache));
12337e34a5eSandi    }else{
124fd198316Sandi      return array();
12537e34a5eSandi    }
12637e34a5eSandi  }
12737e34a5eSandi
128c112d578Sandi  if( @file_exists($file)                                             // does the source exist
129c112d578Sandi      && $cachetime > @filemtime($file)                               // cache is fresh
130c112d578Sandi      && !isset($_REQUEST['purge'])                                   // no purge param was set
131e7cb32dcSAndreas Gohr      && ($cachetime > @filemtime(DOKU_CONF.'dokuwiki.php'))      // newer than the config file
132e7cb32dcSAndreas Gohr      && ($cachetime > @filemtime(DOKU_CONF.'local.php'))         // newer than the local config file
133c112d578Sandi      && ($cachetime > @filemtime(DOKU_INC.'inc/parser/parser.php'))  // newer than the parser
134c112d578Sandi      && ($cachetime > @filemtime(DOKU_INC.'inc/parser/handler.php')))// newer than the handler
135c112d578Sandi  {
136c112d578Sandi    //well then use the cache
137c112d578Sandi    return unserialize(io_readfile($cache));
138c112d578Sandi  }elseif(@file_exists($file)){
139c112d578Sandi    // no cache - do some work
1406bbae538Sandi    $ins = p_get_instructions(io_readfile($file));
141c112d578Sandi    io_savefile($cache,serialize($ins));
142c112d578Sandi    return $ins;
143c112d578Sandi  }
144c112d578Sandi
145c112d578Sandi  return NULL;
146c112d578Sandi}
147c112d578Sandi
148c112d578Sandi/**
149c112d578Sandi * turns a page into a list of instructions
150c112d578Sandi *
151c112d578Sandi * @author Harry Fuecks <hfuecks@gmail.com>
152c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
153c112d578Sandi */
1546bbae538Sandifunction p_get_instructions($text){
155c112d578Sandi
156107b01d6Sandi  $modes = p_get_parsermodes();
157ee20e7d1Sandi
158c112d578Sandi  // Create the parser
159c112d578Sandi  $Parser = & new Doku_Parser();
160c112d578Sandi
161c112d578Sandi  // Add the Handler
162c112d578Sandi  $Parser->Handler = & new Doku_Handler();
163c112d578Sandi
164107b01d6Sandi  //add modes to parser
165107b01d6Sandi  foreach($modes as $mode){
166107b01d6Sandi    $Parser->addMode($mode['mode'],$mode['obj']);
167c112d578Sandi  }
168c112d578Sandi
169c112d578Sandi  // Do the parsing
170a2d649c4Sandi  $p    = $Parser->parse($text);
171ee20e7d1Sandi//  dbg($p);
172a2d649c4Sandi  return $p;
173c112d578Sandi}
174c112d578Sandi
175c112d578Sandi/**
176107b01d6Sandi * returns all available parser syntax modes in correct order
177107b01d6Sandi *
178107b01d6Sandi * @author Andreas Gohr <andi@splitbrain.org>
179107b01d6Sandi */
180107b01d6Sandifunction p_get_parsermodes(){
181107b01d6Sandi  global $conf;
182107b01d6Sandi
183107b01d6Sandi  //reuse old data
184107b01d6Sandi  static $modes = null;
185107b01d6Sandi  if($modes != null){
186107b01d6Sandi    return $modes;
187107b01d6Sandi  }
188107b01d6Sandi
189107b01d6Sandi  //import parser classes and mode definitions
190107b01d6Sandi  require_once DOKU_INC . 'inc/parser/parser.php';
191107b01d6Sandi
192107b01d6Sandi  // we now collect all syntax modes and their objects, then they will
193107b01d6Sandi  // be sorted and added to the parser in correct order
194107b01d6Sandi  $modes = array();
195107b01d6Sandi
196107b01d6Sandi  // add syntax plugins
197107b01d6Sandi  $pluginlist = plugin_list('syntax');
198107b01d6Sandi  if(count($pluginlist)){
199107b01d6Sandi    global $PARSER_MODES;
200107b01d6Sandi    $obj = null;
201107b01d6Sandi    foreach($pluginlist as $p){
202*c90b2fb1Schris      if(!$obj =& plugin_load('syntax',$p)) continue; //attempt to load plugin into $obj
203107b01d6Sandi      $PARSER_MODES[$obj->getType()][] = "plugin_$p"; //register mode type
204107b01d6Sandi      //add to modes
205107b01d6Sandi      $modes[] = array(
206107b01d6Sandi                   'sort' => $obj->getSort(),
207107b01d6Sandi                   'mode' => "plugin_$p",
208107b01d6Sandi                   'obj'  => $obj,
209107b01d6Sandi                 );
210a46d0d65SAndreas Gohr      unset($obj); //remove the reference
211107b01d6Sandi    }
212107b01d6Sandi  }
213107b01d6Sandi
214107b01d6Sandi  // add default modes
215107b01d6Sandi  $std_modes = array('listblock','preformatted','notoc','nocache',
216107b01d6Sandi                     'header','table','linebreak','footnote','hr',
217107b01d6Sandi                     'unformatted','php','html','code','file','quote',
218107b01d6Sandi                     'multiplyentity','quotes','internallink','rss',
219107b01d6Sandi                     'media','externallink','emaillink','windowssharelink',
220107b01d6Sandi                     'eol');
221107b01d6Sandi  foreach($std_modes as $m){
222107b01d6Sandi    $class = "Doku_Parser_Mode_$m";
223107b01d6Sandi    $obj   = new $class();
224107b01d6Sandi    $modes[] = array(
225107b01d6Sandi                 'sort' => $obj->getSort(),
226107b01d6Sandi                 'mode' => $m,
227107b01d6Sandi                 'obj'  => $obj
228107b01d6Sandi               );
229107b01d6Sandi  }
230107b01d6Sandi
231107b01d6Sandi  // add formatting modes
232107b01d6Sandi  $fmt_modes = array('strong','emphasis','underline','monospace',
233107b01d6Sandi                     'subscript','superscript','deleted');
234107b01d6Sandi  foreach($fmt_modes as $m){
235107b01d6Sandi    $obj   = new Doku_Parser_Mode_formatting($m);
236107b01d6Sandi    $modes[] = array(
237107b01d6Sandi                 'sort' => $obj->getSort(),
238107b01d6Sandi                 'mode' => $m,
239107b01d6Sandi                 'obj'  => $obj
240107b01d6Sandi               );
241107b01d6Sandi  }
242107b01d6Sandi
243107b01d6Sandi  // add modes which need files
244107b01d6Sandi  $obj     = new Doku_Parser_Mode_smiley(array_keys(getSmileys()));
245107b01d6Sandi  $modes[] = array('sort' => $obj->getSort(), 'mode' => 'smiley','obj'  => $obj );
246107b01d6Sandi  $obj     = new Doku_Parser_Mode_acronym(array_keys(getAcronyms()));
247107b01d6Sandi  $modes[] = array('sort' => $obj->getSort(), 'mode' => 'acronym','obj'  => $obj );
248107b01d6Sandi  $obj     = new Doku_Parser_Mode_entity(array_keys(getEntities()));
249107b01d6Sandi  $modes[] = array('sort' => $obj->getSort(), 'mode' => 'entity','obj'  => $obj );
250107b01d6Sandi
251107b01d6Sandi
252107b01d6Sandi  // add optional camelcase mode
253107b01d6Sandi  if($conf['camelcase']){
254107b01d6Sandi    $obj     = new Doku_Parser_Mode_camelcaselink();
255107b01d6Sandi    $modes[] = array('sort' => $obj->getSort(), 'mode' => 'camelcaselink','obj'  => $obj );
256107b01d6Sandi  }
257107b01d6Sandi
258107b01d6Sandi  //sort modes
259107b01d6Sandi  usort($modes,'p_sort_modes');
260107b01d6Sandi
261107b01d6Sandi  return $modes;
262107b01d6Sandi}
263107b01d6Sandi
264107b01d6Sandi/**
265107b01d6Sandi * Callback function for usort
266107b01d6Sandi *
267107b01d6Sandi * @author Andreas Gohr <andi@splitbrain.org>
268107b01d6Sandi */
269107b01d6Sandifunction p_sort_modes($a, $b){
270107b01d6Sandi  if($a['sort'] == $b['sort']) return 0;
271107b01d6Sandi  return ($a['sort'] < $b['sort']) ? -1 : 1;
272107b01d6Sandi}
273107b01d6Sandi
274107b01d6Sandi/**
275ac83b9d8Sandi * Renders a list of instruction to the specified output mode
276c112d578Sandi *
2779dc2c2afSandi * In the $info array are informations from the renderer returned
2789dc2c2afSandi *
279c112d578Sandi * @author Harry Fuecks <hfuecks@gmail.com>
280c112d578Sandi * @author Andreas Gohr <andi@splitbrain.org>
281c112d578Sandi */
2829dc2c2afSandifunction p_render($mode,$instructions,& $info){
283c112d578Sandi  if(is_null($instructions)) return '';
284c112d578Sandi
285c112d578Sandi  // Create the renderer
286ac83b9d8Sandi  if(!@file_exists(DOKU_INC."inc/parser/$mode.php")){
287ac83b9d8Sandi    msg("No renderer for $mode found",-1);
288ac83b9d8Sandi    return null;
289ac83b9d8Sandi  }
290ac83b9d8Sandi
291ac83b9d8Sandi  require_once DOKU_INC."inc/parser/$mode.php";
292ac83b9d8Sandi  $rclass = "Doku_Renderer_$mode";
293ac83b9d8Sandi  $Renderer = & new $rclass(); #FIXME any way to check for class existance?
294c112d578Sandi
295c112d578Sandi  $Renderer->smileys = getSmileys();
296c112d578Sandi  $Renderer->entities = getEntities();
297c112d578Sandi  $Renderer->acronyms = getAcronyms();
298c112d578Sandi  $Renderer->interwiki = getInterwiki();
299c112d578Sandi  #$Renderer->badwords = getBadWords();
300c112d578Sandi
301c112d578Sandi  // Loop through the instructions
302c112d578Sandi  foreach ( $instructions as $instruction ) {
303c112d578Sandi      // Execute the callback against the Renderer
304c112d578Sandi      call_user_func_array(array(&$Renderer, $instruction[0]),$instruction[1]);
305c112d578Sandi  }
3069dc2c2afSandi
3079dc2c2afSandi  //set info array
3089dc2c2afSandi  $info = $Renderer->info;
3099dc2c2afSandi
310c112d578Sandi  // Return the output
311c112d578Sandi  return $Renderer->doc;
312c112d578Sandi}
313c112d578Sandi
314bb0a59d4Sjan/**
315bb0a59d4Sjan * Gets the first heading from a file
316bb0a59d4Sjan *
317bb0a59d4Sjan * @author Jan Decaluwe <jan@jandecaluwe.com>
318bb0a59d4Sjan */
319bb0a59d4Sjanfunction p_get_first_heading($id){
320bb0a59d4Sjan  $file = wikiFN($id);
321bb0a59d4Sjan  if (@file_exists($file)) {
3226e38d921Sandi    $instructions = p_cached_instructions($file,true);
323bb0a59d4Sjan    foreach ( $instructions as $instruction ) {
324bb0a59d4Sjan      if ($instruction[0] == 'header') {
32587c434ceSAndreas Gohr        return trim($instruction[1][0]);
326bb0a59d4Sjan      }
327bb0a59d4Sjan    }
328bb0a59d4Sjan  }
329bb0a59d4Sjan  return NULL;
330bb0a59d4Sjan}
331bb0a59d4Sjan
332c112d578Sandi//Setup VIM: ex: et ts=2 enc=utf-8 :
333