1f3f0262cSandi<?php 215fae107Sandi/** 315fae107Sandi * XML feed export 415fae107Sandi * 515fae107Sandi * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 715fae107Sandi */ 815fae107Sandi 9d0a27cb0SAndreas Gohrif(!defined('DOKU_INC')) define('DOKU_INC',dirname(__FILE__).'/'); 10ed7b5f09Sandirequire_once(DOKU_INC.'inc/init.php'); 11f3f0262cSandi 127131b668SAndreas Gohr//close session 138746e727Sandisession_write_close(); 148746e727Sandi 154ab889eaSAndreas Gohr// get params 164ab889eaSAndreas Gohr$opt = rss_parseOptions(); 17f3f0262cSandi 187131b668SAndreas Gohr// the feed is dynamic - we need a cache for each combo 197131b668SAndreas Gohr// (but most people just use the default feed so it's still effective) 202f1faf49SAndreas Gohr$cache = getCacheName(join('',array_values($opt)).$_SERVER['REMOTE_USER'],'.feed'); 210d67055cSMichael Klier$key = join('', array_values($opt)) . $_SERVER['REMOTE_USER']; 220d67055cSMichael Klier$cache = new cache($key, '.feed'); 230d67055cSMichael Klier 240d67055cSMichael Klier// prepare cache depends 250d67055cSMichael Klier$depends['files'] = getConfigFiles('main'); 260d67055cSMichael Klier$depends['age'] = $conf['rss_update']; 27b5a0be43SAdrian Lang$depends['purge'] = isset($_REQUEST['purge']); 28f3f0262cSandi 297131b668SAndreas Gohr// check cacheage and deliver if nothing has changed since last 30fbf82939SBen Coburn// time or the update interval has not passed, also handles conditional requests 31fbf82939SBen Coburnheader('Cache-Control: must-revalidate, post-check=0, pre-check=0'); 32fbf82939SBen Coburnheader('Pragma: public'); 337131b668SAndreas Gohrheader('Content-Type: application/xml; charset=utf-8'); 3450ddb617SAndreas Gohrheader('X-Robots-Tag: noindex'); 350d67055cSMichael Klierif($cache->useCache($depends)) { 360d67055cSMichael Klier http_conditionalRequest($cache->_time); 370d67055cSMichael Klier if($conf['allowdebug']) header("X-CacheUsed: $cache->cache"); 380d67055cSMichael Klier print $cache->retrieveCache(); 397131b668SAndreas Gohr exit; 40fbf82939SBen Coburn} else { 41fbf82939SBen Coburn http_conditionalRequest(time()); 427131b668SAndreas Gohr } 437131b668SAndreas Gohr 447131b668SAndreas Gohr// create new feed 45f3f0262cSandi$rss = new DokuWikiFeedCreator(); 464ab889eaSAndreas Gohr$rss->title = $conf['title'].(($opt['namespace']) ? ' '.$opt['namespace'] : ''); 47ed7b5f09Sandi$rss->link = DOKU_URL; 48f62ea8a1Sandi$rss->syndicationURL = DOKU_URL.'feed.php'; 49615960feSTom N Harris$rss->cssStyleSheet = DOKU_URL.'lib/exe/css.php?s=feed'; 50f3f0262cSandi 5179b608ceSandi$image = new FeedImage(); 5279b608ceSandi$image->title = $conf['title']; 53*d6b643faSMatthias Schulte$image->url = tpl_getFavicon(true); 5479b608ceSandi$image->link = DOKU_URL; 5579b608ceSandi$rss->image = $image; 5679b608ceSandi 574bf3df7cSGina Haeussge$data = null; 58b5a0be43SAdrian Lang$modes = array('list' => 'rssListNamespace', 59b5a0be43SAdrian Lang 'search' => 'rssSearch', 60b5a0be43SAdrian Lang 'recent' => 'rssRecentChanges'); 61b5a0be43SAdrian Langif (isset($modes[$opt['feed_mode']])) { 62b5a0be43SAdrian Lang $data = $modes[$opt['feed_mode']]($opt); 63f3f0262cSandi} else { 644bf3df7cSGina Haeussge $eventData = array( 654bf3df7cSGina Haeussge 'opt' => &$opt, 664bf3df7cSGina Haeussge 'data' => &$data, 674bf3df7cSGina Haeussge ); 684bf3df7cSGina Haeussge $event = new Doku_Event('FEED_MODE_UNKNOWN', $eventData); 694bf3df7cSGina Haeussge if ($event->advise_before(true)) { 70b5a0be43SAdrian Lang echo sprintf('<error>Unknown feed mode %s</error>', hsc($opt['feed_mode'])); 71b5a0be43SAdrian Lang exit; 724bf3df7cSGina Haeussge } 734bf3df7cSGina Haeussge $event->advise_after(); 74f3f0262cSandi} 75f3f0262cSandi 764bf3df7cSGina Haeussgerss_buildItems($rss, $data, $opt); 774ab889eaSAndreas Gohr$feed = $rss->createFeed($opt['feed_type'],'utf-8'); 787131b668SAndreas Gohr 797131b668SAndreas Gohr// save cachefile 800d67055cSMichael Klier$cache->storeCache($feed); 817131b668SAndreas Gohr 827131b668SAndreas Gohr// finally deliver 837131b668SAndreas Gohrprint $feed; 84f3f0262cSandi 8515fae107Sandi// ---------------------------------------------------------------- // 86f3f0262cSandi 8715fae107Sandi/** 88b5a0be43SAdrian Lang * Get URL parameters and config options and return an initialized option array 8915fae107Sandi * 9015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 9115fae107Sandi */ 924ab889eaSAndreas Gohrfunction rss_parseOptions(){ 93f62ea8a1Sandi global $conf; 94c0f9af6dSNathan Neulinger 95b5a0be43SAdrian Lang $opt = array(); 96f62ea8a1Sandi 97b5a0be43SAdrian Lang foreach(array( 98b5a0be43SAdrian Lang // Basic feed properties 99b5a0be43SAdrian Lang // Plugins may probably want to add new values to these 100b5a0be43SAdrian Lang // properties for implementing own feeds 101b5a0be43SAdrian Lang 102b5a0be43SAdrian Lang // One of: list, search, recent 103b5a0be43SAdrian Lang 'feed_mode' => array('mode', 'recent'), 104b5a0be43SAdrian Lang // One of: diff, page, rev, current 105b5a0be43SAdrian Lang 'link_to' => array('linkto', $conf['rss_linkto']), 106b5a0be43SAdrian Lang // One of: abstract, diff, htmldiff, html 107b5a0be43SAdrian Lang 'item_content' => array('content', $conf['rss_content']), 108b5a0be43SAdrian Lang 109b5a0be43SAdrian Lang // Special feed properties 110b5a0be43SAdrian Lang // These are only used by certain feed_modes 111b5a0be43SAdrian Lang 112b5a0be43SAdrian Lang // String, used for feed title, in list and rc mode 113b5a0be43SAdrian Lang 'namespace' => array('ns', null), 114b5a0be43SAdrian Lang // Positive integer, only used in rc mode 115b5a0be43SAdrian Lang 'items' => array('num', $conf['recent']), 116b5a0be43SAdrian Lang // Boolean, only used in rc mode 117b5a0be43SAdrian Lang 'show_minor' => array('minor', false), 118b5a0be43SAdrian Lang // String, only used in search mode 119b5a0be43SAdrian Lang 'search_query' => array('q', null), 120b5a0be43SAdrian Lang 121b5a0be43SAdrian Lang ) as $name => $val) { 122b5a0be43SAdrian Lang $opt[$name] = (isset($_REQUEST[$val[0]]) && !empty($_REQUEST[$val[0]])) 123b5a0be43SAdrian Lang ? $_REQUEST[$val[0]] : $val[1]; 124b5a0be43SAdrian Lang } 125b5a0be43SAdrian Lang 126b5a0be43SAdrian Lang $opt['items'] = max(0, (int) $opt['items']); 127b5a0be43SAdrian Lang $opt['show_minor'] = (bool) $opt['show_minor']; 128b5a0be43SAdrian Lang 1294ab889eaSAndreas Gohr $opt['guardmail'] = ($conf['mailguard'] != '' && $conf['mailguard'] != 'none'); 130b6912aeaSAndreas Gohr 131b5a0be43SAdrian Lang $type = valid_input_set('type', array('rss','rss2','atom','atom1','rss1', 132b5a0be43SAdrian Lang 'default' => $conf['rss_type']), 133b5a0be43SAdrian Lang $_REQUEST); 134b5a0be43SAdrian Lang switch ($type){ 1354ab889eaSAndreas Gohr case 'rss': 1364ab889eaSAndreas Gohr $opt['feed_type'] = 'RSS0.91'; 1374ab889eaSAndreas Gohr $opt['mime_type'] = 'text/xml'; 1384ab889eaSAndreas Gohr break; 1394ab889eaSAndreas Gohr case 'rss2': 1404ab889eaSAndreas Gohr $opt['feed_type'] = 'RSS2.0'; 1414ab889eaSAndreas Gohr $opt['mime_type'] = 'text/xml'; 1424ab889eaSAndreas Gohr break; 1434ab889eaSAndreas Gohr case 'atom': 1444ab889eaSAndreas Gohr $opt['feed_type'] = 'ATOM0.3'; 1454ab889eaSAndreas Gohr $opt['mime_type'] = 'application/xml'; 1464ab889eaSAndreas Gohr break; 1474ab889eaSAndreas Gohr case 'atom1': 1484ab889eaSAndreas Gohr $opt['feed_type'] = 'ATOM1.0'; 1494ab889eaSAndreas Gohr $opt['mime_type'] = 'application/atom+xml'; 1504ab889eaSAndreas Gohr break; 1514ab889eaSAndreas Gohr default: 1524ab889eaSAndreas Gohr $opt['feed_type'] = 'RSS1.0'; 1534ab889eaSAndreas Gohr $opt['mime_type'] = 'application/xml'; 1544ab889eaSAndreas Gohr } 1554bf3df7cSGina Haeussge 1564bf3df7cSGina Haeussge $eventData = array( 1574bf3df7cSGina Haeussge 'opt' => &$opt, 1584bf3df7cSGina Haeussge ); 1594bf3df7cSGina Haeussge trigger_event('FEED_OPTS_POSTPROCESS', $eventData); 1604ab889eaSAndreas Gohr return $opt; 1614ab889eaSAndreas Gohr} 162b6912aeaSAndreas Gohr 1634ab889eaSAndreas Gohr/** 1644ab889eaSAndreas Gohr * Add recent changed pages to a feed object 1654ab889eaSAndreas Gohr * 1664ab889eaSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 1674ab889eaSAndreas Gohr * @param object $rss - the FeedCreator Object 1684ab889eaSAndreas Gohr * @param array $data - the items to add 1694ab889eaSAndreas Gohr * @param array $opt - the feed options 1704ab889eaSAndreas Gohr */ 1714ab889eaSAndreas Gohrfunction rss_buildItems(&$rss,&$data,$opt){ 1724ab889eaSAndreas Gohr global $conf; 1734ab889eaSAndreas Gohr global $lang; 1743d581c29SAndreas Gohr global $auth; 175883480fbSAndreas Gohr 1764bf3df7cSGina Haeussge $eventData = array( 1774bf3df7cSGina Haeussge 'rss' => &$rss, 1784bf3df7cSGina Haeussge 'data' => &$data, 1794bf3df7cSGina Haeussge 'opt' => &$opt, 1804bf3df7cSGina Haeussge ); 1814bf3df7cSGina Haeussge $event = new Doku_Event('FEED_DATA_PROCESS', $eventData); 1824bf3df7cSGina Haeussge if ($event->advise_before(false)){ 1834ab889eaSAndreas Gohr foreach($data as $ditem){ 1844bb1b5aeSAndreas Gohr if(!is_array($ditem)){ 1854bb1b5aeSAndreas Gohr // not an array? then only a list of IDs was given 1864bb1b5aeSAndreas Gohr $ditem = array( 'id' => $ditem ); 1874bb1b5aeSAndreas Gohr } 1884bb1b5aeSAndreas Gohr 189f3f0262cSandi $item = new FeedItem(); 1904ab889eaSAndreas Gohr $id = $ditem['id']; 1914ab889eaSAndreas Gohr $meta = p_get_metadata($id); 19203ee62cbSjoe.lapp 1934ab889eaSAndreas Gohr // add date 1944ab889eaSAndreas Gohr if($ditem['date']){ 1954ab889eaSAndreas Gohr $date = $ditem['date']; 1964ab889eaSAndreas Gohr }elseif($meta['date']['modified']){ 1974ab889eaSAndreas Gohr $date = $meta['date']['modified']; 1984ab889eaSAndreas Gohr }else{ 1994ab889eaSAndreas Gohr $date = @filemtime(wikiFN($id)); 2004ab889eaSAndreas Gohr } 2014ab889eaSAndreas Gohr if($date) $item->date = date('r',$date); 2024ab889eaSAndreas Gohr 2034ab889eaSAndreas Gohr // add title 2048716966dSAndreas Gohr if($conf['useheading'] && $meta['title']){ 2058716966dSAndreas Gohr $item->title = $meta['title']; 2068716966dSAndreas Gohr }else{ 2074ab889eaSAndreas Gohr $item->title = $ditem['id']; 20803ee62cbSjoe.lapp } 2094ab889eaSAndreas Gohr if($conf['rss_show_summary'] && !empty($ditem['sum'])){ 2104ab889eaSAndreas Gohr $item->title .= ' - '.strip_tags($ditem['sum']); 211b1a1915cSandi } 2124d58bd99Sandi 2134ab889eaSAndreas Gohr // add item link 2144ab889eaSAndreas Gohr switch ($opt['link_to']){ 2154d58bd99Sandi case 'page': 2164ab889eaSAndreas Gohr $item->link = wl($id,'rev='.$date,true,'&'); 2174d58bd99Sandi break; 2184d58bd99Sandi case 'rev': 2194ab889eaSAndreas Gohr $item->link = wl($id,'do=revisions&rev='.$date,true,'&'); 2204d58bd99Sandi break; 22192e52d8dSjoe.lapp case 'current': 2224ab889eaSAndreas Gohr $item->link = wl($id, '', true,'&'); 22392e52d8dSjoe.lapp break; 22492e52d8dSjoe.lapp case 'diff': 2254d58bd99Sandi default: 2264ab889eaSAndreas Gohr $item->link = wl($id,'rev='.$date.'&do=diff',true,'&'); 2274d58bd99Sandi } 2284d58bd99Sandi 2294ab889eaSAndreas Gohr // add item content 2304ab889eaSAndreas Gohr switch ($opt['item_content']){ 2314ab889eaSAndreas Gohr case 'diff': 2324ab889eaSAndreas Gohr case 'htmldiff': 2334ab889eaSAndreas Gohr require_once(DOKU_INC.'inc/DifferenceEngine.php'); 2344ab889eaSAndreas Gohr $revs = getRevisions($id, 0, 1); 2354ab889eaSAndreas Gohr $rev = $revs[0]; 2367a98db20Sjoe.lapp 2374ab889eaSAndreas Gohr if($rev){ 2384ab889eaSAndreas Gohr $df = new Diff(explode("\n",htmlspecialchars(rawWiki($id,$rev))), 2394ab889eaSAndreas Gohr explode("\n",htmlspecialchars(rawWiki($id,'')))); 2404ab889eaSAndreas Gohr }else{ 2414ab889eaSAndreas Gohr $df = new Diff(array(''), 2424ab889eaSAndreas Gohr explode("\n",htmlspecialchars(rawWiki($id,'')))); 2434ab889eaSAndreas Gohr } 2444ab889eaSAndreas Gohr 2454ab889eaSAndreas Gohr if($opt['item_content'] == 'htmldiff'){ 2464ab889eaSAndreas Gohr $tdf = new TableDiffFormatter(); 2474ab889eaSAndreas Gohr $content = '<table>'; 2484ab889eaSAndreas Gohr $content .= '<tr><th colspan="2" width="50%">'.$rev.'</th>'; 2494ab889eaSAndreas Gohr $content .= '<th colspan="2" width="50%">'.$lang['current'].'</th></tr>'; 2504ab889eaSAndreas Gohr $content .= $tdf->format($df); 2514ab889eaSAndreas Gohr $content .= '</table>'; 2524ab889eaSAndreas Gohr }else{ 2534ab889eaSAndreas Gohr $udf = new UnifiedDiffFormatter(); 2544ab889eaSAndreas Gohr $content = "<pre>\n".$udf->format($df)."\n</pre>"; 2554ab889eaSAndreas Gohr } 2564ab889eaSAndreas Gohr break; 2574ab889eaSAndreas Gohr case 'html': 2584ab889eaSAndreas Gohr $content = p_wiki_xhtml($id,$date,false); 2594ab889eaSAndreas Gohr // no TOC in feeds 2604ab889eaSAndreas Gohr $content = preg_replace('/(<!-- TOC START -->).*(<!-- TOC END -->)/s','',$content); 2614ab889eaSAndreas Gohr 2624ab889eaSAndreas Gohr // make URLs work when canonical is not set, regexp instead of rerendering! 2634ab889eaSAndreas Gohr if(!$conf['canonical']){ 2644ab889eaSAndreas Gohr $base = preg_quote(DOKU_REL,'/'); 2654ab889eaSAndreas Gohr $content = preg_replace('/(<a href|<img src)="('.$base.')/s','$1="'.DOKU_URL,$content); 2664ab889eaSAndreas Gohr } 2674ab889eaSAndreas Gohr 2684ab889eaSAndreas Gohr break; 2694ab889eaSAndreas Gohr case 'abstract': 2704ab889eaSAndreas Gohr default: 2714ab889eaSAndreas Gohr $content = $meta['description']['abstract']; 2724ab889eaSAndreas Gohr } 2734ab889eaSAndreas Gohr $item->description = $content; //FIXME a plugin hook here could be senseful 2744ab889eaSAndreas Gohr 2754ab889eaSAndreas Gohr // add user 2764ab889eaSAndreas Gohr # FIXME should the user be pulled from metadata as well? 2774ab889eaSAndreas Gohr $user = @$ditem['user']; // the @ spares time repeating lookup 2787a98db20Sjoe.lapp $item->author = ''; 2790f4f4adfSAndreas Gohr if($user && $conf['useacl'] && $auth){ 280c0f9af6dSNathan Neulinger $userInfo = $auth->getUserData($user); 281681a59b2SGina Haeussge if ($userInfo){ 282681a59b2SGina Haeussge switch ($conf['showuseras']){ 283681a59b2SGina Haeussge case 'username': 2847a98db20Sjoe.lapp $item->author = $userInfo['name']; 285681a59b2SGina Haeussge break; 286681a59b2SGina Haeussge default: 287681a59b2SGina Haeussge $item->author = $user; 288681a59b2SGina Haeussge break; 289681a59b2SGina Haeussge } 290681a59b2SGina Haeussge } else { 291681a59b2SGina Haeussge $item->author = $user; 292681a59b2SGina Haeussge } 293c1791678SAndreas Gohr if($userInfo && !$opt['guardmail']){ 294c1791678SAndreas Gohr $item->authorEmail = $userInfo['mail']; 295c1791678SAndreas Gohr }else{ 2967a98db20Sjoe.lapp //cannot obfuscate because some RSS readers may check validity 2970bda0363SAdrian Lang $item->authorEmail = $user.'@'.$ditem['ip']; 298f3f0262cSandi } 299c5983034SAndreas Gohr }elseif($user){ 300c5983034SAndreas Gohr // this happens when no ACL but some Apache auth is used 301c5983034SAndreas Gohr $item->author = $user; 3020bda0363SAdrian Lang $item->authorEmail = $user.'@'.$ditem['ip']; 3037a98db20Sjoe.lapp }else{ 3040bda0363SAdrian Lang $item->authorEmail = 'anonymous@'.$ditem['ip']; 3057a98db20Sjoe.lapp } 3064ab889eaSAndreas Gohr 3074ab889eaSAndreas Gohr // add category 308b5a0be43SAdrian Lang if(isset($meta['subject'])) { 3094ab889eaSAndreas Gohr $item->category = $meta['subject']; 3104ab889eaSAndreas Gohr }else{ 3114ab889eaSAndreas Gohr $cat = getNS($id); 3124ab889eaSAndreas Gohr if($cat) $item->category = $cat; 3134ab889eaSAndreas Gohr } 3144ab889eaSAndreas Gohr 315883480fbSAndreas Gohr // finally add the item to the feed object, after handing it to registered plugins 316883480fbSAndreas Gohr $evdata = array('item' => &$item, 317883480fbSAndreas Gohr 'opt' => &$opt, 318883480fbSAndreas Gohr 'ditem' => &$ditem, 319883480fbSAndreas Gohr 'rss' => &$rss); 320209cd8e1SAndreas Gohr $evt = new Doku_Event('FEED_ITEM_ADD', $evdata); 321883480fbSAndreas Gohr if ($evt->advise_before()){ 322f3f0262cSandi $rss->addItem($item); 323f3f0262cSandi } 324883480fbSAndreas Gohr $evt->advise_after(); // for completeness 325883480fbSAndreas Gohr } 326f3f0262cSandi } 3274bf3df7cSGina Haeussge $event->advise_after(); 3284bf3df7cSGina Haeussge} 329f3f0262cSandi 3304ab889eaSAndreas Gohr 3314ab889eaSAndreas Gohr/** 3324bb1b5aeSAndreas Gohr * Add recent changed pages to the feed object 3334ab889eaSAndreas Gohr * 3344ab889eaSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 3354ab889eaSAndreas Gohr */ 3364bf3df7cSGina Haeussgefunction rssRecentChanges($opt){ 3374ab889eaSAndreas Gohr $flags = RECENTS_SKIP_DELETED; 3384ab889eaSAndreas Gohr if(!$opt['show_minor']) $flags += RECENTS_SKIP_MINORS; 3394ab889eaSAndreas Gohr 3404ab889eaSAndreas Gohr $recents = getRecents(0,$opt['items'],$opt['namespace'],$flags); 3414bf3df7cSGina Haeussge return $recents; 3424ab889eaSAndreas Gohr} 3434ab889eaSAndreas Gohr 34415fae107Sandi/** 3454bb1b5aeSAndreas Gohr * Add all pages of a namespace to the feed object 34615fae107Sandi * 34715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 34815fae107Sandi */ 3494bf3df7cSGina Haeussgefunction rssListNamespace($opt){ 350f62ea8a1Sandi require_once(DOKU_INC.'inc/search.php'); 351f3f0262cSandi global $conf; 352f3f0262cSandi 3534ab889eaSAndreas Gohr $ns=':'.cleanID($opt['namespace']); 354f3f0262cSandi $ns=str_replace(':','/',$ns); 355f3f0262cSandi 356f3f0262cSandi $data = array(); 357f3f0262cSandi sort($data); 358f3f0262cSandi search($data,$conf['datadir'],'search_list','',$ns); 35985cf8195SAndreas Gohr 3604bf3df7cSGina Haeussge return $data; 36185cf8195SAndreas Gohr} 36285cf8195SAndreas Gohr 3634bb1b5aeSAndreas Gohr/** 3644bb1b5aeSAndreas Gohr * Add the result of a full text search to the feed object 3654bb1b5aeSAndreas Gohr * 3664bb1b5aeSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 3674bb1b5aeSAndreas Gohr */ 3684bf3df7cSGina Haeussgefunction rssSearch($opt){ 3694bb1b5aeSAndreas Gohr if(!$opt['search_query']) return; 3704ab889eaSAndreas Gohr 3714bb1b5aeSAndreas Gohr require_once(DOKU_INC.'inc/fulltext.php'); 3724bb1b5aeSAndreas Gohr $data = ft_pageSearch($opt['search_query'],$poswords); 3734bb1b5aeSAndreas Gohr $data = array_keys($data); 3744bf3df7cSGina Haeussge 3754bf3df7cSGina Haeussge return $data; 3764bb1b5aeSAndreas Gohr} 377f3f0262cSandi 378e3776c06SMichael Hamann//Setup VIM: ex: et ts=4 : 379