semantic_connector-8.x-1.1/src/SemanticConnectorCurlConnection.php

src/SemanticConnectorCurlConnection.php
<?php

namespace Drupal\semantic_connector;
use Drupal\Component\Serialization\Json;
use Drupal\Core\Logger\RfcLogLevel;

/**
 * Class SemanticConnectorCurlConnection
 *
 * API for calling cUrl requests.
 */
class SemanticConnectorCurlConnection {

  protected $endpoint;
  protected $credentials;
  protected $error;
  protected $logErrors;

  /**
   * The constructor of the PoolParty cURL connection class.
   *
   * @param string $endpoint
   *   URL of the endpoint of the PoolParty-server.
   * @param string $credentials
   *   Username and password if required (format: "username:password").
   * @param boolean $logErrors
   *   TRUE to log errors in the watchdog, FALSE to ignore any errors.
   */
  public function __construct($endpoint, $credentials = '', $logErrors = TRUE) {
    $this->endpoint = $endpoint;
    $this->credentials = $credentials;
    $this->error = '';
    $this->logErrors = $logErrors;
  }

  /**
   * Get the endpoint of the cURL connection.
   *
   * @return string
   *   The URL of the endpoint.
   */
  public function getEndpoint() {
    return $this->endpoint;
  }

  /**
   * Get the last request error.
   *
   * @return string
   *   Error (code - message).
   */
  public function error() {
    return $this->error;
  }

  /**
   * Activate or deactivate error logging.
   *
   * @param boolean $error_log_state
   *   TRUE to turn on error logging, FALSE to turn it off.
   */
  public function setErrorLogging($error_log_state) {
    $this->logErrors = $error_log_state;
  }

  /**
   * Make a GET request.
   *
   * @param string $resource_path
   *   The path to the REST method. You can include wildcards in the string
   *   which will be filled in by the $parameters array. Ex: /Courses/%session.
   * @param array $variables
   *   An array of variables with the following keys:
   *   - parameters [optional] : Key/value pairs of parameters to inject into
   *     the resource path to replace dynamic values.
   *     Ex: array('%session' => 20111).
   *   - query [optional] : Key/value pairs of query string parameters.
   *     Ex: array('personid' => 2896263).
   *   - headers [optional] : Key/value pairs of extra header data to include
   *     in the request.
   *   - timeout [optional] : Timeout in seconds. Defaults to 30.
   *
   * @return object
   *   Returns an object containing the response data, FALSE otherwise.
   */
  public function get($resource_path, array $variables = array()) {
    if (isset($variables['query'])) {
      if (!isset($variables['headers']['Content-Type'])) {
        $variables['headers']['Content-Type'] = 'application/json;charset=UTF-8';
      }
    }

    return $this->call($resource_path, $variables, 'GET');
  }

  /**
   * Make a POST request.
   *
   * @param string $resource_path
   *   The path to the REST method.
   * @param array $variables
   *   An array of variables.
   *
   * @return object
   *   Returns an object containing the response data, FALSE otherwise.
   */
  public function post($resource_path, array $variables = array()) {
    if (isset($variables['data'])) {
      if (is_string($variables['data'])) {
        $variables['headers']['Content-Length'] = strlen($variables['data']);
        $variables['headers']['Content-Type'] = 'application/json;charset=UTF-8';
      }
      elseif (!isset($variables['headers']['Content-Type'])) {
        // Make data ready for proxy server
        $variables['data'] = http_build_query($variables['data']);
        $variables['headers']['Content-Type'] = 'application/x-www-form-urlencoded';
      }
      if (is_array($variables['data']) && isset($variables['data']['file'])) {
        // The @ prefix for the file is deprecated as of PHP 5.6.
        // Convert @ prefixed file names to CURLFile class.
        if (class_exists('CURLFile')) {
          $file_name = ltrim($variables['data']['file'], '@');
          $variables['data']['file'] = new \CURLFile($file_name);
        }
        // Otherwise enable the support for the @ prefix.
        // Is not enabled for all PHP versions by default.
        elseif (defined('CURLOPT_SAFE_UPLOAD')) {
          $variables['curl_opt'][CURLOPT_SAFE_UPLOAD] = 1;
        }
      }
    }
    else {
      $variables['data'] = array();
      $variables['headers']['Content-Length'] = 0;
    }

    return $this->call($resource_path, $variables, 'POST');
  }

  /**
   * Make a PUT request.
   *
   * @param string $resource_path
   *   The path to the REST method.
   * @param array $variables
   *   An array of variables.
   *
   * @return object
   *   Returns an object containing the response data, FALSE otherwise.
   */
  public function put($resource_path, array $variables = array()) {
    return $this->call($resource_path, $variables, 'PUT');
  }

  /**
   * Make a DELETE request.
   *
   * @param string $resource_path
   *   The path to the REST method.
   * @param array $variables
   *   An array of variables.
   *
   * @return object
   *   Returns an object containing the response data, FALSE otherwise.
   */
  public function delete($resource_path, array $variables = array()) {
    return $this->call($resource_path, $variables, 'DELETE');
  }

  /**
   * Basic request (Compatible with GET and DELETE).
   *
   * @param string $resource_path
   *   The path to the REST method.
   * @param array $variables
   *   An array of variables.
   * @param string $method
   *   The request-method (GET, POST, PUT, DELETE).
   *
   * @return boolean|object
   *   The response object or FALSE on error
   */
  protected function call($resource_path, array $variables = array(), $method = 'GET') {
    // Check if cURL is enabled.
    if (!in_array('curl', get_loaded_extensions())) {
      $this->watchdog($method, 'The PHP library cURL is not enabled. It is impossible to connect to the PoolParty server.', -1001);
      return FALSE;
    }

    $variables['method'] = $method;

    // Initialize the cURL request.
    $ch = curl_init();

    // Set the default parameters for the cURL request.
    if (!isset($variables['headers']['Accept'])) {
      $variables['headers']['Accept'] = 'application/json';
    }
    $this->setRequestDefaults($ch, $variables);

    // Prepare the URL parameters.
    if (isset($variables['parameters']) && !empty($variables['parameters'])) {
      $this->prepareUrlParameters($resource_path, $variables['parameters']);
    }

    // Build the URL.
    $url = $this->buildUrl($resource_path, $variables);
    curl_setopt($ch, CURLOPT_URL, $url);
    curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);

