| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 | <?phpnamespace Drupal\bootstrap\Backport\Plugin\Provider;/** * CDN provider base class. * * Note: this class is a backport from the 8.x-3.x code base. * * @see https://drupal-bootstrap.org/api/bootstrap/namespace/Drupal%21bootstrap%21Plugin%21Provider/8 * * @ingroup plugins_provider */abstract class ProviderBase {  /**   * The plugin_id.   *   * @var string   */  protected $pluginId;  /**   * The currently set CDN assets.   *   * @var array   */  protected $cdnAssets;  /**   * The current theme name.   *   * @var string   */  protected $themeName;  /**   * The versions supplied by the CDN provider.   *   * @var array   */  protected $versions;  /**   * ProviderBase constructor.   */  public function __construct() {    $this->themeName = !empty($GLOBALS['theme_key']) ? $GLOBALS['theme_key'] : '';  }  /**   * {@inheritdoc}   */  public function alterFrameworkLibrary(array &$framework, $min = NULL) {    // In Drupal 7, CSS and JS are separated into individual hooks and alters,    // so this has the potential to be invoked at a minimum of 3 times.    static $drupal_static_fast;    if (!isset($drupal_static_fast)) {      // Attempt to retrieve CDN assets from a sort of permanent cached in the      // theme settings. This is primarily used to avoid unnecessary API requests      // and speed up the process during a cache rebuild. Theme settings are used      // as they persist through cache rebuilds. In order to prevent stale data,      // a hash is used based on current CDN settings and this "permacache" is      // reset at least once a week regardless.      // @see https://www.drupal.org/project/bootstrap/issues/3031415      $cdnCache = variable_get('bootstrap_cdn_cache') ?: array();      // Reset cache if expired.      if (isset($cdnCache['expire']) && (empty($cdnCache['expire']) || REQUEST_TIME > $cdnCache['expire'])) {        $cdnCache = array();      }      // Set expiration date (1 week by default).      if (!isset($cdnCache['expire'])) {        $cdnCache['expire'] = REQUEST_TIME + variable_get('bootstrap_cdn_cache_expire', 604800);      }      $cdnVersion = $this->getCdnVersion();      $cdnTheme = $this->getCdnTheme();      // Cache not found.      $cdnHash = drupal_hash_base64("{$this->pluginId}:$cdnTheme:$cdnVersion");      if (!isset($cdnCache[$cdnHash])) {        // Retrieve assets and reset cache (should only cache one at a time).        $cdnCache = array(          'expire' => $cdnCache['expire'],          $cdnHash => $this->getCdnAssets($cdnVersion, $cdnTheme),        );        variable_set('bootstrap_cdn_cache', $cdnCache);      }      // Immediately return if there are no theme CDN assets to use.      if (empty($cdnCache[$cdnHash])) {        return;      }      // Retrieve the system performance config.      if (!isset($min)) {        $min = array(          'css' => variable_get('preprocess_css', FALSE),          'js' => variable_get('preprocess_js', FALSE),        );      }      else {        $min = array('css' => !!$min, 'js' => !!$min);      }      // Iterate over each type.      $assets = array();      foreach (array('css', 'js') as $type) {        $files = !empty($min[$type]) && isset($cdnCache[$cdnHash]['min'][$type]) ? $cdnCache[$cdnHash]['min'][$type] : (isset($cdnCache[$cdnHash][$type]) ? $cdnCache[$cdnHash][$type] : array());        foreach ($files as $asset) {          $assets[$type][$asset] = array('data' => $asset, 'type' => 'external');        }      }      // Merge the assets into the library info.      $drupal_static_fast = drupal_array_merge_deep_array(array($assets, $framework));      // Override the framework version with the CDN version that is being used.      $drupal_static_fast['version'] = $cdnVersion;    }    $framework = $drupal_static_fast;  }  /**   * Retrieves a value from the CDN provider cache.   *   * @param string $key   *   The name of the item to retrieve. Note: this can be in the form of dot   *   notation if the value is nested in an array.   * @param mixed $default   *   Optional. The default value to return if $key is not set.   * @param callable $builder   *   Optional. If provided, a builder will be invoked when there is no cache   *   currently set.   *   * @return mixed   *   The cached value if it's set or the value supplied to $default if not.   */  protected function cacheGet($key, $default = NULL, $builder = NULL) {    $cid = $this->getCacheId();    $cache = cache_get($cid);    $data = $cache && isset($cache->data) && is_array($cache->data) ? $cache->data : array();    $parts = static::splitDelimiter($key);    $value = drupal_array_get_nested_value($data, $parts, $key_exists);    // Build the cache.    if (!$key_exists && is_callable($builder)) {      $value = $builder($default);      if (!isset($value)) {        $value = $default;      }      drupal_array_set_nested_value($data, $parts, $value);      cache_set($cid, $data);      return $value;    }    return $key_exists ? $value : $default;  }  /**   * Sets a value in the CDN provider cache.   *   * @param string $key   *   The name of the item to set. Note: this can be in the form of dot   *   notation if the value is nested in an array.   * @param mixed $value   *   Optional. The value to set.   */  protected function cacheSet($key, $value = NULL) {    $cid = $this->getCacheId();    $cache = cache_get($cid);    $data = $cache && isset($cache->data) && is_array($cache->data) ? $cache->data : array();    $parts = static::splitDelimiter($key);    drupal_array_set_nested_value($data, $parts, $value);    cache_set($cid, $data);  }  /**   * {@inheritdoc}   */  protected function discoverCdnAssets($version, $theme) {    return array();  }  /**   * Retrieves the unique cache identifier for the CDN provider.   *   * @return string   *   The CDN provider cache identifier.   */  protected function getCacheId() {    return "theme_registry:{$this->themeName}:provider:{$this->pluginId}";  }  /**   * {@inheritdoc}   */  public function getCdnAssets($version = NULL, $theme = NULL) {    if (!isset($version)) {      $version = $this->getCdnVersion();    }    if (!isset($theme)) {      $theme = $this->getCdnTheme();    }    if (!isset($this->cdnAssets)) {      $this->cdnAssets = $this->cacheGet('cdn.assets', array());    }    if (!isset($this->cdnAssets[$version][$theme])) {      $escapedVersion = static::escapeDelimiter($version);      $instance = $this;      $this->cdnAssets[$version][$theme] = $this->cacheGet("cdn.assets.$escapedVersion.$theme", array(), function () use ($version, $theme, $instance) {        return $instance->discoverCdnAssets($version, $theme);      });    }    return $this->cdnAssets[$version][$theme];  }  /**   * {@inheritdoc}   */  public function getCdnTheme() {    return bootstrap_setting("cdn_{$this->pluginId}_theme") ?: 'bootstrap';  }  /**   * {@inheritdoc}   */  public function getCdnThemes($version = NULL) {    return array();  }  /**   * {@inheritdoc}   */  public function getCdnVersion() {    return bootstrap_setting("cdn_{$this->pluginId}_version") ?: BOOTSTRAP_VERSION;  }  /**   * {@inheritdoc}   */  public function getCdnVersions() {    return array();  }  /**   * {@inheritdoc}   */  public function getDescription() {    return '';  }  /**   * {@inheritdoc}   */  public function getLabel() {    return t(ucfirst($this->pluginId));  }  /**   * Allows providers a way to map a version to a different version.   *   * @param string $version   *   The version to map.   *   * @return string   *   The mapped version.   */  protected function mapVersion($version) {    return $version;  }  /**   * Retrieves JSON from a URI.   *   * @param string $uri   *   The URI to retrieve JSON from.   * @param array $options   *   The options to pass to the HTTP client.   * @param \Exception|null $exception   *   The exception thrown if there was an error, passed by reference.   *   * @return array   *   The requested JSON array.   */  protected function requestJson($uri, array $options = array(), &$exception = NULL) {    $json = array();    $options += array(      'method' => 'GET',      'headers' => array(        'User-Agent' => 'Drupal Bootstrap 7.x-3.x (https://www.drupal.org/project/bootstrap)',      ),    );    try {      $response = drupal_http_request($uri, $options);      if (!empty($response->error)) {        throw new \Exception("$uri: {$response->error}", $response->code);      }      if ($response->code >= 200 && $response->code < 400) {        $json = drupal_json_decode($response->data) ?: array();      }      else {        throw new \Exception("$uri: Invalid response", $response->code);      }    }    catch (\Exception $e) {      $exception = $e;    }    return $json;  }  /**   * Escapes a delimiter in a string.   *   * Note: this is primarily useful in situations where dot notation is used   * where the values also contain dots, like in a semantic version string.   *   * @param string $string   *   The string to search in.   * @param string $delimiter   *   The delimiter to escape.   *   * @return string   *   The escaped string.   *   * @see \Drupal\bootstrap\Plugin\Provider\ProviderBase::splitDelimiter()   */  public static function escapeDelimiter($string, $delimiter = '.') {    return str_replace($delimiter, "\\$delimiter", $string);  }  /**   * Splits a string by a specified delimiter, allowing them to be escaped.   *   * Note: this is primarily useful in situations where dot notation is used   * where the values also contain dots, like in a semantic version string.   *   * @param string $string   *   The string to split into parts.   * @param string $delimiter   *   The delimiter used to split the string.   * @param bool $escapable   *   Flag indicating whether the $delimiter can be escaped using a backward   *   slash (\).   *   * @return array   *   An array of strings, split where the specified $delimiter was present.   *   * @see \Drupal\bootstrap\Plugin\Provider\ProviderBase::escapeDelimiter()   * @see https://stackoverflow.com/a/6243797   */  public static function splitDelimiter($string, $delimiter = '.', $escapable = TRUE) {    if (!$escapable) {      return explode($delimiter, $string);    }    // Split based on delimiter.    $parts = preg_split('~\\\\' . preg_quote($delimiter, '~') . '(*SKIP)(*FAIL)|\.~s', $string);    // Iterate over the parts and remove backslashes from delimiters.    return array_map(function ($string) use ($delimiter) {      return str_replace("\\$delimiter", $delimiter, $string);    }, $parts);  }}
 |