Fix file deletions to always use a delayed retry on windows, fixes #3074
parent
f53994fcf2
commit
745dcbce33
|
@ -136,7 +136,7 @@ class Cache
|
||||||
{
|
{
|
||||||
$file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
$file = preg_replace('{[^'.$this->whitelist.']}i', '-', $file);
|
||||||
if ($this->enabled && file_exists($this->root . $file)) {
|
if ($this->enabled && file_exists($this->root . $file)) {
|
||||||
return unlink($this->root . $file);
|
return $this->filesystem->unlink($this->root . $file);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@ -150,7 +150,7 @@ class Cache
|
||||||
|
|
||||||
$finder = $this->getFinder()->date('until '.$expire->format('Y-m-d H:i:s'));
|
$finder = $this->getFinder()->date('until '.$expire->format('Y-m-d H:i:s'));
|
||||||
foreach ($finder as $file) {
|
foreach ($finder as $file) {
|
||||||
unlink($file->getPathname());
|
$this->filesystem->unlink($file->getPathname());
|
||||||
}
|
}
|
||||||
|
|
||||||
$totalSize = $this->filesystem->size($this->root);
|
$totalSize = $this->filesystem->size($this->root);
|
||||||
|
@ -159,7 +159,7 @@ class Cache
|
||||||
while ($totalSize > $maxSize && $iterator->valid()) {
|
while ($totalSize > $maxSize && $iterator->valid()) {
|
||||||
$filepath = $iterator->current()->getPathname();
|
$filepath = $iterator->current()->getPathname();
|
||||||
$totalSize -= $this->filesystem->size($filepath);
|
$totalSize -= $this->filesystem->size($filepath);
|
||||||
unlink($filepath);
|
$this->filesystem->unlink($filepath);
|
||||||
$iterator->next();
|
$iterator->next();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,7 +48,7 @@ abstract class ArchiveDownloader extends FileDownloader
|
||||||
throw $e;
|
throw $e;
|
||||||
}
|
}
|
||||||
|
|
||||||
unlink($fileName);
|
$this->filesystem->unlink($fileName);
|
||||||
|
|
||||||
$contentDir = $this->getFolderContent($temporaryDir);
|
$contentDir = $this->getFolderContent($temporaryDir);
|
||||||
|
|
||||||
|
|
|
@ -205,12 +205,9 @@ class FileDownloader implements DownloaderInterface
|
||||||
{
|
{
|
||||||
$this->io->write(" - Removing <info>" . $package->getName() . "</info> (<comment>" . VersionParser::formatVersion($package) . "</comment>)");
|
$this->io->write(" - Removing <info>" . $package->getName() . "</info> (<comment>" . VersionParser::formatVersion($package) . "</comment>)");
|
||||||
if (!$this->filesystem->removeDirectory($path)) {
|
if (!$this->filesystem->removeDirectory($path)) {
|
||||||
// retry after a bit on windows since it tends to be touchy with mass removals
|
|
||||||
if (!defined('PHP_WINDOWS_VERSION_BUILD') || (usleep(250000) && !$this->filesystem->removeDirectory($path))) {
|
|
||||||
throw new \RuntimeException('Could not completely delete '.$path.', aborting.');
|
throw new \RuntimeException('Could not completely delete '.$path.', aborting.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets file name for specific package
|
* Gets file name for specific package
|
||||||
|
|
|
@ -162,12 +162,9 @@ abstract class VcsDownloader implements DownloaderInterface, ChangeReportInterfa
|
||||||
$this->io->write(" - Removing <info>" . $package->getName() . "</info> (<comment>" . $package->getPrettyVersion() . "</comment>)");
|
$this->io->write(" - Removing <info>" . $package->getName() . "</info> (<comment>" . $package->getPrettyVersion() . "</comment>)");
|
||||||
$this->cleanChanges($package, $path, false);
|
$this->cleanChanges($package, $path, false);
|
||||||
if (!$this->filesystem->removeDirectory($path)) {
|
if (!$this->filesystem->removeDirectory($path)) {
|
||||||
// retry after a bit on windows since it tends to be touchy with mass removals
|
|
||||||
if (!defined('PHP_WINDOWS_VERSION_BUILD') || (usleep(250) && !$this->filesystem->removeDirectory($path))) {
|
|
||||||
throw new \RuntimeException('Could not completely delete '.$path.', aborting.');
|
throw new \RuntimeException('Could not completely delete '.$path.', aborting.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Download progress information is not available for all VCS downloaders.
|
* Download progress information is not available for all VCS downloaders.
|
||||||
|
|
|
@ -259,10 +259,10 @@ class LibraryInstaller implements InstallerInterface
|
||||||
foreach ($binaries as $bin) {
|
foreach ($binaries as $bin) {
|
||||||
$link = $this->binDir.'/'.basename($bin);
|
$link = $this->binDir.'/'.basename($bin);
|
||||||
if (is_link($link) || file_exists($link)) {
|
if (is_link($link) || file_exists($link)) {
|
||||||
unlink($link);
|
$this->filesystem->unlink($link);
|
||||||
}
|
}
|
||||||
if (file_exists($link.'.bat')) {
|
if (file_exists($link.'.bat')) {
|
||||||
unlink($link.'.bat');
|
$this->filesystem->unlink($link.'.bat');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,7 +77,7 @@ class PearInstaller extends LibraryInstaller
|
||||||
if ($this->io->isVerbose()) {
|
if ($this->io->isVerbose()) {
|
||||||
$this->io->write(' Cleaning up');
|
$this->io->write(' Cleaning up');
|
||||||
}
|
}
|
||||||
unlink($packageArchive);
|
$this->filesystem->unlink($packageArchive);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getBinaries(PackageInterface $package)
|
protected function getBinaries(PackageInterface $package)
|
||||||
|
|
|
@ -36,7 +36,7 @@ class Filesystem
|
||||||
}
|
}
|
||||||
|
|
||||||
if (file_exists($file)) {
|
if (file_exists($file)) {
|
||||||
return unlink($file);
|
return $this->unlink($file);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@ -62,7 +62,7 @@ class Filesystem
|
||||||
public function emptyDirectory($dir, $ensureDirectoryExists = true)
|
public function emptyDirectory($dir, $ensureDirectoryExists = true)
|
||||||
{
|
{
|
||||||
if (file_exists($dir) && is_link($dir)) {
|
if (file_exists($dir) && is_link($dir)) {
|
||||||
unlink($dir);
|
$this->unlink($dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($ensureDirectoryExists) {
|
if ($ensureDirectoryExists) {
|
||||||
|
@ -94,10 +94,10 @@ class Filesystem
|
||||||
public function removeDirectory($directory)
|
public function removeDirectory($directory)
|
||||||
{
|
{
|
||||||
if (file_exists($directory) && is_link($directory)) {
|
if (file_exists($directory) && is_link($directory)) {
|
||||||
return unlink($directory);
|
return $this->unlink($directory);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_dir($directory)) {
|
if (!file_exists($directory) || !is_dir($directory)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -117,11 +117,11 @@ class Filesystem
|
||||||
|
|
||||||
$result = $this->getProcess()->execute($cmd, $output) === 0;
|
$result = $this->getProcess()->execute($cmd, $output) === 0;
|
||||||
|
|
||||||
if ($result) {
|
|
||||||
// clear stat cache because external processes aren't tracked by the php stat cache
|
// clear stat cache because external processes aren't tracked by the php stat cache
|
||||||
clearstatcache();
|
clearstatcache();
|
||||||
|
|
||||||
return !is_dir($directory);
|
if ($result && !file_exists($directory)) {
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->removeDirectoryPhp($directory);
|
return $this->removeDirectoryPhp($directory);
|
||||||
|
@ -144,13 +144,13 @@ class Filesystem
|
||||||
|
|
||||||
foreach ($ri as $file) {
|
foreach ($ri as $file) {
|
||||||
if ($file->isDir()) {
|
if ($file->isDir()) {
|
||||||
rmdir($file->getPathname());
|
$this->rmdir($file->getPathname());
|
||||||
} else {
|
} else {
|
||||||
unlink($file->getPathname());
|
$this->unlink($file->getPathname());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return rmdir($directory);
|
return $this->rmdir($directory);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function ensureDirectoryExists($directory)
|
public function ensureDirectoryExists($directory)
|
||||||
|
@ -169,6 +169,54 @@ class Filesystem
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to unlink a file and in case of failure retries after 350ms on windows
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function unlink($path)
|
||||||
|
{
|
||||||
|
if (!@unlink($path)) {
|
||||||
|
// retry after a bit on windows since it tends to be touchy with mass removals
|
||||||
|
if (!defined('PHP_WINDOWS_VERSION_BUILD') || (usleep(350000) && !@unlink($path))) {
|
||||||
|
$error = error_get_last();
|
||||||
|
$message = 'Could not delete '.$path.': ' . @$error['message'];
|
||||||
|
if (defined('PHP_WINDOWS_VERSION_BUILD')) {
|
||||||
|
$message .= "\nThis can be due to an antivirus or the Windows Search Indexer locking the file while they are analyzed";
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new \RuntimeException($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to rmdir a file and in case of failure retries after 350ms on windows
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function rmdir($path)
|
||||||
|
{
|
||||||
|
if (!@rmdir($path)) {
|
||||||
|
// retry after a bit on windows since it tends to be touchy with mass removals
|
||||||
|
if (!defined('PHP_WINDOWS_VERSION_BUILD') || (usleep(350000) && !@rmdir($path))) {
|
||||||
|
$error = error_get_last();
|
||||||
|
$message = 'Could not delete '.$path.': ' . @$error['message'];
|
||||||
|
if (defined('PHP_WINDOWS_VERSION_BUILD')) {
|
||||||
|
$message .= "\nThis can be due to an antivirus or the Windows Search Indexer locking the file while they are analyzed";
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new \RuntimeException($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copy then delete is a non-atomic version of {@link rename}.
|
* Copy then delete is a non-atomic version of {@link rename}.
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in New Issue