| 
									
										
										
										
											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/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-08 10:42:59 -07:00
										 |  |  | if (!defined('STATUSNET')) { | 
					
						
							|  |  |  |     exit(1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * PuSH feed subscription record | 
					
						
							|  |  |  |  * @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 $lease; | 
					
						
							|  |  |  |     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
										 |  |  | 
 | 
					
						
							|  |  |  |     protected static function hashkey($topic, $callback) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         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'), | 
					
						
							| 
									
										
										
										
											2015-02-07 10:46:13 -05:00
										 |  |  |                 'lease' => array('type' => 'int', 'description' => 'HubSub leasetime'), | 
					
						
							| 
									
										
										
										
											2013-08-21 11:01:31 +02: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 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) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->lease = $length; | 
					
						
							|  |  |  |         $this->start_sub = common_sql_now(); | 
					
						
							|  |  |  |         $this->end_sub = common_sql_date(time() + $length); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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, | 
					
						
							|  |  |  |                         'hub.topic' => $this->topic, | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |                         'hub.challenge' => $challenge); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         if ($mode == 'subscribe') { | 
					
						
							|  |  |  |             $params['hub.lease_seconds'] = $this->lease; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2013-11-02 17:42:32 +01:00
										 |  |  |             common_log(LOG_INFO, "Verified {$mode} of {$this->callback}:{$this->topic}"); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-21 11:25:08 +02:00
										 |  |  |         $old = HubSub::getByHashkey($this->topic, $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
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Insert wrapper; transparently set the hash key from topic and callback columns. | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |      * @return mixed success | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     function insert() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->hashkey = self::hashkey($this->topic, $this->callback); | 
					
						
							| 
									
										
										
										
											2010-02-21 14:46:26 -08:00
										 |  |  |         $this->created = common_sql_now(); | 
					
						
							|  |  |  |         $this->modified = common_sql_now(); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  |         return parent::insert(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							| 
									
										
										
										
											2013-08-21 11:25:08 +02:00
										 |  |  |         $sub = HubSub::getByHashkey($this->topic, $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); | 
					
						
							| 
									
										
										
										
											2010-03-01 18:46:34 -08:00
										 |  |  |         common_log(LOG_INFO, "Queuing PuSH: $this->topic 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 | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function bulkDistribute($atom, $pushCallbacks) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							|  |  |  |                       'topic' => $this->topic, | 
					
						
							|  |  |  |                       'pushCallbacks' => $pushCallbacks); | 
					
						
							|  |  |  |         common_log(LOG_INFO, "Queuing PuSH batch: $this->topic to " . | 
					
						
							|  |  |  |                              count($pushCallbacks) . " sites"); | 
					
						
							|  |  |  |         $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)'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         common_log(LOG_INFO, "About to push feed to $this->callback for $this->topic, HMAC $hmac"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |         $request = new HTTPClient(); | 
					
						
							|  |  |  |         $request->setBody($atom); | 
					
						
							|  |  |  |         $response = $request->post($this->callback, $headers); | 
					
						
							| 
									
										
										
										
											2010-02-08 11:06:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 14:28:06 -08:00
										 |  |  |         if ($response->isOk()) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-09-19 15:17:36 +02: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
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |