2010-02-08 19:06:03 +00:00
< ? php
/*
* StatusNet - the distributed open - source microblogging tool
* Copyright ( C ) 2010 , StatusNet , Inc .
*
* 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 />.
*/
2010-10-08 18:42:59 +01:00
if ( ! defined ( 'STATUSNET' )) {
exit ( 1 );
}
2010-02-08 19:06:03 +00:00
/**
2010-02-24 02:19:13 +00:00
* Prepare PuSH and Salmon distributions for an outgoing message .
*
* @ package OStatusPlugin
2010-02-08 19:06:03 +00:00
* @ author Brion Vibber < brion @ status . net >
*/
2010-02-24 02:19:13 +00:00
class OStatusQueueHandler extends QueueHandler
2010-02-08 19:06:03 +00:00
{
2010-06-04 19:48:54 +01:00
// If we have more than this many subscribing sites on a single feed,
// break up the PuSH distribution into smaller batches which will be
// rolled into the queue progressively. This reduces disruption to
// other, shorter activities being enqueued while we work.
const MAX_UNBATCHED = 50 ;
// Each batch (a 'hubprep' entry) will have this many items.
// Selected to provide a balance between queue packet size
// and number of batches that will end up getting processed.
// For 20,000 target sites, 1000 should work acceptably.
const BATCH_SIZE = 1000 ;
2010-02-08 19:06:03 +00:00
function transport ()
{
2010-02-24 02:19:13 +00:00
return 'ostatus' ;
2010-02-08 19:06:03 +00:00
}
function handle ( $notice )
{
assert ( $notice instanceof Notice );
2010-02-24 02:19:13 +00:00
$this -> notice = $notice ;
2013-08-18 12:04:58 +01:00
$this -> user = User :: getKV ( 'id' , $notice -> profile_id );
2010-02-24 02:19:13 +00:00
2011-09-18 19:36:49 +01:00
try {
$profile = $this -> notice -> getProfile ();
} catch ( Exception $e ) {
common_log ( LOG_ERR , " Can't get profile for notice; skipping: " . $e -> getMessage ());
return true ;
}
2016-01-08 00:31:47 +00:00
foreach ( $notice -> getAttentionProfiles () as $target ) {
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: Attention target profile { $target -> getNickname () } ( { $target -> getID () } ) " );
if ( $target -> isGroup ()) {
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: { $target -> getID () } is a group " );
$oprofile = Ostatus_profile :: getKV ( 'group_id' , $target -> getGroup () -> getID ());
if ( ! $oprofile instanceof Ostatus_profile ) {
// we don't save profiles like this yet, but in the future
$oprofile = Ostatus_profile :: getKV ( 'profile_id' , $target -> getID ());
}
if ( $oprofile instanceof Ostatus_profile ) {
// remote group
if ( $notice -> isLocal ()) {
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: notice is local and remote group with profile ID { $target -> getID () } gets a ping " );
$this -> pingReply ( $oprofile );
}
} else {
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: local group with profile id { $target -> getID () } gets pushed out " );
// local group
$this -> pushGroup ( $target -> getGroup ());
}
} elseif ( $notice -> isLocal ()) {
// Notices generated on other sites will have already
// pinged their reply-targets, so only do these things
// if the target is not a group and the notice is locally generated
$oprofile = Ostatus_profile :: getKV ( 'profile_id' , $target -> getID ());
if ( $oprofile instanceof Ostatus_profile ) {
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: Notice is local and { $target -> getID () } is remote profile, getting pingReply " );
2012-07-29 23:17:16 +01:00
$this -> pingReply ( $oprofile );
}
2010-02-24 02:19:13 +00:00
}
2010-02-10 02:32:52 +00:00
}
2012-07-29 23:17:16 +01:00
if ( $notice -> isLocal ()) {
2016-01-09 23:29:32 +00:00
// Notices generated on remote sites will have already
// been pushed to user's subscribers by their origin sites.
$this -> pushUser ();
2016-01-08 00:31:47 +00:00
try {
$parent = $this -> notice -> getParent ();
foreach ( $parent -> getAttentionProfiles () as $related ) {
if ( $related -> isGroup ()) {
// don't ping groups in parent notices since we might not be a member of them,
// though it could be useful if we study this and use it correctly
continue ;
}
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: parent notice { $parent -> getID () } has related profile id== { $related -> getID () } " );
// FIXME: don't ping twice in case someone is in both notice attention spans!
$oprofile = Ostatus_profile :: getKV ( 'profile_id' , $related -> getID ());
if ( $oprofile instanceof Ostatus_profile ) {
$this -> pingReply ( $oprofile );
2010-09-01 19:35:43 +01:00
}
}
2016-01-08 00:31:47 +00:00
} catch ( NoParentNoticeException $e ) {
// nothing to do then
2010-09-01 19:35:43 +01:00
}
2011-03-06 19:15:34 +00:00
2012-07-29 23:17:16 +01:00
foreach ( $notice -> getProfileTags () as $ptag ) {
$oprofile = Ostatus_profile :: getKV ( 'peopletag_id' , $ptag -> id );
if ( ! $oprofile ) {
$this -> pushPeopletag ( $ptag );
}
2011-03-06 19:15:34 +00:00
}
}
2010-02-10 20:47:42 +00:00
return true ;
2010-02-10 02:32:52 +00:00
}
2010-02-24 02:19:13 +00:00
function pushUser ()
2010-02-10 02:32:52 +00:00
{
2010-02-24 02:19:13 +00:00
if ( $this -> user ) {
2016-01-08 00:31:47 +00:00
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: pushing feed for local user { $this -> user -> getID () } " );
2010-02-24 02:19:13 +00:00
// For local posts, ping the PuSH hub to update their feed.
// http://identi.ca/api/statuses/user_timeline/1.atom
$feed = common_local_url ( 'ApiTimelineUser' ,
array ( 'id' => $this -> user -> id ,
'format' => 'atom' ));
$this -> pushFeed ( $feed , array ( $this , 'userFeedForNotice' ));
}
2010-02-08 19:06:03 +00:00
}
2016-01-08 00:31:47 +00:00
function pushGroup ( User_group $group )
2010-02-10 02:32:52 +00:00
{
2016-01-08 00:31:47 +00:00
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: pushing group ' { $group -> getNickname () } ' profile_id= { $group -> profile_id } " );
2010-02-24 02:19:13 +00:00
// For a local group, ping the PuSH hub to update its feed.
// Updates may come from either a local or a remote user.
2010-02-10 02:32:52 +00:00
$feed = common_local_url ( 'ApiTimelineGroup' ,
2016-01-08 00:31:47 +00:00
array ( 'id' => $group -> getID (),
2010-02-10 02:32:52 +00:00
'format' => 'atom' ));
2016-01-08 00:31:47 +00:00
$this -> pushFeed ( $feed , array ( $this , 'groupFeedForNotice' ), $group -> getID ());
2010-02-24 02:19:13 +00:00
}
2011-03-06 19:15:34 +00:00
function pushPeopletag ( $ptag )
{
2016-01-08 00:31:47 +00:00
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: pushing peopletag ' { $ptag -> id } ' " );
2011-03-06 19:15:34 +00:00
// For a local people tag, ping the PuSH hub to update its feed.
// Updates may come from either a local or a remote user.
$feed = common_local_url ( 'ApiTimelineList' ,
array ( 'id' => $ptag -> id ,
'user' => $ptag -> tagger ,
'format' => 'atom' ));
$this -> pushFeed ( $feed , array ( $this , 'peopletagFeedForNotice' ), $ptag );
}
2016-01-08 00:31:47 +00:00
function pingReply ( Ostatus_profile $oprofile )
2010-02-24 02:19:13 +00:00
{
if ( $this -> user ) {
2016-01-08 00:31:47 +00:00
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: pinging reply to { $oprofile -> localProfile () -> getNickname () } for local user ' { $this -> user -> getID () } ' " );
2010-02-24 20:36:36 +00:00
// For local posts, send a Salmon ping to the mentioned
// remote user or group.
// @fixme as an optimization we can skip this if the
// remote profile is subscribed to the author.
2010-02-26 19:21:21 +00:00
$oprofile -> notifyDeferred ( $this -> notice , $this -> user );
2010-02-24 02:19:13 +00:00
}
2010-02-10 22:58:39 +00:00
}
/**
* @ param string $feed URI to the feed
* @ param callable $callback function to generate Atom feed update if needed
* any additional params are passed to the callback .
*/
function pushFeed ( $feed , $callback )
{
2016-03-23 14:25:21 +00:00
// NOTE: external hub pings will not be fixed by
// our legacy_http thing!
2010-02-10 22:58:39 +00:00
$hub = common_config ( 'ostatus' , 'hub' );
if ( $hub ) {
$this -> pushFeedExternal ( $feed , $hub );
}
2016-03-23 14:25:21 +00:00
// If we used to be http but now are https, see if we find an http entry for this feed URL
// and then upgrade it. This self-healing feature needs to be enabled manually in config.
// This code is based on a patch by @hannes2peer@quitter.se
if ( common_config ( 'fix' , 'legacy_http' ) && parse_url ( $feed , PHP_URL_SCHEME ) === 'https' ) {
2016-03-23 16:52:02 +00:00
common_log ( LOG_DEBUG , " OSTATUS [ { $this -> notice -> getID () } ]: Searching for http scheme instead for HubSub feed topic: " . _ve ( $feed ));
2016-03-23 14:25:21 +00:00
$http_feed = str_replace ( 'https://' , 'http://' , $feed );
$sub = new HubSub ();
$sub -> topic = $http_feed ;
// If we find it we upgrade the rows in the hubsub table.
if ( $sub -> find ()) {
2016-03-23 16:52:02 +00:00
common_log ( LOG_INFO , " OSTATUS [ { $this -> notice -> getID () } ]: Found topic with http scheme for " . _ve ( $feed ) . " , will update the rows to use https instead! " );
2016-03-23 14:25:21 +00:00
// we found an http:// URL but we use https:// now
// so let's update the rows to reflect on this!
while ( $sub -> fetch ()) {
2016-03-23 16:52:02 +00:00
common_debug ( " OSTATUS [ { $this -> notice -> getID () } ]: Changing topic URL to https for feed callback " . _ve ( $sub -> callback ));
2016-03-23 14:25:21 +00:00
$orig = clone ( $sub );
$sub -> topic = $feed ;
// hashkey column will be set automagically in HubSub->onUpdateKeys through updateWithKeys
$sub -> updateWithKeys ( $orig );
unset ( $orig );
}
}
}
2010-02-10 02:32:52 +00:00
$sub = new HubSub ();
$sub -> topic = $feed ;
if ( $sub -> find ()) {
2010-02-10 22:58:39 +00:00
$args = array_slice ( func_get_args (), 2 );
$atom = call_user_func_array ( $callback , $args );
$this -> pushFeedInternal ( $atom , $sub );
2010-02-10 02:32:52 +00:00
} else {
2016-03-23 16:52:02 +00:00
common_log ( LOG_INFO , " OSTATUS [ { $this -> notice -> getID () } ]: No PuSH subscribers for $feed " );
2010-02-10 02:32:52 +00:00
}
}
2010-02-10 22:58:39 +00:00
/**
* Ping external hub about this update .
* The hub will pull the feed and check for new items later .
* Not guaranteed safe in an environment with database replication .
*
* @ param string $feed feed topic URI
* @ param string $hub PuSH hub URI
* @ fixme can consolidate pings for user & group posts
*/
function pushFeedExternal ( $feed , $hub )
{
$client = new HTTPClient ();
try {
$data = array ( 'hub.mode' => 'publish' ,
'hub.url' => $feed );
$response = $client -> post ( $hub , array (), $data );
if ( $response -> getStatus () == 204 ) {
common_log ( LOG_INFO , " PuSH ping to hub $hub for $feed ok " );
return true ;
} else {
common_log ( LOG_ERR , " PuSH ping to hub $hub for $feed failed with HTTP " .
$response -> getStatus () . ': ' .
$response -> getBody ());
}
} catch ( Exception $e ) {
common_log ( LOG_ERR , " PuSH ping to hub $hub for $feed failed: " . $e -> getMessage ());
return false ;
}
}
/**
* Queue up direct feed update pushes to subscribers on our internal hub .
2010-06-04 19:48:54 +01:00
* If there are a large number of subscriber sites , intermediate bulk
* distribution triggers may be queued .
2010-09-01 19:35:43 +01:00
*
2010-02-10 22:58:39 +00:00
* @ param string $atom update feed , containing only new / changed items
* @ param HubSub $sub open query of subscribers
*/
function pushFeedInternal ( $atom , $sub )
2010-02-10 02:32:52 +00:00
{
common_log ( LOG_INFO , " Preparing $sub->N PuSH distribution(s) for $sub->topic " );
2010-06-04 19:48:54 +01:00
$n = 0 ;
$batch = array ();
2010-02-10 02:32:52 +00:00
while ( $sub -> fetch ()) {
2010-06-04 19:48:54 +01:00
$n ++ ;
if ( $n < self :: MAX_UNBATCHED ) {
$sub -> distribute ( $atom );
} else {
$batch [] = $sub -> callback ;
if ( count ( $batch ) >= self :: BATCH_SIZE ) {
$sub -> bulkDistribute ( $atom , $batch );
$batch = array ();
}
}
}
2016-01-16 16:18:14 +00:00
if ( count ( $batch ) > 0 ) {
2010-06-04 19:48:54 +01:00
$sub -> bulkDistribute ( $atom , $batch );
2010-02-10 02:32:52 +00:00
}
}
2010-02-08 19:06:03 +00:00
/**
* Build a single - item version of the sending user ' s Atom feed .
* @ return string
*/
2010-02-24 02:19:13 +00:00
function userFeedForNotice ()
2010-02-08 19:06:03 +00:00
{
2010-03-03 20:51:23 +00:00
$atom = new AtomUserNoticeFeed ( $this -> user );
$atom -> addEntryFromNotice ( $this -> notice );
$feed = $atom -> getString ();
2010-02-08 19:06:03 +00:00
return $feed ;
}
2010-02-10 02:32:52 +00:00
2010-02-24 02:19:13 +00:00
function groupFeedForNotice ( $group_id )
2010-02-10 02:32:52 +00:00
{
2013-08-18 12:04:58 +01:00
$group = User_group :: getKV ( 'id' , $group_id );
2010-03-03 20:51:23 +00:00
$atom = new AtomGroupNoticeFeed ( $group );
$atom -> addEntryFromNotice ( $this -> notice );
$feed = $atom -> getString ();
2010-02-10 02:32:52 +00:00
return $feed ;
}
2011-03-06 19:15:34 +00:00
function peopletagFeedForNotice ( $ptag )
{
$atom = new AtomListNoticeFeed ( $ptag );
$atom -> addEntryFromNotice ( $this -> notice );
$feed = $atom -> getString ();
return $feed ;
}
2010-02-08 19:06:03 +00:00
}