 d880d29729
			
		
	
	
		d880d29729
		
	
	
	
	
		
			
			Refactored Command/Composite pattern, now as expected Server recursively executes commands Above changes fixed issues of server/client not being notified on forced disconnects
		
			
				
	
	
		
			157 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| namespace Ratchet;
 | |
| use Ratchet\Server\Aggregator;
 | |
| use Ratchet\Protocol\ProtocolInterface;
 | |
| use Ratchet\Logging\LoggerInterface;
 | |
| use Ratchet\Logging\NullLogger;
 | |
| use Ratchet\Command\CommandInterface;
 | |
| 
 | |
| /**
 | |
|  * Creates an open-ended socket to listen on a port for incomming connections.  Events are delegated through this to attached applications
 | |
|  * @todo Consider using _connections as master reference and passing iterator_to_array(_connections) to socket_select
 | |
|  * @todo Currently passing Socket object down the decorated chain - should be sending reference to it instead; Receivers do not interact with the Socket directly, they do so through the Command pattern
 | |
|  */
 | |
| class Server implements SocketObserver, \IteratorAggregate {
 | |
|     /**
 | |
|      * The master socket, receives all connections
 | |
|      * @type Socket
 | |
|      */
 | |
|     protected $_master = null;
 | |
| 
 | |
|     /**
 | |
|      * @var array of Socket Resources
 | |
|      */
 | |
|     protected $_resources   = array();
 | |
| 
 | |
|     /**
 | |
|      * @var ArrayIterator of Resouces (Socket type) as keys, Ratchet\Socket as values
 | |
|      */
 | |
|     protected $_connections;
 | |
| 
 | |
|     /**
 | |
|      * @type Logging\LoggerInterface;
 | |
|      */
 | |
|     protected $_log;
 | |
| 
 | |
|     /**
 | |
|      * @var SocketObserver
 | |
|      * Maybe temporary?
 | |
|      */
 | |
|     protected $_app;
 | |
| 
 | |
|     /**
 | |
|      * @param Ratchet\Socket
 | |
|      * @param SocketObserver
 | |
|      * @param Logging\LoggerInterface
 | |
|      */
 | |
|     public function __construct(SocketInterface $host, SocketObserver $application, LoggerInterface $logger = null) {
 | |
|         $this->_master = $host;
 | |
|         $socket = $host->getResource();
 | |
|         $this->_resources[] = $socket;
 | |
| 
 | |
|         if (null === $logger) {
 | |
|             $logger = new NullLogger;
 | |
|         }
 | |
|         $this->_log = $logger;
 | |
| 
 | |
|         $this->_connections = new \ArrayIterator(array());
 | |
| 
 | |
|         $this->_app = $application;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return ArrayIterator of SocketInterfaces
 | |
|      */
 | |
|     public function getIterator() {
 | |
|         return $this->_connections;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * @param mixed The address to listen for incoming connections on.  "0.0.0.0" to listen from anywhere
 | |
|      * @param int The port to listen to connections on
 | |
|      * @throws Exception
 | |
|      * @todo Validate address.  Use socket_get_option, if AF_INET must be IP, if AF_UNIX must be path
 | |
|      * @todo Should I make handling open/close/msg an application?
 | |
|      * @todo Consider making the 4kb listener changable
 | |
|      */
 | |
|     public function run($address = '127.0.0.1', $port = 1025) {
 | |
|         set_time_limit(0);
 | |
|         ob_implicit_flush();
 | |
| 
 | |
|         if (false === ($this->_master->bind($address, (int)$port))) {
 | |
|             throw new Exception;
 | |
|         }
 | |
| 
 | |
|         if (false === ($this->_master->listen())) {
 | |
|             throw new Exception;
 | |
|         }
 | |
| 
 | |
|         $this->_master->set_nonblock();
 | |
| 
 | |
|         do {
 | |
|             try {
 | |
|                 $changed     = $this->_resources;
 | |
|                 $num_changed = $this->_master->select($changed, $write = null, $except = null, null);
 | |
| 
 | |
|     			foreach($changed as $resource) {
 | |
|                     if ($this->_master->getResource() === $resource) {
 | |
|                         $res = $this->onOpen($this->_master);
 | |
|                     } else {
 | |
|                         $conn  = $this->_connections[$resource];
 | |
|                         $data  = null;
 | |
|                         $bytes = $conn->recv($data, 4096, 0);
 | |
| 
 | |
|                         if ($bytes == 0) {
 | |
|                             $res = $this->onClose($conn);
 | |
|                         } else {
 | |
|                             $res = $this->onRecv($conn, $data);
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     while ($res instanceof CommandInterface) {
 | |
|                         $res = $res->execute($this);
 | |
|                     }
 | |
|                 }
 | |
|             } catch (Exception $se) {
 | |
|                 $this->_log->error($se->getMessage());
 | |
|             } catch (\Exception $e) {
 | |
|                 $this->_log->error('Big uh oh: ' . $e->getMessage());
 | |
|             }
 | |
|         } while (true);
 | |
| 
 | |
| //        $this->_master->set_nonblock();
 | |
| //        declare(ticks = 1); 
 | |
|     }
 | |
| 
 | |
|     public function onOpen(SocketInterface $conn) {
 | |
|         $new_connection     = clone $conn;
 | |
|         $this->_resources[] = $new_connection->getResource();
 | |
|         $this->_connections[$new_connection->getResource()] = $new_connection;
 | |
| 
 | |
|         $this->_log->note('New connection, ' . count($this->_connections) . ' total');
 | |
| 
 | |
|         return $this->_app->onOpen($new_connection);
 | |
|     }
 | |
| 
 | |
|     public function onRecv(SocketInterface $from, $msg) {
 | |
|         $this->_log->note('New message "' . trim($msg) . '"');
 | |
| 
 | |
|         return $this->_app->onRecv($from, $msg);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @todo Make sure it's OK to executre the command after resources have been free'd
 | |
|      */
 | |
|     public function onClose(SocketInterface $conn) {
 | |
|         $resource = $conn->getResource();
 | |
| 
 | |
|         $cmd = $this->_app->onClose($conn);
 | |
| 
 | |
|         unset($this->_connections[$resource]);
 | |
|         unset($this->_resources[array_search($resource, $this->_resources)]);
 | |
| 
 | |
|         $this->_log->note('Connection closed, ' . count($this->_connections) . ' connections remain');
 | |
| 
 | |
|         return $cmd;
 | |
|     }
 | |
| } |