123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 |
- <?php
- /*
- * This file is part of php-cache organization.
- *
- * (c) 2015 Aaron Scherer <aequasi@gmail.com>, Tobias Nyholm <tobias.nyholm@gmail.com>
- *
- * This source file is subject to the MIT license that is bundled
- * with this source code in the file LICENSE.
- */
- namespace Cache\Adapter\Common;
- use Cache\Adapter\Common\Exception\InvalidArgumentException;
- use Cache\TagInterop\TaggableCacheItemInterface;
- /**
- * @author Aaron Scherer <aequasi@gmail.com>
- * @author Tobias Nyholm <tobias.nyholm@gmail.com>
- */
- class CacheItem implements PhpCacheItem
- {
- /**
- * @type array
- */
- private $prevTags = [];
- /**
- * @type array
- */
- private $tags = [];
- /**
- * @type \Closure
- */
- private $callable;
- /**
- * @type string
- */
- private $key;
- /**
- * @type mixed
- */
- private $value;
- /**
- * The expiration timestamp is the source of truth. This is the UTC timestamp
- * when the cache item expire. A value of zero means it never expires. A nullvalue
- * means that no expiration is set.
- *
- * @type int|null
- */
- private $expirationTimestamp = null;
- /**
- * @type bool
- */
- private $hasValue = false;
- /**
- * @param string $key
- * @param \Closure|bool $callable or boolean hasValue
- */
- public function __construct($key, $callable = null, $value = null)
- {
- $this->key = $key;
- if ($callable === true) {
- $this->hasValue = true;
- $this->value = $value;
- } elseif ($callable !== false) {
- // This must be a callable or null
- $this->callable = $callable;
- }
- }
- /**
- * {@inheritdoc}
- */
- public function getKey()
- {
- return $this->key;
- }
- /**
- * {@inheritdoc}
- */
- public function set($value)
- {
- $this->value = $value;
- $this->hasValue = true;
- $this->callable = null;
- return $this;
- }
- /**
- * {@inheritdoc}
- */
- public function get()
- {
- if (!$this->isHit()) {
- return;
- }
- return $this->value;
- }
- /**
- * {@inheritdoc}
- */
- public function isHit()
- {
- $this->initialize();
- if (!$this->hasValue) {
- return false;
- }
- if ($this->expirationTimestamp !== null) {
- return $this->expirationTimestamp > time();
- }
- return true;
- }
- /**
- * {@inheritdoc}
- */
- public function getExpirationTimestamp()
- {
- return $this->expirationTimestamp;
- }
- /**
- * {@inheritdoc}
- */
- public function expiresAt($expiration)
- {
- if ($expiration instanceof \DateTimeInterface) {
- $this->expirationTimestamp = $expiration->getTimestamp();
- } elseif (is_int($expiration) || null === $expiration) {
- $this->expirationTimestamp = $expiration;
- } else {
- throw new InvalidArgumentException('Cache item ttl/expiresAt must be of type integer or \DateTimeInterface.');
- }
- return $this;
- }
- /**
- * {@inheritdoc}
- */
- public function expiresAfter($time)
- {
- if ($time === null) {
- $this->expirationTimestamp = null;
- } elseif ($time instanceof \DateInterval) {
- $date = new \DateTime();
- $date->add($time);
- $this->expirationTimestamp = $date->getTimestamp();
- } elseif (is_int($time)) {
- $this->expirationTimestamp = time() + $time;
- } else {
- throw new InvalidArgumentException('Cache item ttl/expiresAfter must be of type integer or \DateInterval.');
- }
- return $this;
- }
- /**
- * {@inheritdoc}
- */
- public function getPreviousTags()
- {
- $this->initialize();
- return $this->prevTags;
- }
- /**
- * {@inheritdoc}
- */
- public function getTags()
- {
- return $this->tags;
- }
- /**
- * {@inheritdoc}
- */
- public function setTags(array $tags)
- {
- $this->tags = [];
- $this->tag($tags);
- return $this;
- }
- /**
- * Adds a tag to a cache item.
- *
- * @param string|string[] $tags A tag or array of tags
- *
- * @throws InvalidArgumentException When $tag is not valid.
- *
- * @return TaggableCacheItemInterface
- */
- private function tag($tags)
- {
- $this->initialize();
- if (!is_array($tags)) {
- $tags = [$tags];
- }
- foreach ($tags as $tag) {
- if (!is_string($tag)) {
- throw new InvalidArgumentException(sprintf('Cache tag must be string, "%s" given', is_object($tag) ? get_class($tag) : gettype($tag)));
- }
- if (isset($this->tags[$tag])) {
- continue;
- }
- if (!isset($tag[0])) {
- throw new InvalidArgumentException('Cache tag length must be greater than zero');
- }
- if (isset($tag[strcspn($tag, '{}()/\@:')])) {
- throw new InvalidArgumentException(sprintf('Cache tag "%s" contains reserved characters {}()/\@:', $tag));
- }
- $this->tags[$tag] = $tag;
- }
- return $this;
- }
- /**
- * If callable is not null, execute it an populate this object with values.
- */
- private function initialize()
- {
- if ($this->callable !== null) {
- // $func will be $adapter->fetchObjectFromCache();
- $func = $this->callable;
- $result = $func();
- $this->hasValue = $result[0];
- $this->value = $result[1];
- $this->prevTags = isset($result[2]) ? $result[2] : [];
- $this->expirationTimestamp = null;
- if (isset($result[3]) && is_int($result[3])) {
- $this->expirationTimestamp = $result[3];
- }
- $this->callable = null;
- }
- }
- /**
- * @internal This function should never be used and considered private.
- *
- * Move tags from $tags to $prevTags
- */
- public function moveTagsToPrevious()
- {
- $this->prevTags = $this->tags;
- $this->tags = [];
- }
- }
|