You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							262 lines
						
					
					
						
							6.8 KiB
						
					
					
				
			
		
		
	
	
							262 lines
						
					
					
						
							6.8 KiB
						
					
					
				| <?php
 | |
| 
 | |
| /**
 | |
|  * @file
 | |
|  * TeamSpeak 3 PHP Framework
 | |
|  *
 | |
|  * $Id: ServerQuery.php 06/06/2016 22:27:13 scp@Svens-iMac $
 | |
|  *
 | |
|  * This program 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 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 General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program. If not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * @package   TeamSpeak3
 | |
|  * @version   1.1.24
 | |
|  * @author    Sven 'ScP' Paulsen
 | |
|  * @copyright Copyright (c) 2010 by Planet TeamSpeak. All rights reserved.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class TeamSpeak3_Adapter_ServerQuery
 | |
|  * @brief Provides low-level methods for ServerQuery communication with a TeamSpeak 3 Server.
 | |
|  */
 | |
| class TeamSpeak3_Adapter_ServerQuery extends TeamSpeak3_Adapter_Abstract
 | |
| {
 | |
|   /**
 | |
|    * Stores a singleton instance of the active TeamSpeak3_Node_Host object.
 | |
|    *
 | |
|    * @var TeamSpeak3_Node_Host
 | |
|    */
 | |
|   protected $host = null;
 | |
| 
 | |
|   /**
 | |
|    * Stores the timestamp of the last command.
 | |
|    *
 | |
|    * @var integer
 | |
|    */
 | |
|   protected $timer = null;
 | |
| 
 | |
|   /**
 | |
|    * Number of queries executed on the server.
 | |
|    *
 | |
|    * @var integer
 | |
|    */
 | |
|   protected $count = 0;
 | |
| 
 | |
|   /**
 | |
|    * Stores an array with unsupported commands.
 | |
|    *
 | |
|    * @var array
 | |
|    */
 | |
|   protected $block = array("help");
 | |
| 
 | |
|   /**
 | |
|    * Connects the TeamSpeak3_Transport_Abstract object and performs initial actions on the remote
 | |
|    * server.
 | |
|    *
 | |
|    * @throws TeamSpeak3_Adapter_Exception
 | |
|    * @return void
 | |
|    */
 | |
|   protected function syn()
 | |
|   {
 | |
|     $this->initTransport($this->options);
 | |
|     $this->transport->setAdapter($this);
 | |
| 
 | |
|     TeamSpeak3_Helper_Profiler::init(spl_object_hash($this));
 | |
| 
 | |
|     if(!$this->getTransport()->readLine()->startsWith(TeamSpeak3::READY))
 | |
|     {
 | |
|       throw new TeamSpeak3_Adapter_Exception("invalid reply from the server");
 | |
|     }
 | |
| 
 | |
|     TeamSpeak3_Helper_Signal::getInstance()->emit("serverqueryConnected", $this);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * The TeamSpeak3_Adapter_ServerQuery destructor.
 | |
|    *
 | |
|    * @return void
 | |
|    */
 | |
|   public function __destruct()
 | |
|   {
 | |
|     if($this->getTransport() instanceof TeamSpeak3_Transport_Abstract && $this->transport->isConnected())
 | |
|     {
 | |
|       try
 | |
|       {
 | |
|         $this->request("quit");
 | |
|       }
 | |
|       catch(Exception $e)
 | |
|       {
 | |
|         return;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Sends a prepared command to the server and returns the result.
 | |
|    *
 | |
|    * @param  string  $cmd
 | |
|    * @param  boolean $throw
 | |
|    * @throws TeamSpeak3_Adapter_Exception
 | |
|    * @return TeamSpeak3_Adapter_ServerQuery_Reply
 | |
|    */
 | |
|   public function request($cmd, $throw = TRUE)
 | |
|   {
 | |
|     $query = TeamSpeak3_Helper_String::factory($cmd)->section(TeamSpeak3::SEPARATOR_CELL);
 | |
| 
 | |
|     if(strstr($cmd, "\r") || strstr($cmd, "\n"))
 | |
|     {
 | |
|       throw new TeamSpeak3_Adapter_Exception("illegal characters in command '" . $query . "'");
 | |
|     }
 | |
|     elseif(in_array($query, $this->block))
 | |
|     {
 | |
|       throw new TeamSpeak3_Adapter_ServerQuery_Exception("command not found", 0x100);
 | |
|     }
 | |
| 
 | |
|     TeamSpeak3_Helper_Signal::getInstance()->emit("serverqueryCommandStarted", $cmd);
 | |
| 
 | |
|     $this->getProfiler()->start();
 | |
|     $this->getTransport()->sendLine($cmd);
 | |
|     $this->timer = time();
 | |
|     $this->count++;
 | |
| 
 | |
|     $rpl = array();
 | |
| 
 | |
|     do {
 | |
|       $str = $this->getTransport()->readLine();
 | |
|       $rpl[] = $str;
 | |
|     } while($str instanceof TeamSpeak3_Helper_String && $str->section(TeamSpeak3::SEPARATOR_CELL) != TeamSpeak3::ERROR);
 | |
| 
 | |
|     $this->getProfiler()->stop();
 | |
| 
 | |
|     $reply = new TeamSpeak3_Adapter_ServerQuery_Reply($rpl, $cmd, $this->getHost(), $throw);
 | |
| 
 | |
|     TeamSpeak3_Helper_Signal::getInstance()->emit("serverqueryCommandFinished", $cmd, $reply);
 | |
| 
 | |
|     return $reply;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Waits for the server to send a notification message and returns the result.
 | |
|    *
 | |
|    * @throws TeamSpeak3_Adapter_Exception
 | |
|    * @return TeamSpeak3_Adapter_ServerQuery_Event
 | |
|    */
 | |
|   public function wait()
 | |
|   {
 | |
|     if($this->getTransport()->getConfig("blocking"))
 | |
|     {
 | |
|       throw new TeamSpeak3_Adapter_Exception("only available in non-blocking mode");
 | |
|     }
 | |
| 
 | |
|     do {
 | |
|       $evt = $this->getTransport()->readLine();
 | |
|     } while($evt instanceof TeamSpeak3_Helper_String && !$evt->section(TeamSpeak3::SEPARATOR_CELL)->startsWith(TeamSpeak3::EVENT));
 | |
| 
 | |
|     return new TeamSpeak3_Adapter_ServerQuery_Event($evt, $this->getHost());
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Uses given parameters and returns a prepared ServerQuery command.
 | |
|    *
 | |
|    * @param  string $cmd
 | |
|    * @param  array  $params
 | |
|    * @return string
 | |
|    */
 | |
|   public function prepare($cmd, array $params = array())
 | |
|   {
 | |
|     $args = array();
 | |
|     $cells = array();
 | |
| 
 | |
|     foreach($params as $ident => $value)
 | |
|     {
 | |
|       $ident = is_numeric($ident) ? "" : strtolower($ident) . TeamSpeak3::SEPARATOR_PAIR;
 | |
| 
 | |
|       if(is_array($value))
 | |
|       {
 | |
|         $value = array_values($value);
 | |
| 
 | |
|         for($i = 0; $i < count($value); $i++)
 | |
|         {
 | |
|           if($value[$i] === null) continue;
 | |
|           elseif($value[$i] === FALSE) $value[$i] = 0x00;
 | |
|           elseif($value[$i] === TRUE) $value[$i] = 0x01;
 | |
|           elseif($value[$i] instanceof TeamSpeak3_Node_Abstract) $value[$i] = $value[$i]->getId();
 | |
| 
 | |
|           $cells[$i][] = $ident . TeamSpeak3_Helper_String::factory($value[$i])->escape()->toUtf8();
 | |
|         }
 | |
|       }
 | |
|       else
 | |
|       {
 | |
|         if($value === null) continue;
 | |
|         elseif($value === FALSE) $value = 0x00;
 | |
|         elseif($value === TRUE) $value = 0x01;
 | |
|         elseif($value instanceof TeamSpeak3_Node_Abstract) $value = $value->getId();
 | |
| 
 | |
|         $args[] = $ident . TeamSpeak3_Helper_String::factory($value)->escape()->toUtf8();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     foreach(array_keys($cells) as $ident) $cells[$ident] = implode(TeamSpeak3::SEPARATOR_CELL, $cells[$ident]);
 | |
| 
 | |
|     if(count($args)) $cmd .= " " . implode(TeamSpeak3::SEPARATOR_CELL, $args);
 | |
|     if(count($cells)) $cmd .= " " . implode(TeamSpeak3::SEPARATOR_LIST, $cells);
 | |
| 
 | |
|     return trim($cmd);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the timestamp of the last command.
 | |
|    *
 | |
|    * @return integer
 | |
|    */
 | |
|   public function getQueryLastTimestamp()
 | |
|   {
 | |
|     return $this->timer;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the number of queries executed on the server.
 | |
|    *
 | |
|    * @return integer
 | |
|    */
 | |
|   public function getQueryCount()
 | |
|   {
 | |
|     return $this->count;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the total runtime of all queries.
 | |
|    *
 | |
|    * @return mixed
 | |
|    */
 | |
|   public function getQueryRuntime()
 | |
|   {
 | |
|     return $this->getProfiler()->getRuntime();
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the TeamSpeak3_Node_Host object of the current connection.
 | |
|    *
 | |
|    * @return TeamSpeak3_Node_Host
 | |
|    */
 | |
|   public function getHost()
 | |
|   {
 | |
|     if($this->host === null)
 | |
|     {
 | |
|       $this->host = new TeamSpeak3_Node_Host($this);
 | |
|     }
 | |
| 
 | |
|     return $this->host;
 | |
|   }
 | |
| }
 |