Merge branch 'master' of git.gnu.io:gnu/gnu-social into nightly

Conflicts:
	plugins/OStatus/OStatusPlugin.php

master vs. nightly thing
This commit is contained in:
Mikael Nordfeldth 2015-11-05 16:16:02 +01:00
commit 7ccd36849e
7 changed files with 281 additions and 24 deletions

View File

@ -194,7 +194,7 @@ class FavoritePlugin extends ActivityVerbHandlerPlugin
$actobj = $act->objects[0]; $actobj = $act->objects[0];
$object = Fave::saveActivityObject($actobj, $stored); $object = Fave::saveActivityObject($actobj, $stored);
$stored->object_type = ActivityUtils::resolveUri($object->getObjectType(), true); $stored->object_type = $object->getObjectType();
return $object; return $object;
} }

View File

@ -65,10 +65,28 @@ class LinkbackPlugin extends Plugin
// notice content // notice content
$c = $notice->content; $c = $notice->content;
$this->notice = $notice; $this->notice = $notice;
if(!$notice->getProfile()->
getPref("linkbackplugin", "disable_linkbacks")
) {
// Ignoring results // Ignoring results
common_replace_urls_callback($c, common_replace_urls_callback($c,
array($this, 'linkbackUrl')); array($this, 'linkbackUrl'));
} }
if($notice->isRepeat()) {
$repeat = Notice::getByID($notice->repeat_of);
$this->linkbackUrl($repeat->getUrl());
} else if(!empty($notice->reply_to)) {
$parent = $notice->getParent();
$this->linkbackUrl($parent->getUrl());
}
$replyProfiles = Profile::multiGet('id', $notice->getReplies());
foreach($replyProfiles->fetchAll('profileurl') as $profileurl) {
$this->linkbackUrl($profileurl);
}
}
return true; return true;
} }
@ -95,32 +113,89 @@ class LinkbackPlugin extends Plugin
return $orig; return $orig;
} }
$pb = null; // XXX: Should handle relative-URI resolution in these detections
$tb = null;
if (array_key_exists('X-Pingback', $result->headers)) { $wm = $this->getWebmention($result);
$pb = $result->headers['X-Pingback']; if(!empty($wm)) {
} else if (preg_match('/<link rel="pingback" href="([^"]+)" ?\/?>/', // It is the webmention receiver's job to resolve source
$result->body, // Ref: https://github.com/converspace/webmention/issues/43
$match)) { $this->webmention($url, $wm);
$pb = $match[1];
}
if (!empty($pb)) {
$this->pingback($result->final_url, $pb);
} else { } else {
$tb = $this->getTrackback($result->body, $result->final_url); $pb = $this->getPingback($result);
if (!empty($pb)) {
// Pingback still looks for exact URL in our source, so we
// must send what we have
$this->pingback($url, $pb);
} else {
$tb = $this->getTrackback($result);
if (!empty($tb)) { if (!empty($tb)) {
$this->trackback($result->final_url, $tb); $this->trackback($result->final_url, $tb);
} }
} }
}
return $orig; return $orig;
} }
// Based on https://github.com/indieweb/mention-client-php
// which is licensed Apache 2.0
function getWebmention($result) {
// XXX: the fetcher only gives back one of each header, so this may fail on multiple Link headers
if(preg_match('~<((?:https?://)?[^>]+)>; rel="webmention"~', $result->headers['Link'], $match)) {
return $match[1];
} elseif(preg_match('~<((?:https?://)?[^>]+)>; rel="http://webmention.org/?"~', $result->headers['Link'], $match)) {
return $match[1];
}
if(preg_match('/<(?:link|a)[ ]+href="([^"]+)"[ ]+rel="[^" ]* ?webmention ?[^" ]*"[ ]*\/?>/i', $result->body, $match)
|| preg_match('/<(?:link|a)[ ]+rel="[^" ]* ?webmention ?[^" ]*"[ ]+href="([^"]+)"[ ]*\/?>/i', $result->body, $match)) {
return $match[1];
} elseif(preg_match('/<(?:link|a)[ ]+href="([^"]+)"[ ]+rel="http:\/\/webmention\.org\/?"[ ]*\/?>/i', $result->body, $match)
|| preg_match('/<(?:link|a)[ ]+rel="http:\/\/webmention\.org\/?"[ ]+href="([^"]+)"[ ]*\/?>/i', $result->body, $match)) {
return $match[1];
}
}
function webmention($url, $endpoint) {
$source = $this->notice->getUrl();
$payload = array(
'source' => $source,
'target' => $url
);
$request = HTTPClient::start();
try {
$response = $request->post($endpoint,
array(
'Content-type: application/x-www-form-urlencoded',
'Accept: application/json'
),
$payload
);
if(!in_array($response->getStatus(), array(200,202))) {
common_log(LOG_WARNING,
"Webmention request failed for '$url' ($endpoint)");
}
} catch (HTTP_Request2_Exception $e) {
common_log(LOG_WARNING,
"Webmention request failed for '$url' ($endpoint)");
}
}
function getPingback($result) {
if (array_key_exists('X-Pingback', $result->headers)) {
return $result->headers['X-Pingback'];
} else if(preg_match('/<(?:link|a)[ ]+href="([^"]+)"[ ]+rel="[^" ]* ?pingback ?[^" ]*"[ ]*\/?>/i', $result->body, $match)
|| preg_match('/<(?:link|a)[ ]+rel="[^" ]* ?pingback ?[^" ]*"[ ]+href="([^"]+)"[ ]*\/?>/i', $result->body, $match)) {
return $match[1];
}
}
function pingback($url, $endpoint) function pingback($url, $endpoint)
{ {
$args = array($this->notice->uri, $url); $args = array($this->notice->getUrl(), $url);
if (!extension_loaded('xmlrpc')) { if (!extension_loaded('xmlrpc')) {
if (!dl('xmlrpc.so')) { if (!dl('xmlrpc.so')) {
@ -131,9 +206,10 @@ class LinkbackPlugin extends Plugin
$request = HTTPClient::start(); $request = HTTPClient::start();
try { try {
$request->setBody(xmlrpc_encode_request('pingback.ping', $args));
$response = $request->post($endpoint, $response = $request->post($endpoint,
array('Content-Type: text/xml'), array('Content-Type: text/xml'),
xmlrpc_encode_request('pingback.ping', $args)); false);
$response = xmlrpc_decode($response->getBody()); $response = xmlrpc_decode($response->getBody());
if (xmlrpc_is_fault($response)) { if (xmlrpc_is_fault($response)) {
common_log(LOG_WARNING, common_log(LOG_WARNING,
@ -153,8 +229,11 @@ class LinkbackPlugin extends Plugin
// Largely cadged from trackback_cls.php by // Largely cadged from trackback_cls.php by
// Ran Aroussi <ran@blogish.org>, GPL2 or any later version // Ran Aroussi <ran@blogish.org>, GPL2 or any later version
// http://phptrackback.sourceforge.net/ // http://phptrackback.sourceforge.net/
function getTrackback($text, $url) function getTrackback($result)
{ {
$text = $result->body;
$url = $result->final_url;
if (preg_match_all('/(<rdf:RDF.*?<\/rdf:RDF>)/sm', $text, $match, PREG_SET_ORDER)) { if (preg_match_all('/(<rdf:RDF.*?<\/rdf:RDF>)/sm', $text, $match, PREG_SET_ORDER)) {
for ($i = 0; $i < count($match); $i++) { for ($i = 0; $i < count($match); $i++) {
if (preg_match('|dc:identifier="' . preg_quote($url) . '"|ms', $match[$i][1])) { if (preg_match('|dc:identifier="' . preg_quote($url) . '"|ms', $match[$i][1])) {
@ -246,4 +325,23 @@ class LinkbackPlugin extends Plugin
'or <a href="http://www.movabletype.org/docs/mttrackback.html">Trackback</a> protocols.')); 'or <a href="http://www.movabletype.org/docs/mttrackback.html">Trackback</a> protocols.'));
return true; return true;
} }
public function onStartInitializeRouter(URLMapper $m)
{
$m->connect('settings/linkback', array('action' => 'linkbacksettings'));
return true;
}
function onEndAccountSettingsNav($action)
{
$action_name = $action->trimmed('action');
$action->menuItem(common_local_url('linkbacksettings'),
// TRANS: OpenID plugin menu item on user settings page.
_m('MENU', 'Send Linkbacks'),
// TRANS: OpenID plugin tooltip for user settings menu item.
_m('Opt-out of sending linkbacks.'),
$action_name === 'linkbacksettings');
return true;
}
} }

View File

@ -0,0 +1,91 @@
<?php
/**
* Settings for Linkback
*
* PHP version 5
*
* LICENCE: This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* @category Settings
* @package StatusNet
* @author Stephen Paul Weber <singpolyma@singpolyma.net>
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
*/
if (!defined('GNUSOCIAL')) { exit(1); }
/**
* Settings for Linkback
*
* Lets users opt out of sending linkbacks
*
* @category Settings
* @author Stephen Paul Weber <singpolyma@singpolyma.net>
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
*/
class LinkbacksettingsAction extends SettingsAction
{
/**
* Title of the page
*
* @return string Page title
*/
function title()
{
// TRANS: Title of Linkback settings page for a user.
return _m('TITLE','Linkback settings');
}
/**
* Instructions for use
*
* @return string Instructions for use
*/
function getInstructions()
{
// TRANS: Form instructions for Linkback settings.
return _m('Linkbacks inform post authors when you link to them. ' .
'You can disable this feature here.');
}
function showContent()
{
$this->elementStart('form', array('method' => 'post',
'class' => 'form_settings',
'action' =>
common_local_url('linkbacksettings')));
$this->hidden('token', common_session_token());
$this->elementStart('fieldset');
$this->element('legend', null, _m('LEGEND','Preferences'));
$this->checkbox('disable_linkbacks', "Opt out of sending linkbacks for URLs you post", $this->scoped->getPref("linkbackplugin", "disable_linkbacks"));
// TRANS: Button text to save OpenID prefs
$this->submit('settings_linkback_prefs_save', _m('BUTTON','Save'), 'submit', 'save_prefs');
$this->elementEnd('fieldset');
$this->elementEnd('form');
}
/**
* Handle a POST request
*
* @return void
*/
protected function doPost()
{
$x = $this->scoped->setPref("linkbackplugin", "disable_linkbacks", $this->boolean('disable_linkbacks'));
return _m('Linkback preferences saved.');
}
}

View File

@ -118,6 +118,9 @@ class OStatusPlugin extends Plugin
// Incoming from a foreign PuSH hub // Incoming from a foreign PuSH hub
$qm->connect('pushin', 'PushInQueueHandler'); $qm->connect('pushin', 'PushInQueueHandler');
// Re-subscribe feeds that need renewal
$qm->connect('pushrenew', 'PushRenewQueueHandler');
return true; return true;
} }
@ -1396,4 +1399,20 @@ class OStatusPlugin extends Plugin
// Since we completed the salmon slap, we discontinue the event // Since we completed the salmon slap, we discontinue the event
return false; return false;
} }
public function onCronDaily()
{
try {
$sub = FeedSub::renewalCheck();
} catch (NoResultException $e) {
common_log(LOG_INFO, "There were no expiring feeds.");
return;
}
$qm = QueueManager::get();
while ($sub->fetch()) {
$item = array('feedsub_id' => $sub->id);
$qm->enqueue($item, 'pushrenew');
}
}
} }

View File

@ -295,7 +295,7 @@ class FeedSub extends Managed_DataObject
{ {
$fs = new FeedSub(); $fs = new FeedSub();
// the "" empty string check is because we historically haven't saved unsubscribed feeds as NULL // the "" empty string check is because we historically haven't saved unsubscribed feeds as NULL
$fs->whereAdd('sub_end IS NOT NULL AND sub_end!="" AND sub_end < NOW() - INTERVAL 1 day'); $fs->whereAdd('sub_end IS NOT NULL AND sub_end!="" AND sub_end < NOW() + INTERVAL 1 day');
if (!$fs->find()) { // find can be both false and 0, depending on why nothing was found if (!$fs->find()) { // find can be both false and 0, depending on why nothing was found
throw new NoResultException($fs); throw new NoResultException($fs);
} }

View File

@ -0,0 +1,49 @@
<?php
/*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
if (!defined('STATUSNET')) {
exit(1);
}
/**
* Renew an expiring feedsub
* @package FeedSub
* @author Stephen Paul Weber <singpolyma@singpolyma.net>
*/
class PushRenewQueueHandler extends QueueHandler
{
function transport()
{
return 'pushrenew';
}
function handle($data)
{
$feedsub_id = $data['feedsub_id'];
$feedsub = FeedSub::getKV('id', $feedsub_id);
if ($feedsub instanceof FeedSub) {
try {
common_log(LOG_INFO, "Renewing feed subscription\n\tExp.: {$feedsub->sub_end}\n\tFeed: {$feedsub->uri}\n\tHub: {$feedsub->huburi}");
$feedsub->renew();
} catch(Exception $e) {
common_log(LOG_ERR, "Exception during PuSH renew processing for $feedsub->uri: " . $e->getMessage());
}
} else {
common_log(LOG_ERR, "Discarding renew for unknown feed subscription id $feedsub_id");
}
return true;
}
}

View File

@ -148,7 +148,7 @@ class WebFingerPlugin extends Plugin
$url = common_local_url('webfinger') . '?resource='.$acct; $url = common_local_url('webfinger') . '?resource='.$acct;
foreach (array(Discovery::JRD_MIMETYPE, Discovery::XRD_MIMETYPE) as $type) { foreach (array(Discovery::JRD_MIMETYPE, Discovery::XRD_MIMETYPE) as $type) {
header('Link: <'.$url.'>; rel="'. Discovery::LRDD_REL.'"; type="'.$type.'"'); header('Link: <'.$url.'>; rel="'. Discovery::LRDD_REL.'"; type="'.$type.'"', false);
} }
} }
} }