    switch ($method) {
      case 'GET':
        curl_setopt($ch, CURLOPT_HTTPGET, TRUE);
        break;

      case 'POST':
        curl_setopt($ch, CURLOPT_POST, TRUE);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $variables['data']);
        break;

      case 'PUT':
        curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
        curl_setopt($ch, CURLOPT_HTTPHEADER, array('X-HTTP-Method-Override: PUT', 'Content-Type: application/json;charset=UTF-8'));
        curl_setopt($ch, CURLOPT_POSTFIELDS, $variables['data']);
        break;

      case 'DELETE':
        curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'DELETE');
        curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type: application/json;charset=UTF-8'));
        break;
    }

    // Make the request.
    $response_raw = curl_exec($ch);
    $http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);

    // There has been an error.
    if ($http_code != 200) {
      // Log the error.
      $error = curl_error($ch);
      $response = Json::decode($response_raw);
      // The error is not cURL-specific.
      if (empty($error)) {
        if ($response) {
          if (isset($response['message'])) {
            $error = $response['message'];
          }
          elseif (isset($response['errorMessage'])) {
            $error = $response['errorMessage'];
          }
          elseif (isset($response['responseBase']) && isset($response['responseBase']['message'])) {
            $error = $response['responseBase']['message'];
          }
        }
        // In case the response can't be parsed as JSON, return the raw HTML.
        else {
          $error = $response_raw;
        }
      }
      $this->watchdog($method, $error, $http_code, $url);

      // Close the cURL request.
      curl_close($ch);

      return FALSE;
    }

    // Close the cURL request.
    curl_close($ch);

    // No error occurred, return the response.
    return $response_raw;
  }

  /**
   * Set the default parameters for the cURL request.
   *
   * @param resource $ch
   *   The cURL request object.
   * @param array $variables
   *   Array of variables.
   */
  protected function setRequestDefaults($ch, array &$variables) {
    // Set the credentials.
    if (!empty($this->credentials)) {
      curl_setopt($ch, CURLOPT_USERPWD, $this->credentials);
    }

    // Set timeout.
    if (!(isset($variables['timeout']) && is_numeric($variables['timeout']) && intval($variables['timeout']) >= 0)) {
      $variables['timeout'] = 30;
    }
    curl_setopt($ch, CURLOPT_TIMEOUT, $variables['timeout']);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_SSLVERSION, 1);
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);

    // Set headers.
    if (isset($variables['headers']) && is_array($variables['headers'])) {
      $headers = array();
      foreach ($variables['headers'] as $key => $value) {
        $headers[] = trim($key) . ": " . trim($value);
      }
      curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
    }

    // Set other cURL options.
    if (isset($variables['curl_opt'])) {
      foreach ($variables['curl_opt'] as $option => $value) {
        curl_setopt($ch, $option, $value);
      }
    }
  }

  /**
   * Encodes and appends any URL parameters to the resource path.
   *
   * @param string $resource_path
   *   Resource path.
   * @param array $parameters
   *   Array of URL parameters.
   */
  protected function prepareUrlParameters(&$resource_path, array $parameters) {
    // URL Encode all parameters.
    foreach ($parameters as $key => $param) {
      $parameters[$key] = urlencode($param);
    }

    // Add the parameters to the resource path.
    $resource_path = strtr($resource_path, $parameters);
  }

  /**
   * Build the URL to connect to.
   *
   * @param string $resource_path
   *   Base path.
   * @param array $variables
   *   Configuration variables.
   *
   * @return string
   *   Returns a full URL.
   */
  protected function buildUrl($resource_path, array $variables) {
    $url = $this->endpoint;
    if (!empty($resource_path)) {
      $url .= $resource_path;
    }

    // Set the options to be used by url().
    if (isset($variables['query']) && !empty($variables['query'])) {
      $options = array(
        'query' => $variables['query'],
        'absolute' => TRUE,
        'alias' => TRUE,
        'external' => TRUE,
      );

      $url = \Drupal\Core\Url::fromUri($url, $options)->toString();
    }

    return $url;
  }

  /**
   * Log error messages into the watchdog.
   *
   * @param string $method
   *   Request method.
   * @param int $code
   *   Error code.
   * @param string $error
   *   Error string.
   * @param string $url
   *   Request URL.
   * @param array $extra
   *   Extra data to show in the log.
   */
  protected function watchdog($method, $error, $code = 0, $url = '', array $extra = array()) {
    if (!$this->logErrors) {
      return;
    }

    $debug = "";
    if (!empty($extra)) {
      $debug .= "\n" . '[<pre>' . print_r($extra, TRUE) . '</pre>]';
    }
    $this->error = $code . ' - ' . $error;

    SemanticConnectorWatchdog::message('cURL Connection', '@method request error @url (Code @code): @message @debug', array(
      '@method' => $method,
      '@url' => $url,
      '@code' => $code,
      '@message' => $error,
      '@debug' => $debug,
    ), RfcLogLevel::ERROR, TRUE);
  }
}

Главная | Обратная связь

drupal hosting | друпал хостинг | it patrol .inc