* @link http://wiki.foosel.net/snippets/dokuwiki/linkback */ use dokuwiki\Extension\PluginInterface; use IXR\Server\Server; use IXR\Message\Error; if (!defined('DOKU_INC')) define('DOKU_INC', realpath(dirname(__FILE__) . '/../../../../') . '/'); require_once (DOKU_INC . 'inc/init.php'); require_once (DOKU_PLUGIN . 'linkback/tools.php'); require_once (DOKU_PLUGIN . 'linkback/http.php'); // Pingback Faultcodes const PINGBACK_ERROR_GENERIC = 0; const PINGBACK_ERROR_SOURCEURI_DOES_NOT_EXIST = 16; const PINGBACK_ERROR_SOURCEURI_DOES_NOT_CONTAIN_LINK = 17; const PINGBACK_ERROR_TARGETURI_DOES_NOT_EXIST = 32; const PINGBACK_ERROR_TARGETURI_CANNOT_BE_USED = 33; const PINGBACK_ERROR_PINGBACK_ALREADY_MADE = 48; const PINGBACK_ERROR_ACCESS_DENIED = 49; const PINGBACK_ERROR_NO_UPSTREAM = 50; class PingbackServer extends Server{ // helper instance /** @var PluginInterface|tools_plugin_linkback */ var $tools; /** * Register service and construct helper */ function __construct() { $this->tools = plugin_load('tools', 'linkback'); //TODO type 'tools'? is that possible? parent::__construct(array ( 'pingback.ping' => 'this:ping', )); } /** * @param $sourceUri * @param $targetUri * @return Error|void */ function ping($sourceUri, $targetUri) { // Plugin not enabled? Quit if (plugin_isdisabled('linkback')) return new Error(PINGBACK_ERROR_TARGETURI_CANNOT_BE_USED, ''); // pingback disabled? Quit if (!$this->tools->getConf('enable_pingback')) return new Error(PINGBACK_ERROR_TARGETURI_CANNOT_BE_USED, ''); // Given URLs are no urls? Quit if (!preg_match("#^([a-z0-9\-\.+]+?)://.*#i", $sourceUri)) return new Error(PINGBACK_ERROR_GENERIC, ''); if (!preg_match("#^([a-z0-9\-\.+]+?)://.*#i", $targetUri)) return new Error(PINGBACK_ERROR_GENERIC, ''); // Source URL does not exist? Quit $page = $this->tools->getPage($sourceUri); if (!$page['success'] && ($page['status'] < 200 || $page['status'] >= 300)) return new Error(PINGBACK_ERROR_SOURCEURI_DOES_NOT_EXIST, ''); // Target URL does not match with request? Quit $ID = substr($_SERVER['PATH_INFO'], 1); if ($targetUri != wl($ID, '', true)) return new Error(PINGBACK_ERROR_GENERIC, ''); $file = metaFN($ID, '.linkbacks'); $data = array ( 'send' => false, 'receive' => false, 'display' => false, 'sentpings' => array (), 'receivedpings' => array (), 'number' => 0, ); if (@ file_exists($file)) $data = unserialize(io_readFile($file, false)); // Target URL is not pingback enabled? Quit if (!$data['receive']) return new Error(PINGBACK_ERROR_TARGETURI_CANNOT_BE_USED, ''); // Pingback already done? Quit if ($data['receivedpings'][md5($sourceUri)]) return new Error(PINGBACK_ERROR_PINGBACK_ALREADY_MADE, ''); // Retrieve data from source $linkback = $this->_getTrackbackData($sourceUri, $targetUri, $page); // Source URL does not contain link to target? Quit if (!$linkback) return new Error(PINGBACK_ERROR_SOURCEURI_DOES_NOT_CONTAIN_LINK, ''); // Prepare event for Antispam plugins $evt_data = array ( 'linkback' => $linkback, 'page' => $page, 'target' => $targetUri, 'show' => true, 'log' => array( date('Y/m/d H:i', time()) . ': Received pingback from ' . $linkback['url'] . ' (' . $linkback['lid'] . ')', ), ); $event = new Doku_Event('ACTION_LINKBACK_RECEIVED', $evt_data); if ($event->advise_before()) { $linkback['show'] = $evt_data['show']; if ($this->tools->getConf('usefavicon')) { $linkback['favicon'] = $this->tools->getFavicon($linkback['url'], $page['body']); } // add pingback $data['receivedpings'][$linkback['lid']] = $linkback; if ($linkback['show']) $data['number']++; io_saveFile($file, serialize($data)); $this->tools->addLogEntry($linkback['received'], $ID, 'cl', '', $linkback['lid']); $this->tools->notify($ID, $linkback); if ($this->tools->getConf('log_processing')) $this->tools->addProcessLogEntry($evt_data['log']); $event->advise_after(); } else { // Pingback was denied if ($this->tools->getConf('log_processing')) $this->tools->addProcessLogEntry($evt_data['log']); $event->advise_after(); return new Error(PINGBACK_ERROR_ACCESS_DENIED, $this->tools->getLang('error_noreason')); } } /** * Constructs linkback data and checks if source contains a link to target and a title. */ function _getTrackbackData($sourceUri, $targetUri, $page) { // construct unique id for pingback $lid = md5($sourceUri); $linkback = array ( 'lid' => $lid, 'title' => '', 'url' => $sourceUri, 'excerpt' => '', 'raw_excerpt' => '', 'blog_name' => '', 'received' => time(), 'submitter_ip' => $_SERVER['REMOTE_ADDR'], 'submitter_useragent' => $_SERVER['HTTP_USER_AGENT'], 'submitter_referer' => $_SERVER['HTTP_REFERER'], 'type' => 'pingback', 'show' => true, ); $searchurl = preg_quote($targetUri, '!'); $regex = '!]+?href="' . $searchurl . '"[^>]*?>(.*?)!is'; $regex2 = '!\s(' . $searchurl . ')\s!is'; if (!preg_match($regex, $page['body'], $match) && !preg_match($regex2, $page['body'], $match)) { if ($this->tools->getConf('ping_internal') && (strstr($targetUri, DOKU_URL) == $targetUri)) { $ID = substr($_SERVER['PATH_INFO'], 1); $searchurl = preg_quote(wl($ID), '!'); $regex = '!]+?href="' . $searchurl . '"[^>]*?>(.*?)!is'; $regex2 = '!\s(' . $searchurl . ')\s!is'; if (!preg_match($regex, $page['body'], $match) && !preg_match($regex2, $page['body'], $match)) return false; } else { return false; } } $linkback['raw_excerpt'] = '[...] ' . $match[1] . ' [...]'; $linkback['excerpt'] = '[...] ' . strip_tags($match[1]) . ' [...]'; $regex = '!