123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- <?php
-
- namespace Illuminate\Cache;
-
- use Exception;
- use Illuminate\Contracts\Cache\Store;
- use Illuminate\Filesystem\Filesystem;
- use Illuminate\Support\InteractsWithTime;
-
- class FileStore implements Store
- {
- use InteractsWithTime, RetrievesMultipleKeys;
-
- /**
- * The Illuminate Filesystem instance.
- *
- * @var \Illuminate\Filesystem\Filesystem
- */
- protected $files;
-
- /**
- * The file cache directory.
- *
- * @var string
- */
- protected $directory;
-
- /**
- * Create a new file cache store instance.
- *
- * @param \Illuminate\Filesystem\Filesystem $files
- * @param string $directory
- * @return void
- */
- public function __construct(Filesystem $files, $directory)
- {
- $this->files = $files;
- $this->directory = $directory;
- }
-
- /**
- * Retrieve an item from the cache by key.
- *
- * @param string|array $key
- * @return mixed
- */
- public function get($key)
- {
- return $this->getPayload($key)['data'] ?? null;
- }
-
- /**
- * Store an item in the cache for a given number of minutes.
- *
- * @param string $key
- * @param mixed $value
- * @param float|int $minutes
- * @return void
- */
- public function put($key, $value, $minutes)
- {
- $this->ensureCacheDirectoryExists($path = $this->path($key));
-
- $this->files->put(
- $path, $this->expiration($minutes).serialize($value), true
- );
- }
-
- /**
- * Create the file cache directory if necessary.
- *
- * @param string $path
- * @return void
- */
- protected function ensureCacheDirectoryExists($path)
- {
- if (! $this->files->exists(dirname($path))) {
- $this->files->makeDirectory(dirname($path), 0777, true, true);
- }
- }
-
- /**
- * Increment the value of an item in the cache.
- *
- * @param string $key
- * @param mixed $value
- * @return int
- */
- public function increment($key, $value = 1)
- {
- $raw = $this->getPayload($key);
-
- return tap(((int) $raw['data']) + $value, function ($newValue) use ($key, $raw) {
- $this->put($key, $newValue, $raw['time'] ?? 0);
- });
- }
-
- /**
- * Decrement the value of an item in the cache.
- *
- * @param string $key
- * @param mixed $value
- * @return int
- */
- public function decrement($key, $value = 1)
- {
- return $this->increment($key, $value * -1);
- }
-
- /**
- * Store an item in the cache indefinitely.
- *
- * @param string $key
- * @param mixed $value
- * @return void
- */
- public function forever($key, $value)
- {
- $this->put($key, $value, 0);
- }
-
- /**
- * Remove an item from the cache.
- *
- * @param string $key
- * @return bool
- */
- public function forget($key)
- {
- if ($this->files->exists($file = $this->path($key))) {
- return $this->files->delete($file);
- }
-
- return false;
- }
-
- /**
- * Remove all items from the cache.
- *
- * @return bool
- */
- public function flush()
- {
- if (! $this->files->isDirectory($this->directory)) {
- return false;
- }
-
- foreach ($this->files->directories($this->directory) as $directory) {
- if (! $this->files->deleteDirectory($directory)) {
- return false;
- }
- }
-
- return true;
- }
-
- /**
- * Retrieve an item and expiry time from the cache by key.
- *
- * @param string $key
- * @return array
- */
- protected function getPayload($key)
- {
- $path = $this->path($key);
-
- // If the file doesn't exist, we obviously cannot return the cache so we will
- // just return null. Otherwise, we'll get the contents of the file and get
- // the expiration UNIX timestamps from the start of the file's contents.
- try {
- $expire = substr(
- $contents = $this->files->get($path, true), 0, 10
- );
- } catch (Exception $e) {
- return $this->emptyPayload();
- }
-
- // If the current time is greater than expiration timestamps we will delete
- // the file and return null. This helps clean up the old files and keeps
- // this directory much cleaner for us as old files aren't hanging out.
- if ($this->currentTime() >= $expire) {
- $this->forget($key);
-
- return $this->emptyPayload();
- }
-
- $data = unserialize(substr($contents, 10));
-
- // Next, we'll extract the number of minutes that are remaining for a cache
- // so that we can properly retain the time for things like the increment
- // operation that may be performed on this cache on a later operation.
- $time = ($expire - $this->currentTime()) / 60;
-
- return compact('data', 'time');
- }
-
- /**
- * Get a default empty payload for the cache.
- *
- * @return array
- */
- protected function emptyPayload()
- {
- return ['data' => null, 'time' => null];
- }
-
- /**
- * Get the full path for the given cache key.
- *
- * @param string $key
- * @return string
- */
- protected function path($key)
- {
- $parts = array_slice(str_split($hash = sha1($key), 2), 0, 2);
-
- return $this->directory.'/'.implode('/', $parts).'/'.$hash;
- }
-
- /**
- * Get the expiration time based on the given minutes.
- *
- * @param float|int $minutes
- * @return int
- */
- protected function expiration($minutes)
- {
- $time = $this->availableAt((int) ($minutes * 60));
-
- return $minutes === 0 || $time > 9999999999 ? 9999999999 : (int) $time;
- }
-
- /**
- * Get the Filesystem instance.
- *
- * @return \Illuminate\Filesystem\Filesystem
- */
- public function getFilesystem()
- {
- return $this->files;
- }
-
- /**
- * Get the working directory of the cache.
- *
- * @return string
- */
- public function getDirectory()
- {
- return $this->directory;
- }
-
- /**
- * Get the cache key prefix.
- *
- * @return string
- */
- public function getPrefix()
- {
- return '';
- }
- }
|