<?php /** * XMPPHP: The PHP XMPP Library * Copyright (C) 2008 Nathanael C. Fritz * This file is part of SleekXMPP. * * XMPPHP is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * XMPPHP 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with XMPPHP; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * * @category xmpphp * @package XMPPHP * @author Nathanael C. Fritz <JID: fritzy@netflint.net> * @author Stephan Wentz <JID: stephan@jabber.wentz.it> * @author Michael Garvin <JID: gar@netflint.net> * @author Alexander Birkner (https://github.com/BirknerAlex) * @author zorn-v (https://github.com/zorn-v/xmpphp/) * @author GNU social * @copyright 2008 Nathanael C. Fritz */ namespace XMPPHP; use SimpleXMLElement; /** XMPPHP_XMLStream */ require_once __DIR__ . DIRECTORY_SEPARATOR . 'XMPP.php'; /** * XMPPHP BOSH * * @property int lat * @package XMPPHP * @author Nathanael C. Fritz <JID: fritzy@netflint.net> * @author Stephan Wentz <JID: stephan@jabber.wentz.it> * @author Michael Garvin <JID: gar@netflint.net> * @copyright 2008 Nathanael C. Fritz * @version $Id$ */ class BOSH extends XMPP { /** * @var int */ protected $rid; /** * @var string */ protected $sid; /** * @var string */ protected $http_server; /** * @var array */ protected $http_buffer = []; /** * @var string */ protected $session = false; /** * @var int */ protected $inactivity; public function __construct( string $host, int $port, string $user, string $password, string $resource, ?string $server = null, bool $print_log = false, ?string $log_level = null ) { parent::__construct($host, $port, $user, $password, $resource, $server, $print_log, $log_level); if (is_null($server)) { // If we aren't given the server http url, try and guess it $port_string = ($this->port and $this->port != 80) ? ':' . $this->port : ''; $this->http_server = 'http://' . $this->host . $port_string . '/http-bind/'; } else { $this->http_server = $server; } } /** * Connect * * @param bool $persistent * @param bool $send_init * @param int $timeout * @throws Exception */ public function connect(bool $persistent = false, bool $send_init = true, int $timeout = 30): void { $this->use_encryption = false; $this->session = $persistent; $this->rid = 3001; $this->sid = null; $this->inactivity = 0; if ($persistent) { $this->loadSession(); } if (!$this->sid) { $body = $this->__buildBody(); $body->addAttribute('hold', '1'); $body->addAttribute('to', $this->server); $body->addAttribute('route', 'xmpp:' . $this->host . ':' . $this->port); $body->addAttribute('secure', 'true'); $body->addAttribute('xmpp:version', '1.0', 'urn:xmpp:xbosh'); $body->addAttribute('wait', strval($timeout)); $body->addAttribute('ack', '1'); $body->addAttribute('xmlns:xmpp', 'urn:xmpp:xbosh'); $buff = '<stream:stream xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams">'; xml_parse($this->parser, $buff, false); $response = $this->__sendBody($body); $rxml = new SimpleXMLElement($response); $this->sid = $rxml['sid']; $this->inactivity = $rxml['inactivity']; } else { $buff = '<stream:stream xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams">'; xml_parse($this->parser, $buff, false); } } /** * Load session * */ public function loadSession(): void { if ($this->session == 'ON_FILE') { // Session not started so use session_file $session_file = $this->getSessionFile(); // manage multiple accesses if (!file_exists($session_file)) { file_put_contents($session_file, ''); } $session_file_fp = fopen($session_file, 'r'); flock($session_file_fp, LOCK_EX); $session_serialized = file_get_contents($session_file, null, null, 6); flock($session_file_fp, LOCK_UN); fclose($session_file_fp); $this->log->log('SESSION: reading ' . $session_serialized . ' from ' . $session_file, Log::LEVEL_VERBOSE); if ($session_serialized != '') { $_SESSION['XMPPHP_BOSH'] = unserialize($session_serialized); } } if (isset($_SESSION['XMPPHP_BOSH']['inactivity'])) { $this->inactivity = $_SESSION['XMPPHP_BOSH']['inactivity']; } $this->lat = (time() - (isset($_SESSION['XMPPHP_BOSH']['lat']))) ? $_SESSION['XMPPHP_BOSH']['lat'] : 0; if ($this->lat < $this->inactivity) { if (isset($_SESSION['XMPPHP_BOSH']['RID'])) { $this->rid = $_SESSION['XMPPHP_BOSH']['RID']; } if (isset($_SESSION['XMPPHP_BOSH']['SID'])) { $this->sid = $_SESSION['XMPPHP_BOSH']['SID']; } if (isset($_SESSION['XMPPHP_BOSH']['authed'])) { $this->authed = $_SESSION['XMPPHP_BOSH']['authed']; } if (isset($_SESSION['XMPPHP_BOSH']['basejid'])) { $this->basejid = $_SESSION['XMPPHP_BOSH']['basejid']; } if (isset($_SESSION['XMPPHP_BOSH']['fulljid'])) { $this->fulljid = $_SESSION['XMPPHP_BOSH']['fulljid']; } } } /** * Get the session file location * */ public function getSessionFile(): string { return sys_get_temp_dir() . '/' . $this->user . '_' . $this->server . '_session'; } /** * Build body * * @param SimpleXMLElement|null $sub * @return SimpleXMLElement */ private function __buildBody(?SimpleXMLElement $sub = null): SimpleXMLElement { $xml = new SimpleXMLElement('<body xmlns="http://jabber.org/protocol/httpbind" xmlns:xmpp="urn:xmpp:xbosh" />'); $xml->addAttribute('content', 'text/xml; charset=utf-8'); $xml->addAttribute('rid', $this->rid); ++$this->rid; if ($this->sid) { $xml->addAttribute('sid', $this->sid); } $xml->addAttribute('xml:lang', 'en'); if ($sub !== null) { // Ok, so simplexml is lame $parent = dom_import_simplexml($xml); $content = dom_import_simplexml($sub); $child = $parent->ownerDocument->importNode($content, true); $parent->appendChild($child); $xml = simplexml_import_dom($parent); } return $xml; } /** * Send body * * @param SimpleXMLElement|null $body * @param bool $recv * @return bool|string * @throws Exception */ private function __sendBody(?SimpleXMLElement $body = null, bool $recv = true) { if (!$body) { $body = $this->__buildBody(); } $output = ''; $header = ['Accept-Encoding: gzip, deflate', 'Content-Type: text/xml; charset=utf-8']; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $this->http_server); curl_setopt($ch, CURLOPT_HEADER, 0); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_POSTFIELDS, $body->asXML()); curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true); curl_setopt($ch, CURLOPT_HTTPHEADER, $header); curl_setopt($ch, CURLOPT_VERBOSE, 0); if ($recv) { curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $output = curl_exec($ch); if (curl_getinfo($ch, CURLINFO_HTTP_CODE) != '200') { throw new Exception('Wrong response from server!'); } $this->http_buffer[] = $output; } curl_close($ch); return $output; } /** * Process * * @param $null1 * @param $null2 * * null params are not used and just to statify Strict Function Declaration * @return bool * @throws Exception * @throws Exception */ private function __process($null1 = null, $null2 = null) { if ($this->http_buffer) { $this->__parseBuffer(); } else { $this->__sendBody(); $this->__parseBuffer(); } $this->saveSession(); return true; } private function __parseBuffer() { while ($this->http_buffer) { $idx = key($this->http_buffer); $buffer = $this->http_buffer[$idx]; unset($this->http_buffer[$idx]); if ($buffer) { $xml = new SimpleXMLElement($buffer); $children = $xml->xpath('child::node()'); foreach ($children as $child) { $buff = $child->asXML(); $this->log->log('RECV: ' . $buff, Log::LEVEL_VERBOSE); xml_parse($this->parser, $buff, false); } } } } /** * Save session * */ public function saveSession(): void { $_SESSION['XMPPHP_BOSH']['RID'] = (string)$this->rid; $_SESSION['XMPPHP_BOSH']['SID'] = (string)$this->sid; $_SESSION['XMPPHP_BOSH']['authed'] = (boolean)$this->authed; $_SESSION['XMPPHP_BOSH']['basejid'] = (string)$this->basejid; $_SESSION['XMPPHP_BOSH']['fulljid'] = (string)$this->fulljid; $_SESSION['XMPPHP_BOSH']['inactivity'] = (string)$this->inactivity; $_SESSION['XMPPHP_BOSH']['lat'] = (string)time(); if ($this->session == 'ON_FILE') { $session_file = $this->getSessionFile(); $session_file_fp = fopen($session_file, 'r'); flock($session_file_fp, LOCK_EX); // <?php prefix used to mask the content of the session file $session_serialized = '<?php ' . serialize($_SESSION); file_put_contents($session_file, $session_serialized); flock($session_file_fp, LOCK_UN); fclose($session_file_fp); } } /** * Process * * @param $msg * @param int|null $_ unused * @throws Exception */ public function send($msg, ?int $_ = null) { $this->log->log('SEND: ' . $msg, Log::LEVEL_VERBOSE); $msg = new SimpleXMLElement($msg); $this->__sendBody($this->__buildBody($msg), true); } /** * Reset * * @throws Exception */ public function reset(): void { $this->xml_depth = 0; unset($this->xmlobj); $this->xmlobj = []; $this->setupParser(); $body = $this->__buildBody(); $body->addAttribute('to', $this->host); $body->addAttribute('xmpp:restart', 'true', 'urn:xmpp:xbosh'); $buff = '<stream:stream xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams">'; $this->__sendBody($body); $this->been_reset = true; xml_parse($this->parser, $buff, false); } /** * Disconnect * * @throws Exception */ public function disconnect(): void { parent::disconnect(); if ($this->session == 'ON_FILE') { unlink($this->getSessionFile()); } else { $keys = ['RID', 'SID', 'authed', 'basejid', 'fulljid', 'inactivity', 'lat']; foreach ($keys as $key) { unset($_SESSION['XMPPHP_BOSH'][$key]); } } } }