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
/**
* PuSH feed subscription record
* @ package Hub
* @ author Brion Vibber < brion @ status . net >
*/
2013-08-18 11:10:44 +01:00
class HubSub extends Managed_DataObject
2010-02-08 19:06:03 +00:00
{
public $__table = 'hubsub' ;
public $hashkey ; // sha1(topic . '|' . $callback); (topic, callback) key is too long for myisam in utf8
2015-02-12 17:18:55 +00:00
public $topic ; // varchar(191) not 255 because utf8mb4 takes more space
public $callback ; // varchar(191) not 255 because utf8mb4 takes more space
2010-02-08 19:06:03 +00:00
public $secret ;
public $lease ;
public $sub_start ;
public $sub_end ;
public $created ;
2010-02-21 22:46:26 +00:00
public $modified ;
2010-02-08 19:06:03 +00:00
2016-01-11 18:55:02 +00:00
static function hashkey ( $topic , $callback )
2010-02-08 19:06:03 +00:00
{
return sha1 ( $topic . '|' . $callback );
}
2013-08-21 10:25:08 +01:00
public static function getByHashkey ( $topic , $callback )
{
return self :: getKV ( 'hashkey' , self :: hashkey ( $topic , $callback ));
}
2013-08-21 10:01:31 +01:00
public static function schemaDef ()
2010-02-08 19:06:03 +00:00
{
2013-08-21 10:01:31 +01:00
return array (
'fields' => array (
'hashkey' => array ( 'type' => 'char' , 'not null' => true , 'length' => 40 , 'description' => 'HubSub hashkey' ),
2015-02-12 17:18:55 +00:00
'topic' => array ( 'type' => 'varchar' , 'not null' => true , 'length' => 191 , 'description' => 'HubSub topic' ),
'callback' => array ( 'type' => 'varchar' , 'not null' => true , 'length' => 191 , 'description' => 'HubSub callback' ),
2013-08-21 10:01:31 +01:00
'secret' => array ( 'type' => 'text' , 'description' => 'HubSub stored secret' ),
2015-02-07 15:46:13 +00:00
'lease' => array ( 'type' => 'int' , 'description' => 'HubSub leasetime' ),
2013-08-21 10:01:31 +01:00
'sub_start' => array ( 'type' => 'datetime' , 'description' => 'subscription start' ),
'sub_end' => array ( 'type' => 'datetime' , 'description' => 'subscription end' ),
'created' => array ( 'type' => 'datetime' , 'not null' => true , 'description' => 'date this record was created' ),
'modified' => array ( 'type' => 'timestamp' , 'not null' => true , 'description' => 'date this record was modified' ),
),
'primary key' => array ( 'hashkey' ),
'indexes' => array (
'hubsub_topic_idx' => array ( 'topic' ),
),
);
2010-02-08 19:06:03 +00:00
}
/**
* Validates a requested lease length , sets length plus
* subscription start & end dates .
*
* Does not save to database -- use before insert () or update () .
*
* @ param int $length in seconds
*/
function setLease ( $length )
{
2016-01-13 18:23:34 +00:00
common_debug ( 'PuSH hub got requested lease_seconds==' . _ve ( $length ));
2010-02-08 19:06:03 +00:00
assert ( is_int ( $length ));
$min = 86400 ;
$max = 86400 * 30 ;
if ( $length == 0 ) {
// We want to garbage collect dead subscriptions!
$length = $max ;
} elseif ( $length < $min ) {
$length = $min ;
} else if ( $length > $max ) {
$length = $max ;
}
2016-01-13 18:23:34 +00:00
common_debug ( 'PuSH hub after sanitation: lease_seconds==' . _ve ( $length ));
2010-02-08 19:06:03 +00:00
$this -> lease = $length ;
2016-01-13 18:23:34 +00:00
$this -> sub_start = common_sql_now ();
$this -> sub_end = common_sql_date ( time () + $length );
2010-02-08 19:06:03 +00:00
}
/**
2010-02-21 22:46:26 +00:00
* Schedule a future verification ping to the subscriber .
* If queues are disabled , will be immediate .
*
2010-02-08 19:06:03 +00:00
* @ param string $mode 'subscribe' or 'unsubscribe'
2010-02-18 18:20:48 +00:00
* @ param string $token hub . verify_token value , if provided by client
2010-02-08 19:06:03 +00:00
*/
2010-02-21 22:46:26 +00:00
function scheduleVerify ( $mode , $token = null , $retries = null )
{
if ( $retries === null ) {
$retries = intval ( common_config ( 'ostatus' , 'hub_retries' ));
}
$data = array ( 'sub' => clone ( $this ),
'mode' => $mode ,
2013-11-02 19:02:28 +00:00
'token' => $token , // let's put it in there if remote uses PuSH <0.4
2010-02-21 22:46:26 +00:00
'retries' => $retries );
$qm = QueueManager :: get ();
2010-02-24 20:36:36 +00:00
$qm -> enqueue ( $data , 'hubconf' );
2010-02-21 22:46:26 +00:00
}
2016-01-11 18:55:02 +00:00
public function getTopic ()
{
return $this -> topic ;
}
2010-02-21 22:46:26 +00:00
/**
* Send a verification ping to subscriber , and if confirmed apply the changes .
* This may create , update , or delete the database record .
*
* @ param string $mode 'subscribe' or 'unsubscribe'
* @ param string $token hub . verify_token value , if provided by client
* @ throws ClientException on failure
*/
2010-02-18 18:20:48 +00:00
function verify ( $mode , $token = null )
2010-02-08 19:06:03 +00:00
{
assert ( $mode == 'subscribe' || $mode == 'unsubscribe' );
2013-10-21 12:20:30 +01:00
$challenge = common_random_hexstr ( 32 );
2010-02-08 19:06:03 +00:00
$params = array ( 'hub.mode' => $mode ,
2016-01-11 18:55:02 +00:00
'hub.topic' => $this -> getTopic (),
2010-02-21 22:46:26 +00:00
'hub.challenge' => $challenge );
2010-02-08 19:06:03 +00:00
if ( $mode == 'subscribe' ) {
$params [ 'hub.lease_seconds' ] = $this -> lease ;
}
2013-11-02 19:02:28 +00:00
if ( $token !== null ) { // TODO: deprecated in PuSH 0.4
$params [ 'hub.verify_token' ] = $token ; // let's put it in there if remote uses PuSH <0.4
2010-02-08 19:06:03 +00:00
}
2010-02-21 22:46:26 +00:00
// Any existing query string parameters must be preserved
$url = $this -> callback ;
2010-03-09 06:24:21 +00:00
if ( strpos ( $url , '?' ) !== false ) {
2010-02-21 22:46:26 +00:00
$url .= '&' ;
} else {
$url .= '?' ;
}
$url .= http_build_query ( $params , '' , '&' );
$request = new HTTPClient ();
$response = $request -> get ( $url );
$status = $response -> getStatus ();
if ( $status >= 200 && $status < 300 ) {
2016-01-11 18:55:02 +00:00
common_log ( LOG_INFO , " Verified { $mode } of { $this -> callback } : { $this -> getTopic () } " );
2010-02-21 22:46:26 +00:00
} else {
2010-09-19 14:17:36 +01:00
// TRANS: Client exception. %s is a HTTP status code.
throw new ClientException ( sprintf ( _m ( 'Hub subscriber verification returned HTTP %s.' ), $status ));
2010-02-21 22:46:26 +00:00
}
2010-02-08 19:06:03 +00:00
2016-01-11 18:55:02 +00:00
$old = HubSub :: getByHashkey ( $this -> getTopic (), $this -> callback );
2010-02-21 22:46:26 +00:00
if ( $mode == 'subscribe' ) {
2013-11-02 16:42:32 +00:00
if ( $old instanceof HubSub ) {
2010-02-21 22:46:26 +00:00
$this -> update ( $old );
2010-02-08 19:06:03 +00:00
} else {
2010-02-21 22:46:26 +00:00
$ok = $this -> insert ();
2010-02-08 19:06:03 +00:00
}
2010-02-21 22:46:26 +00:00
} else if ( $mode == 'unsubscribe' ) {
2013-11-02 16:42:32 +00:00
if ( $old instanceof HubSub ) {
2010-02-21 22:46:26 +00:00
$old -> delete ();
} else {
// That's ok, we're already unsubscribed.
2010-02-08 19:06:03 +00:00
}
}
}
/**
* Insert wrapper ; transparently set the hash key from topic and callback columns .
2010-02-21 22:46:26 +00:00
* @ return mixed success
2010-02-08 19:06:03 +00:00
*/
function insert ()
{
2016-01-11 18:55:02 +00:00
$this -> hashkey = self :: hashkey ( $this -> getTopic (), $this -> callback );
2010-02-21 22:46:26 +00:00
$this -> created = common_sql_now ();
$this -> modified = common_sql_now ();
2010-02-08 19:06:03 +00:00
return parent :: insert ();
}
2010-02-21 22:28:06 +00:00
/**
* Schedule delivery of a 'fat ping' to the subscriber ' s callback
* endpoint . If queues are disabled , this will run immediately .
*
* @ param string $atom well - formed Atom feed
* @ param int $retries optional count of retries if POST fails ; defaults to hub_retries from config or 0 if unset
*/
function distribute ( $atom , $retries = null )
{
if ( $retries === null ) {
$retries = intval ( common_config ( 'ostatus' , 'hub_retries' ));
}
2010-03-02 02:46:34 +00:00
// We dare not clone() as when the clone is discarded it'll
// destroy the result data for the parent query.
// @fixme use clone() again when it's safe to copy an
// individual item from a multi-item query again.
2016-01-11 18:55:02 +00:00
$sub = HubSub :: getByHashkey ( $this -> getTopic (), $this -> callback );
2010-03-02 02:46:34 +00:00
$data = array ( 'sub' => $sub ,
2010-02-21 22:28:06 +00:00
'atom' => $atom ,
'retries' => $retries );
2016-01-11 18:55:02 +00:00
common_log ( LOG_INFO , " Queuing PuSH: { $this -> getTopic () } to { $this -> callback } " );
2010-02-21 22:28:06 +00:00
$qm = QueueManager :: get ();
$qm -> enqueue ( $data , 'hubout' );
}
2010-06-04 19:48:54 +01:00
/**
* Queue up a large batch of pushes to multiple subscribers
* for this same topic update .
2010-09-03 00:35:04 +01:00
*
2010-06-04 19:48:54 +01:00
* If queues are disabled , this will run immediately .
2010-09-03 00:35:04 +01:00
*
2010-06-04 19:48:54 +01:00
* @ param string $atom well - formed Atom feed
* @ param array $pushCallbacks list of callback URLs
*/
function bulkDistribute ( $atom , $pushCallbacks )
{
$data = array ( 'atom' => $atom ,
2016-01-11 18:55:02 +00:00
'topic' => $this -> getTopic (),
2010-06-04 19:48:54 +01:00
'pushCallbacks' => $pushCallbacks );
2016-01-11 18:55:02 +00:00
common_log ( LOG_INFO , " Queuing PuSH batch: { $this -> getTopic () } to " . count ( $pushCallbacks ) . " sites " );
2010-06-04 19:48:54 +01:00
$qm = QueueManager :: get ();
$qm -> enqueue ( $data , 'hubprep' );
}
2010-02-08 19:06:03 +00:00
/**
* Send a 'fat ping' to the subscriber ' s callback endpoint
* containing the given Atom feed chunk .
*
* Determination of which items to send should be done at
* a higher level ; don ' t just shove in a complete feed !
*
* @ param string $atom well - formed Atom feed
2010-02-21 22:28:06 +00:00
* @ throws Exception ( HTTP or general )
2010-02-08 19:06:03 +00:00
*/
function push ( $atom )
{
$headers = array ( 'Content-Type: application/atom+xml' );
if ( $this -> secret ) {
2010-02-10 22:58:39 +00:00
$hmac = hash_hmac ( 'sha1' , $atom , $this -> secret );
2010-02-08 19:06:03 +00:00
$headers [] = " X-Hub-Signature: sha1= $hmac " ;
} else {
$hmac = '(none)' ;
}
2016-01-11 18:55:02 +00:00
common_log ( LOG_INFO , " About to push feed to $this->callback for { $this -> getTopic () } , HMAC $hmac " );
2010-02-08 19:06:03 +00:00
2010-02-21 22:28:06 +00:00
$request = new HTTPClient ();
2016-01-12 13:31:14 +00:00
$request -> setConfig ( array ( 'follow_redirects' => false ));
2010-02-21 22:28:06 +00:00
$request -> setBody ( $atom );
2016-01-11 18:55:02 +00:00
try {
$response = $request -> post ( $this -> callback , $headers );
2010-02-08 19:06:03 +00:00
2016-01-11 18:55:02 +00:00
if ( $response -> isOk ()) {
return true ;
}
} catch ( Exception $e ) {
$response = null ;
common_debug ( 'PuSH callback to ' . _ve ( $this -> callback ) . ' for ' . _ve ( $this -> getTopic ()) . ' failed with exception: ' . _ve ( $e -> getMessage ()));
2010-02-08 19:06:03 +00:00
}
2016-01-11 18:55:02 +00:00
// XXX: DO NOT trust a Location header here, _especially_ from 'http' protocols,
// but not 'https' either at least if we don't do proper CA verification. Trust that
// the most common change here is simply switching 'http' to 'https' and we will
// solve 99% of all of these issues for now. There should be a proper mechanism
// if we want to change the callback URLs, preferrably just manual resubscriptions
// from the remote side, combined with implemented PuSH subscription timeouts.
// We failed the PuSH, but it might be that the remote site has changed their configuration to HTTPS
if ( 'http' === parse_url ( $this -> callback , PHP_URL_SCHEME )) {
// Test if the feed callback for this node has migrated to HTTPS
$httpscallback = preg_replace ( '/^http/' , 'https' , $this -> callback , 1 );
if ( $httpscallback === $this -> callback ) {
throw new ServerException ( 'Trying to preg_replace http to https on ' . _ve ( $this -> callback ) . ' failed and resulted in an identical string: ' . _ve ( $httpscallback ) . '.' );
}
common_debug ( 'PuSH callback to ' . _ve ( $this -> callback ) . ' for ' . _ve ( $this -> getTopic ()) . ' testing with HTTPS callback: ' . _ve ( $httpscallback ));
$response = $request -> post ( $httpscallback , $headers );
if ( $response -> isOk ()) {
$orig = clone ( $this );
$this -> callback = $httpscallback ;
$this -> hashkey = self :: hashkey ( $this -> getTopic (), $this -> callback );
common_debug ( 'HubSub DEBUG, from ' . _ve ( $orig ) . ' to ' . _ve ( $this ));
$this -> updateWithKeys ( $orig , 'hashkey' );
return true ;
}
}
// FIXME: Add 'failed' incremental count for this callback.
if ( is_null ( $response )) {
// This means we got a lower-than-HTTP level error, like domain not found or maybe connection refused
// This should be using a more distinguishable exception class, but for now this will do.
2016-01-11 19:10:38 +00:00
throw new Exception ( sprintf ( _m ( 'HTTP request failed without response to URL: %s' ), _ve ( isset ( $httpscallback ) ? $httpscallback : $this -> callback )));
2016-01-11 18:55:02 +00:00
}
// TRANS: Exception. %1$s is a response status code, %2$s is the body of the response.
throw new Exception ( sprintf ( _m ( 'Callback returned status: %1$s. Body: %2$s' ),
$response -> getStatus (), trim ( $response -> getBody ())));
2010-02-08 19:06:03 +00:00
}
}