Location: PHPKode > scripts > JAXL > abhinavsingh-JAXL-c5dba43/core/jaxl_socket_client.php
<?php
/**
* Jaxl (Jabber XMPP Library)
*
* Copyright (c) 2009-2012, Abhinav Singh <hide@address.com>.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* * Neither the name of Abhinav Singh nor the names of his
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRIC
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
*/

require_once JAXL_CWD.'/core/jaxl_loop.php';

/**
 * 
 * Enter description here ...
 * @author abhinavsingh
 *
 */
class JAXLSocketClient {
	
	private $host = null;
	private $port = null;
	private $transport = null;
	private $stream_context = null;
	private $blocking = false;
	
	public $fd = null;
	
	public $errno = null;
	public $errstr = null;
	private $timeout = 10;
	
	private $ibuffer = "";
	private $obuffer = "";
	private $compressed = false;
	
	private $recv_bytes = 0;
	private $send_bytes = 0;
	
	private $recv_cb = null;
	private $recv_chunk_size = 1024;
	private $writing = false;
	
	public function __construct($stream_context=null) {
		$this->stream_context = $stream_context;
	}
	
	public function __destruct() {
		//_debug("cleaning up xmpp socket...");
		$this->disconnect();
	}
	
	public function set_callback($recv_cb) {
		$this->recv_cb = $recv_cb;
	}
	
	public function connect($socket_path) {
		$path_parts = explode(":", $socket_path);
		$this->transport = 	$path_parts[0];
		$this->host = substr($path_parts[1], 2, strlen($path_parts[1]));
		if(sizeof($path_parts) == 3) $this->port = $path_parts[2];
		
		_info("trying ".$socket_path);
		if($this->stream_context) $this->fd = @stream_socket_client($socket_path, $this->errno, $this->errstr, $this->timeout, STREAM_CLIENT_CONNECT, $this->stream_context);
		else $this->fd = @stream_socket_client($socket_path, $this->errno, $this->errstr, $this->timeout);
		
		if($this->fd) {
			_debug("connected to ".$socket_path."");
			stream_set_blocking($this->fd, $this->blocking);
			
			// watch descriptor for read/write events
			JAXLLoop::watch($this->fd, array(
				'read' => array(&$this, 'on_read_ready')
			));
			
			return true;
		}
		else {
			_error("unable to connect ".$socket_path." with error no: ".$this->errno.", error str: ".$this->errstr."");
			$this->disconnect();
			return false;
		}
	}
	
	public function disconnect() {
		@fclose($this->fd);
		$this->fd = null;
	}
	
	public function compress() {
		$this->compressed = true;
		//stream_filter_append($this->fd, 'zlib.inflate', STREAM_FILTER_READ);
		//stream_filter_append($this->fd, 'zlib.deflate', STREAM_FILTER_WRITE);
	}
	
	public function crypt() {
		// set blocking (since tls negotiation fails if stream is non-blocking)
		stream_set_blocking($this->fd, true);
		
		$ret = stream_socket_enable_crypto($this->fd, true, STREAM_CRYPTO_METHOD_TLS_CLIENT);
		if($ret == false) {
			$ret = stream_socket_enable_crypto($this->fd, true, STREAM_CRYPTO_METHOD_SSLv3_CLIENT);
			if($ret == false) {
				$ret = stream_socket_enable_crypto($this->fd, true, STREAM_CRYPTO_METHOD_SSLv2_CLIENT);
				if($ret == false) {
					$ret = stream_socket_enable_crypto($this->fd, true, STREAM_CRYPTO_METHOD_SSLv23_CLIENT);
				}
			}
		}
		
		// switch back to non-blocking
		stream_set_blocking($this->fd, false);
	}
	
	public function send($data) {
		$this->obuffer .= $data;
		
		// add watch for write events
		if($this->writing) return;
		JAXLLoop::watch($this->fd, array(
			'write' => array(&$this, 'on_write_ready')
		));
		$this->writing = true;
	}
	
	public function on_read_ready($fd) {
		//_debug("on read ready called");
		$raw = @fread($fd, $this->recv_chunk_size);
		$bytes = strlen($raw);
		
		if($bytes === 0) {
			$meta = stream_get_meta_data($fd);
			if($meta['eof'] === TRUE) {
				_warning("socket eof, disconnecting");
				JAXLLoop::unwatch($fd, array(
					'read' => true
				));
				$this->disconnect();
				return;
			}
		}
		
		$this->recv_bytes += $bytes;
		$total = $this->ibuffer.$raw;
		
		$this->ibuffer = "";
		_debug("read ".$bytes."/".$this->recv_bytes." of data");
		if($bytes > 0) _debug($raw);
		
		// callback
		if($this->recv_cb) call_user_func($this->recv_cb, $raw);
	}
	
	public function on_write_ready($fd) {
		//_debug("on write ready called");
		$total = strlen($this->obuffer);
		$bytes = @fwrite($fd, $this->obuffer);
		$this->send_bytes += $bytes;
		
		_debug("sent ".$bytes."/".$this->send_bytes." of data");
		_debug($this->obuffer);
		
		$this->obuffer = substr($this->obuffer, $bytes, $total-$bytes);
		
		// unwatch for write if obuffer is empty
		if(strlen($this->obuffer) === 0) {
			JAXLLoop::unwatch($fd, array(
				'write' => true
			));
			$this->writing = false;
		}
		
		//_debug("current obuffer size: ".strlen($this->obuffer)."");
	}
	
}

?>
Return current item: JAXL