mirror of
				https://github.com/Icinga/icingaweb2.git
				synced 2025-10-25 01:14:26 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			298 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			298 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /* Icinga Web 2 | (c) 2016 Icinga Development Team | GPLv2+ */
 | |
| 
 | |
| namespace Icinga\Module\Monitoring\Web\Rest;
 | |
| 
 | |
| use Exception;
 | |
| use Icinga\Application\Logger;
 | |
| use Icinga\Util\Json;
 | |
| use Icinga\Module\Monitoring\Exception\CurlException;
 | |
| 
 | |
| /**
 | |
|  * REST Request
 | |
|  */
 | |
| class RestRequest
 | |
| {
 | |
|     /**
 | |
|      * Request URI
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $uri;
 | |
| 
 | |
|     /**
 | |
|      * Request method
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $method;
 | |
| 
 | |
|     /**
 | |
|      * Request content type
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $contentType;
 | |
| 
 | |
|     /**
 | |
|      * Whether to authenticate with basic auth
 | |
|      *
 | |
|      * @var bool
 | |
|      */
 | |
|     protected $hasBasicAuth;
 | |
| 
 | |
|     /**
 | |
|      * Auth username
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $username;
 | |
| 
 | |
|     /**
 | |
|      * Auth password
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $password;
 | |
| 
 | |
|     /**
 | |
|      * Request payload
 | |
|      *
 | |
|      * @var mixed
 | |
|      */
 | |
|     protected $payload;
 | |
| 
 | |
|     /**
 | |
|      * Whether strict SSL is enabled
 | |
|      *
 | |
|      * @var bool
 | |
|      */
 | |
|     protected $strictSsl = true;
 | |
| 
 | |
|     /**
 | |
|      * Request timeout
 | |
|      *
 | |
|      * @var int
 | |
|      */
 | |
|     protected $timeout = 30;
 | |
| 
 | |
|     /**
 | |
|      * Create a GET REST request
 | |
|      *
 | |
|      * @param   string  $uri
 | |
|      *
 | |
|      * @return  static
 | |
|      */
 | |
|     public static function get($uri)
 | |
|     {
 | |
|         $request = new static;
 | |
|         $request->uri = $uri;
 | |
|         $request->method = 'GET';
 | |
|         return $request;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a POST REST request
 | |
|      *
 | |
|      * @param   string  $uri
 | |
|      *
 | |
|      * @return  static
 | |
|      */
 | |
|     public static function post($uri)
 | |
|     {
 | |
|         $request = new static;
 | |
|         $request->uri = $uri;
 | |
|         $request->method = 'POST';
 | |
|         return $request;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Send content type JSON
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function sendJson()
 | |
|     {
 | |
|         $this->contentType = 'application/json';
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set basic auth credentials
 | |
|      *
 | |
|      * @param   string  $username
 | |
|      * @param   string  $password
 | |
|      *
 | |
|      * @return  $this
 | |
|      */
 | |
|     public function authenticateWith($username, $password)
 | |
|     {
 | |
|         $this->hasBasicAuth = true;
 | |
|         $this->username = $username;
 | |
|         $this->password = $password;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set request payload
 | |
|      *
 | |
|      * @param   mixed   $payload
 | |
|      *
 | |
|      * @return  $this
 | |
|      */
 | |
|     public function setPayload($payload)
 | |
|     {
 | |
|         $this->payload = $payload;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Disable strict SSL
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function noStrictSsl()
 | |
|     {
 | |
|         $this->strictSsl = false;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Serialize payload according to content type
 | |
|      *
 | |
|      * @param   mixed   $payload
 | |
|      * @param   string  $contentType
 | |
|      *
 | |
|      * @return  string
 | |
|      */
 | |
|     public function serializePayload($payload, $contentType)
 | |
|     {
 | |
|         switch ($contentType) {
 | |
|             case 'application/json':
 | |
|                 $payload = Json::encode($payload);
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         return $payload;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Send the request
 | |
|      *
 | |
|      * @return  mixed
 | |
|      *
 | |
|      * @throws  Exception
 | |
|      */
 | |
|     public function send()
 | |
|     {
 | |
|         $defaults = array(
 | |
|             'host'  => 'localhost',
 | |
|             'path'  => '/'
 | |
|         );
 | |
| 
 | |
|         $url = array_merge($defaults, parse_url($this->uri));
 | |
| 
 | |
|         if (isset($url['port'])) {
 | |
|             $url['host'] .= sprintf(':%u', $url['port']);
 | |
|         }
 | |
| 
 | |
|         if (isset($url['query'])) {
 | |
|             $url['path'] .= sprintf('?%s', $url['query']);
 | |
|         }
 | |
| 
 | |
|         $headers = array(
 | |
|             "{$this->method} {$url['path']} HTTP/1.1",
 | |
|             "Host: {$url['host']}",
 | |
|             "Content-Type: {$this->contentType}",
 | |
|             'Accept: application/json',
 | |
|             // Bypass "Expect: 100-continue" timeouts
 | |
|             'Expect:'
 | |
|         );
 | |
| 
 | |
|         $options = array(
 | |
|             CURLOPT_URL     => $this->uri,
 | |
|             CURLOPT_TIMEOUT => $this->timeout,
 | |
|             // Ignore proxy settings
 | |
|             CURLOPT_PROXY           => '',
 | |
|             CURLOPT_CUSTOMREQUEST   => $this->method
 | |
|         );
 | |
| 
 | |
|         // Record cURL command line for debugging
 | |
|         $curlCmd = array('curl', '-s', '-X', $this->method, '-H', escapeshellarg('Accept: application/json'));
 | |
| 
 | |
|         if ($this->strictSsl) {
 | |
|             $options[CURLOPT_SSL_VERIFYHOST] = 2;
 | |
|             $options[CURLOPT_SSL_VERIFYPEER] = true;
 | |
|         } else {
 | |
|             $options[CURLOPT_SSL_VERIFYHOST] = false;
 | |
|             $options[CURLOPT_SSL_VERIFYPEER] = false;
 | |
|             $curlCmd[] = '-k';
 | |
|         }
 | |
| 
 | |
|         if ($this->hasBasicAuth) {
 | |
|             $options[CURLOPT_USERPWD] = sprintf('%s:%s', $this->username, $this->password);
 | |
|             $curlCmd[] = sprintf('-u %s:%s', escapeshellarg($this->username), escapeshellarg($this->password));
 | |
|         }
 | |
| 
 | |
|         if (! empty($this->payload)) {
 | |
|             $payload = $this->serializePayload($this->payload, $this->contentType);
 | |
|             $options[CURLOPT_POSTFIELDS] = $payload;
 | |
|             $curlCmd[] = sprintf('-d %s', escapeshellarg($payload));
 | |
|         }
 | |
| 
 | |
|         $options[CURLOPT_HTTPHEADER] = $headers;
 | |
| 
 | |
|         $stream = null;
 | |
|         $logger = Logger::getInstance();
 | |
|         if ($logger !== null && $logger->getLevel() === Logger::DEBUG) {
 | |
|             $stream = fopen('php://temp', 'w');
 | |
|             $options[CURLOPT_VERBOSE] = true;
 | |
|             $options[CURLOPT_STDERR] = $stream;
 | |
|         }
 | |
| 
 | |
|         Logger::debug(
 | |
|             'Executing %s %s',
 | |
|             implode(' ', $curlCmd),
 | |
|             escapeshellarg($this->uri)
 | |
|         );
 | |
| 
 | |
|         $result = $this->curlExec($options);
 | |
| 
 | |
|         if (is_resource($stream)) {
 | |
|             rewind($stream);
 | |
|             Logger::debug(stream_get_contents($stream));
 | |
|             fclose($stream);
 | |
|         }
 | |
| 
 | |
|         return Json::decode($result, true);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set up a new cURL handle with the given options and call {@link curl_exec()}
 | |
|      *
 | |
|      * @param   array   $options
 | |
|      *
 | |
|      * @return  string  The response
 | |
|      *
 | |
|      * @throws  CurlException
 | |
|      */
 | |
|     protected function curlExec(array $options)
 | |
|     {
 | |
|         $ch = curl_init();
 | |
|         $options[CURLOPT_RETURNTRANSFER] = true;
 | |
|         curl_setopt_array($ch, $options);
 | |
|         $result = curl_exec($ch);
 | |
| 
 | |
|         if ($result === false) {
 | |
|             throw new CurlException('%s', curl_error($ch));
 | |
|         }
 | |
| 
 | |
|         curl_close($ch);
 | |
|         return $result;
 | |
|     }
 | |
| }
 |