Skip to content

Commit

Permalink
Fix(ZipTask): Preserve dir mtime PHP 7.4 compat
Browse files Browse the repository at this point in the history
  • Loading branch information
leagris committed May 31, 2024
1 parent 5e0151b commit f6875a9
Showing 1 changed file with 26 additions and 25 deletions.
51 changes: 26 additions & 25 deletions src/Phing/Task/Ext/Archive/ZipTask.php
Original file line number Diff line number Diff line change
Expand Up @@ -74,35 +74,16 @@ class ZipTask extends MatchingTask
* @var string $comment
*/
private $comment = '';
private string $mtimeDummy;

/**
* Propagates the directory's mTime and permissions in the Zip archive if supported.
*
* @param \ZipArchive $zip The Zip archive
* @param File $f The File handle to the directory
* @param string $pathInZip The path of the directory in the Zip archive
*
* @throws IOException
*/
private static function fixDirAttributes($zip, $f, $pathInZip)
{
$indexInZip = $zip->locateName('/' === mb_substr($pathInZip, -1) ? $pathInZip : $pathInZip . '/');
if (false !== $indexInZip) {
if (method_exists($zip, 'setMtimeIndex')) { // PHP >= 8.0.0, PECL zip >= 1.16.0
$zip->setMtimeIndex($indexInZip, $f->lastModified());
}
$filePerms = fileperms($f->getPath());
if (false !== $filePerms) { // filePerms supported
$zip->setExternalAttributesIndex($indexInZip, \ZipArchive::OPSYS_DEFAULT, $filePerms << 16);
}
}
}
/**
* Removes all external attributes from the Zip archive.
*
* @param \ZipArchive $zip The Zip archive
*
* @return void
*/
private static function clearExternalAttributes($zip)
private static function clearExternalAttributes(\ZipArchive $zip)
{
for ($i = 0, $count = $zip->count(); $i < $count; ++$i) {
$zip->setExternalAttributesIndex($i, \ZipArchive::OPSYS_DOS, 0);
Expand Down Expand Up @@ -267,6 +248,10 @@ public function main()

$this->log("Building zip: " . $this->zipFile->__toString(), Project::MSG_INFO);

if (false === $this->mtimeDummy = tempnam(sys_get_temp_dir(), 'mtimeDummy')) {
throw new Exception('Could not create temp file');
}

$zip = new ZipArchive();
$res = $zip->open($this->zipFile->getAbsolutePath(), ZipArchive::CREATE);

Expand All @@ -288,6 +273,7 @@ public function main()
}

$zip->close();
unlink($this->mtimeDummy);
} catch (IOException $ioe) {
$msg = "Problem creating ZIP: " . $ioe->getMessage();
throw new BuildException($msg, $ioe, $this->getLocation());
Expand Down Expand Up @@ -356,8 +342,7 @@ private function addFilesetsToArchive($zip)

if ($f->isDirectory()) {
if ($pathInZip != '.') {
$zip->addEmptyDir($pathInZip);
static::fixDirAttributes($zip, $f, $pathInZip);
$this->addDirToZip($zip, $f->getAbsolutePath(), $pathInZip . '/');
}
} else {
$zip->addFile($f->getAbsolutePath(), $pathInZip);
Expand All @@ -366,4 +351,20 @@ private function addFilesetsToArchive($zip)
}
}
}

/**
* @param \ZipArchive $zip
* @param string $dirPath
* @param string $entryName
*
* @return void
*/
private function addDirToZip(\ZipArchive $zip, string $dirPath, string $entryName)
{
touch($this->mtimeDummy, filemtime($dirPath)); // Save directory's mtime to dummmy
$zip->addFile($this->mtimeDummy, $entryName); // Add empty dummy as a directory
if (false !== $filePerms = fileperms($dirPath)) { // filePerms supported
$zip->setExternalAttributesName($entryName, \ZipArchive::OPSYS_UNIX, $filePerms << 16);
}
}
}

0 comments on commit f6875a9

Please sign in to comment.