|
@@ -13,6 +13,8 @@
|
|
|
namespace Composer;
|
|
|
|
|
|
use Composer\IO\IOInterface;
|
|
|
+use Composer\Util\Filesystem;
|
|
|
+use Symfony\Component\Finder\Finder;
|
|
|
|
|
|
/**
|
|
|
* Reads/writes to a filesystem cache
|
|
@@ -24,11 +26,21 @@ class Cache
|
|
|
private $io;
|
|
|
private $root;
|
|
|
private $enabled = true;
|
|
|
+ private $whitelist;
|
|
|
+ private $filesystem;
|
|
|
|
|
|
- public function __construct(IOInterface $io, $cacheDir)
|
|
|
+ /**
|
|
|
+ * @param IOInterface $io
|
|
|
+ * @param string $cacheDir location of the cache
|
|
|
+ * @param string $whitelist List of characters that are allowed in path names (used in a regex character class)
|
|
|
+ * @param Filesystem $filesystem optional filesystem instance
|
|
|
+ */
|
|
|
+ public function __construct(IOInterface $io, $cacheDir, $whitelist = 'a-z0-9.', Filesystem $filesystem = null)
|
|
|
{
|
|
|
$this->io = $io;
|
|
|
$this->root = rtrim($cacheDir, '/\\') . '/';
|
|
|
+ $this->whitelist = $whitelist;
|
|
|
+ $this->filesystem = $filesystem ?: new Filesystem();
|
|
|
|
|
|
if (!is_dir($this->root)) {
|
|
|
if (!@mkdir($this->root, 0777, true)) {
|
|
@@ -44,33 +56,79 @@ class Cache
|
|
|
|
|
|
public function read($file)
|
|
|
{
|
|
|
- $file = preg_replace('{[^a-z0-9.]}i', '-', $file);
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
if ($this->enabled && file_exists($this->root . $file)) {
|
|
|
return file_get_contents($this->root . $file);
|
|
|
}
|
|
|
+
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
public function write($file, $contents)
|
|
|
{
|
|
|
if ($this->enabled) {
|
|
|
- $file = preg_replace('{[^a-z0-9.]}i', '-', $file);
|
|
|
- file_put_contents($this->root . $file, $contents);
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
+
|
|
|
+ return file_put_contents($this->root . $file, $contents);
|
|
|
}
|
|
|
+
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ public function copyFrom($file, $source)
|
|
|
+ {
|
|
|
+ if ($this->enabled) {
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
+ $this->filesystem->ensureDirectoryExists(dirname($this->root . $file));
|
|
|
+
|
|
|
+ return copy($source, $this->root . $file);
|
|
|
+ }
|
|
|
+
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ public function copyTo($file, $target)
|
|
|
+ {
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
+ if ($this->enabled && file_exists($this->root . $file)) {
|
|
|
+ touch($this->root . $file);
|
|
|
+
|
|
|
+ return copy($this->root . $file, $target);
|
|
|
+ }
|
|
|
+
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ public function gc($ttl)
|
|
|
+ {
|
|
|
+ $expire = new \DateTime();
|
|
|
+ $expire->modify('-'.$ttl.' seconds');
|
|
|
+
|
|
|
+ $finder = Finder::create()->files()->in($this->root)->date('until '.$expire->format('Y-m-d H:i:s'));
|
|
|
+ foreach ($finder as $file) {
|
|
|
+ unlink($file->getRealPath());
|
|
|
+ }
|
|
|
+
|
|
|
+ return true;
|
|
|
}
|
|
|
|
|
|
public function sha1($file)
|
|
|
{
|
|
|
- $file = preg_replace('{[^a-z0-9.]}i', '-', $file);
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
if ($this->enabled && file_exists($this->root . $file)) {
|
|
|
return sha1_file($this->root . $file);
|
|
|
}
|
|
|
+
|
|
|
+ return false;
|
|
|
}
|
|
|
|
|
|
public function sha256($file)
|
|
|
{
|
|
|
- $file = preg_replace('{[^a-z0-9.]}i', '-', $file);
|
|
|
+ $file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
|
|
if ($this->enabled && file_exists($this->root . $file)) {
|
|
|
return hash_file('sha256', $this->root . $file);
|
|
|
}
|
|
|
+
|
|
|
+ return false;
|
|
|
}
|
|
|
}
|