| 
									
										
										
										
											2010-02-08 11:06:03 -08: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/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-23 15:22:34 +01:00
										 |  |  | if (!defined('GNUSOCIAL')) { exit(1); } | 
					
						
							| 
									
										
										
										
											2010-10-08 10:42:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |  * WebSub (previously PuSH) feed subscription record | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |  * @package Hub | 
					
						
							|  |  |  |  * @author Brion Vibber <brion@status.net> | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-08-18 12:10:44 +02:00
										 |  |  | class HubSub extends Managed_DataObject | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     public $__table = 'hubsub'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public $hashkey; // sha1(topic . '|' . $callback); (topic, callback) key is too long for myisam in utf8
 | 
					
						
							| 
									
										
										
										
											2015-02-12 18:18:55 +01: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 11:06:03 -08:00
										 |  |  |     public $secret; | 
					
						
							|  |  |  |     public $sub_start; | 
					
						
							|  |  |  |     public $sub_end; | 
					
						
							|  |  |  |     public $created; | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |     public $modified; | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |     static function hashkey($topic, $callback) | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         return sha1($topic . '|' . $callback); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-21 11:25:08 +02:00
										 |  |  |     public static function getByHashkey($topic, $callback) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return self::getKV('hashkey', self::hashkey($topic, $callback)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-21 11:01:31 +02:00
										 |  |  |     public static function schemaDef() | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-08-21 11:01:31 +02:00
										 |  |  |         return array( | 
					
						
							|  |  |  |             'fields' => array( | 
					
						
							|  |  |  |                 'hashkey' => array('type' => 'char', 'not null' => true, 'length' => 40, 'description' => 'HubSub hashkey'), | 
					
						
							| 
									
										
										
										
											2015-02-12 18:18:55 +01: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 11:01:31 +02:00
										 |  |  |                 'secret' => array('type' => 'text', 'description' => 'HubSub stored secret'), | 
					
						
							|  |  |  |                 '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( | 
					
						
							| 
									
										
										
										
											2016-01-13 22:57:42 +01:00
										 |  |  |                 'hubsub_callback_idx' => array('callback'), | 
					
						
							| 
									
										
										
										
											2013-08-21 11:01:31 +02:00
										 |  |  |                 'hubsub_topic_idx' => array('topic'), | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08: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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         common_debug('WebSub hub got requested lease_seconds=='._ve($length)); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         assert(is_int($length)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-13 19:24:38 +01:00
										 |  |  |         $min = 86400;   // 3600*24 (one day)
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         $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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         common_debug('WebSub hub after sanitation: lease_seconds=='._ve($length)); | 
					
						
							| 
									
										
										
										
											2016-01-13 19:23:34 +01:00
										 |  |  |         $this->sub_start = common_sql_now(); | 
					
						
							|  |  |  |         $this->sub_end = common_sql_date(time() + $length); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-13 19:55:17 +01:00
										 |  |  |     function getLeaseTime() | 
					
						
							| 
									
										
										
										
											2016-01-13 19:45:20 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (empty($this->sub_start) || empty($this->sub_end)) { | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $length = strtotime($this->sub_end) - strtotime($this->sub_start); | 
					
						
							|  |  |  |         assert($length > 0); | 
					
						
							|  |  |  |         return $length; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function getLeaseRemaining() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (empty($this->sub_end)) { | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return strtotime($this->sub_end) - time(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |      * Schedule a future verification ping to the subscriber. | 
					
						
							|  |  |  |      * If queues are disabled, will be immediate. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08: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 11:06:03 -08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08: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 20:02:28 +01:00
										 |  |  |                       'token' => $token,    // let's put it in there if remote uses PuSH <0.4
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                       'retries' => $retries); | 
					
						
							|  |  |  |         $qm = QueueManager::get(); | 
					
						
							| 
									
										
										
										
											2010-02-24 20:36:36 +00:00
										 |  |  |         $qm->enqueue($data, 'hubconf'); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |     public function getTopic() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->topic; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08: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 11:06:03 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         assert($mode == 'subscribe' || $mode == 'unsubscribe'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-21 13:20:30 +02:00
										 |  |  |         $challenge = common_random_hexstr(32); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         $params = array('hub.mode' => $mode, | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |                         'hub.topic' => $this->getTopic(), | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                         'hub.challenge' => $challenge); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         if ($mode == 'subscribe') { | 
					
						
							| 
									
										
										
										
											2016-01-13 19:55:17 +01:00
										 |  |  |             $params['hub.lease_seconds'] = $this->getLeaseTime(); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-11-02 20:02:28 +01: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 11:06:03 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         // Any existing query string parameters must be preserved
 | 
					
						
							|  |  |  |         $url = $this->callback; | 
					
						
							| 
									
										
										
										
											2010-03-09 01:24:21 -05:00
										 |  |  |         if (strpos($url, '?') !== false) { | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08: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 19:55:02 +01:00
										 |  |  |             common_log(LOG_INFO, "Verified {$mode} of {$this->callback}:{$this->getTopic()}"); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-09-19 15:17:36 +02: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 14:46:26 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |         $old = HubSub::getByHashkey($this->getTopic(), $this->callback); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         if ($mode == 'subscribe') { | 
					
						
							| 
									
										
										
										
											2013-11-02 17:42:32 +01:00
										 |  |  |             if ($old instanceof HubSub) { | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                 $this->update($old); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                 $ok = $this->insert(); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         } else if ($mode == 'unsubscribe') { | 
					
						
							| 
									
										
										
										
											2013-11-02 17:42:32 +01:00
										 |  |  |             if ($old instanceof HubSub) { | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                 $old->delete(); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // That's ok, we're already unsubscribed.
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-23 15:22:34 +01:00
										 |  |  |     // set the hashkey automagically on insert
 | 
					
						
							|  |  |  |     protected function onInsert() | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-03-23 15:22:34 +01:00
										 |  |  |         $this->setHashkey(); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         $this->created = common_sql_now(); | 
					
						
							|  |  |  |         $this->modified = common_sql_now(); | 
					
						
							| 
									
										
										
										
											2016-03-23 15:22:34 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // update the hashkey automagically if needed
 | 
					
						
							|  |  |  |     protected function onUpdateKeys(Managed_DataObject $orig) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($this->topic !== $orig->topic || $this->callback !== $orig->callback) { | 
					
						
							|  |  |  |             $this->setHashkey(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected function setHashkey() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->hashkey = self::hashkey($this->topic, $this->callback); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08: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-01 18:46:34 -08: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 19:55:02 +01:00
										 |  |  |         $sub = HubSub::getByHashkey($this->getTopic(), $this->callback); | 
					
						
							| 
									
										
										
										
											2010-03-01 18:46:34 -08:00
										 |  |  |         $data = array('sub' => $sub, | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |                       'atom' => $atom, | 
					
						
							|  |  |  |                       'retries' => $retries); | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         common_log(LOG_INFO, "Queuing WebSub: {$this->getTopic()} to {$this->callback}"); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |         $qm = QueueManager::get(); | 
					
						
							|  |  |  |         $qm->enqueue($data, 'hubout'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Queue up a large batch of pushes to multiple subscribers | 
					
						
							|  |  |  |      * for this same topic update. | 
					
						
							| 
									
										
										
										
											2010-09-03 01:35:04 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |      * If queues are disabled, this will run immediately. | 
					
						
							| 
									
										
										
										
											2010-09-03 01:35:04 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |      * @param string $atom well-formed Atom feed | 
					
						
							|  |  |  |      * @param array $pushCallbacks list of callback URLs | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-01-16 17:18:14 +01:00
										 |  |  |     function bulkDistribute($atom, array $pushCallbacks) | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-01-16 17:34:27 +01:00
										 |  |  |         if (empty($pushCallbacks)) { | 
					
						
							|  |  |  |             common_log(LOG_ERR, 'Callback list empty for bulkDistribute.'); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |         $data = array('atom' => $atom, | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |                       'topic' => $this->getTopic(), | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |                       'pushCallbacks' => $pushCallbacks); | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         common_log(LOG_INFO, "Queuing WebSub batch: {$this->getTopic()} to ".count($pushCallbacks)." sites"); | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |         $qm = QueueManager::get(); | 
					
						
							|  |  |  |         $qm->enqueue($data, 'hubprep'); | 
					
						
							| 
									
										
										
										
											2016-01-16 17:34:27 +01:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2010-06-04 11:48:54 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08: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 14:28:06 -08:00
										 |  |  |      * @throws Exception (HTTP or general) | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08: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 11:06:03 -08:00
										 |  |  |             $headers[] = "X-Hub-Signature: sha1=$hmac"; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $hmac = '(none)'; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |         common_log(LOG_INFO, "About to push feed to $this->callback for {$this->getTopic()}, HMAC $hmac"); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |         $request = new HTTPClient(); | 
					
						
							| 
									
										
										
										
											2016-01-12 14:31:14 +01:00
										 |  |  |         $request->setConfig(array('follow_redirects' => false)); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |         $request->setBody($atom); | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |         try { | 
					
						
							|  |  |  |             $response = $request->post($this->callback, $headers); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |             if ($response->isOk()) { | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             $response = null; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |             common_debug('WebSub callback to '._ve($this->callback).' for '._ve($this->getTopic()).' failed with exception: '._ve($e->getMessage())); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01: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
 | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         // from the remote side, combined with implemented WebSub subscription timeouts.
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |         // We failed the WebSub, but it might be that the remote site has changed their configuration to HTTPS
 | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |         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); | 
					
						
							| 
									
										
										
										
											2016-01-14 13:15:31 +01:00
										 |  |  |             $alreadyreplaced = self::getByHashKey($this->getTopic(), $httpscallback); | 
					
						
							| 
									
										
										
										
											2016-01-14 13:06:37 +01:00
										 |  |  |             if ($alreadyreplaced instanceof HubSub) { | 
					
						
							|  |  |  |                 $this->delete(); | 
					
						
							|  |  |  |                 throw new AlreadyFulfilledException('The remote side has already established an HTTPS callback, deleting the legacy HTTP entry.'); | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-01-14 13:06:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 11:04:27 +02:00
										 |  |  |             common_debug('WebSub callback to '._ve($this->callback).' for '._ve($this->getTopic()).' trying HTTPS callback: '._ve($httpscallback)); | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |             $response = $request->post($httpscallback, $headers); | 
					
						
							|  |  |  |             if ($response->isOk()) { | 
					
						
							|  |  |  |                 $orig = clone($this); | 
					
						
							|  |  |  |                 $this->callback = $httpscallback; | 
					
						
							| 
									
										
										
										
											2016-03-23 15:22:34 +01:00
										 |  |  |                 // NOTE: hashkey will be set in $this->onUpdateKeys($orig) through updateWithKeys
 | 
					
						
							| 
									
										
										
										
											2016-01-28 16:42:59 +01:00
										 |  |  |                 $this->updateWithKeys($orig); | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01:00
										 |  |  |                 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 20:10:38 +01:00
										 |  |  |             throw new Exception(sprintf(_m('HTTP request failed without response to URL: %s'), _ve(isset($httpscallback) ? $httpscallback : $this->callback))); | 
					
						
							| 
									
										
										
										
											2016-01-11 19:55:02 +01: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 11:06:03 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | } |