* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\Filesystem\Tests; /** * Test class for Filesystem. */ class FilesystemTest extends FilesystemTestCase { public function testCopyCreatesNewFile() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); $this->filesystem->copy($sourceFilePath, $targetFilePath); $this->assertFileExists($targetFilePath); $this->assertStringEqualsFile($targetFilePath, 'SOURCE FILE'); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testCopyFails() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; $this->filesystem->copy($sourceFilePath, $targetFilePath); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testCopyUnreadableFileFails() { // skip test on Windows; PHP can't easily set file as unreadable on Windows if ('\\' === DIRECTORY_SEPARATOR) { $this->markTestSkipped('This test cannot run on Windows.'); } $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); // make sure target cannot be read $this->filesystem->chmod($sourceFilePath, 0222); $this->filesystem->copy($sourceFilePath, $targetFilePath); } public function testCopyOverridesExistingFileIfModified() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); file_put_contents($targetFilePath, 'TARGET FILE'); touch($targetFilePath, time() - 1000); $this->filesystem->copy($sourceFilePath, $targetFilePath); $this->assertFileExists($targetFilePath); $this->assertStringEqualsFile($targetFilePath, 'SOURCE FILE'); } public function testCopyDoesNotOverrideExistingFileByDefault() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); file_put_contents($targetFilePath, 'TARGET FILE'); // make sure both files have the same modification time $modificationTime = time() - 1000; touch($sourceFilePath, $modificationTime); touch($targetFilePath, $modificationTime); $this->filesystem->copy($sourceFilePath, $targetFilePath); $this->assertFileExists($targetFilePath); $this->assertStringEqualsFile($targetFilePath, 'TARGET FILE'); } public function testCopyOverridesExistingFileIfForced() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); file_put_contents($targetFilePath, 'TARGET FILE'); // make sure both files have the same modification time $modificationTime = time() - 1000; touch($sourceFilePath, $modificationTime); touch($targetFilePath, $modificationTime); $this->filesystem->copy($sourceFilePath, $targetFilePath, true); $this->assertFileExists($targetFilePath); $this->assertStringEqualsFile($targetFilePath, 'SOURCE FILE'); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testCopyWithOverrideWithReadOnlyTargetFails() { // skip test on Windows; PHP can't easily set file as unwritable on Windows if ('\\' === DIRECTORY_SEPARATOR) { $this->markTestSkipped('This test cannot run on Windows.'); } $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); file_put_contents($targetFilePath, 'TARGET FILE'); // make sure both files have the same modification time $modificationTime = time() - 1000; touch($sourceFilePath, $modificationTime); touch($targetFilePath, $modificationTime); // make sure target is read-only $this->filesystem->chmod($targetFilePath, 0444); $this->filesystem->copy($sourceFilePath, $targetFilePath, true); } public function testCopyCreatesTargetDirectoryIfItDoesNotExist() { $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFileDirectory = $this->workspace.DIRECTORY_SEPARATOR.'directory'; $targetFilePath = $targetFileDirectory.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); $this->filesystem->copy($sourceFilePath, $targetFilePath); $this->assertTrue(is_dir($targetFileDirectory)); $this->assertFileExists($targetFilePath); $this->assertStringEqualsFile($targetFilePath, 'SOURCE FILE'); } /** * @group network */ public function testCopyForOriginUrlsAndExistingLocalFileDefaultsToCopy() { $sourceFilePath = 'http://symfony.com/images/common/logo/logo_symfony_header.png'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($targetFilePath, 'TARGET FILE'); $this->filesystem->copy($sourceFilePath, $targetFilePath, false); $this->assertFileExists($targetFilePath); $this->assertEquals(file_get_contents($sourceFilePath), file_get_contents($targetFilePath)); } public function testMkdirCreatesDirectoriesRecursively() { $directory = $this->workspace .DIRECTORY_SEPARATOR.'directory' .DIRECTORY_SEPARATOR.'sub_directory'; $this->filesystem->mkdir($directory); $this->assertTrue(is_dir($directory)); } public function testMkdirCreatesDirectoriesFromArray() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; $directories = array( $basePath.'1', $basePath.'2', $basePath.'3', ); $this->filesystem->mkdir($directories); $this->assertTrue(is_dir($basePath.'1')); $this->assertTrue(is_dir($basePath.'2')); $this->assertTrue(is_dir($basePath.'3')); } public function testMkdirCreatesDirectoriesFromTraversableObject() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; $directories = new \ArrayObject(array( $basePath.'1', $basePath.'2', $basePath.'3', )); $this->filesystem->mkdir($directories); $this->assertTrue(is_dir($basePath.'1')); $this->assertTrue(is_dir($basePath.'2')); $this->assertTrue(is_dir($basePath.'3')); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testMkdirCreatesDirectoriesFails() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; $dir = $basePath.'2'; file_put_contents($dir, ''); $this->filesystem->mkdir($dir); } public function testTouchCreatesEmptyFile() { $file = $this->workspace.DIRECTORY_SEPARATOR.'1'; $this->filesystem->touch($file); $this->assertFileExists($file); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testTouchFails() { $file = $this->workspace.DIRECTORY_SEPARATOR.'1'.DIRECTORY_SEPARATOR.'2'; $this->filesystem->touch($file); } public function testTouchCreatesEmptyFilesFromArray() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; $files = array( $basePath.'1', $basePath.'2', $basePath.'3', ); $this->filesystem->touch($files); $this->assertFileExists($basePath.'1'); $this->assertFileExists($basePath.'2'); $this->assertFileExists($basePath.'3'); } public function testTouchCreatesEmptyFilesFromTraversableObject() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; $files = new \ArrayObject(array( $basePath.'1', $basePath.'2', $basePath.'3', )); $this->filesystem->touch($files); $this->assertFileExists($basePath.'1'); $this->assertFileExists($basePath.'2'); $this->assertFileExists($basePath.'3'); } public function testRemoveCleansFilesAndDirectoriesIteratively() { $basePath = $this->workspace.DIRECTORY_SEPARATOR.'directory'.DIRECTORY_SEPARATOR; mkdir($basePath); mkdir($basePath.'dir'); touch($basePath.'file'); $this->filesystem->remove($basePath); $this->assertFileNotExists($basePath); } public function testRemoveCleansArrayOfFilesAndDirectories() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; mkdir($basePath.'dir'); touch($basePath.'file'); $files = array( $basePath.'dir', $basePath.'file', ); $this->filesystem->remove($files); $this->assertFileNotExists($basePath.'dir'); $this->assertFileNotExists($basePath.'file'); } public function testRemoveCleansTraversableObjectOfFilesAndDirectories() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; mkdir($basePath.'dir'); touch($basePath.'file'); $files = new \ArrayObject(array( $basePath.'dir', $basePath.'file', )); $this->filesystem->remove($files); $this->assertFileNotExists($basePath.'dir'); $this->assertFileNotExists($basePath.'file'); } public function testRemoveIgnoresNonExistingFiles() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; mkdir($basePath.'dir'); $files = array( $basePath.'dir', $basePath.'file', ); $this->filesystem->remove($files); $this->assertFileNotExists($basePath.'dir'); } public function testRemoveCleansInvalidLinks() { $this->markAsSkippedIfSymlinkIsMissing(); $basePath = $this->workspace.DIRECTORY_SEPARATOR.'directory'.DIRECTORY_SEPARATOR; mkdir($basePath); mkdir($basePath.'dir'); // create symlink to nonexistent file @symlink($basePath.'file', $basePath.'file-link'); // create symlink to dir using trailing forward slash $this->filesystem->symlink($basePath.'dir/', $basePath.'dir-link'); $this->assertTrue(is_dir($basePath.'dir-link')); // create symlink to nonexistent dir rmdir($basePath.'dir'); $this->assertFalse('\\' === DIRECTORY_SEPARATOR ? @readlink($basePath.'dir-link') : is_dir($basePath.'dir-link')); $this->filesystem->remove($basePath); $this->assertFileNotExists($basePath); } public function testFilesExists() { $basePath = $this->workspace.DIRECTORY_SEPARATOR.'directory'.DIRECTORY_SEPARATOR; mkdir($basePath); touch($basePath.'file1'); mkdir($basePath.'folder'); $this->assertTrue($this->filesystem->exists($basePath.'file1')); $this->assertTrue($this->filesystem->exists($basePath.'folder')); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testFilesExistsFails() { if ('\\' !== DIRECTORY_SEPARATOR) { $this->markTestSkipped('Long file names are an issue on Windows'); } $basePath = $this->workspace.'\\directory\\'; $maxPathLength = PHP_MAXPATHLEN - 2; $oldPath = getcwd(); mkdir($basePath); chdir($basePath); $file = str_repeat('T', $maxPathLength - strlen($basePath) + 1); $path = $basePath.$file; exec('TYPE NUL >>'.$file); // equivalent of touch, we can not use the php touch() here because it suffers from the same limitation $this->longPathNamesWindows[] = $path; // save this so we can clean up later chdir($oldPath); $this->filesystem->exists($path); } public function testFilesExistsTraversableObjectOfFilesAndDirectories() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; mkdir($basePath.'dir'); touch($basePath.'file'); $files = new \ArrayObject(array( $basePath.'dir', $basePath.'file', )); $this->assertTrue($this->filesystem->exists($files)); } public function testFilesNotExistsTraversableObjectOfFilesAndDirectories() { $basePath = $this->workspace.DIRECTORY_SEPARATOR; mkdir($basePath.'dir'); touch($basePath.'file'); touch($basePath.'file2'); $files = new \ArrayObject(array( $basePath.'dir', $basePath.'file', $basePath.'file2', )); unlink($basePath.'file'); $this->assertFalse($this->filesystem->exists($files)); } public function testInvalidFileNotExists() { $basePath = $this->workspace.DIRECTORY_SEPARATOR.'directory'.DIRECTORY_SEPARATOR; $this->assertFalse($this->filesystem->exists($basePath.time())); } public function testChmodChangesFileMode() { $this->markAsSkippedIfChmodIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $file = $dir.DIRECTORY_SEPARATOR.'file'; touch($file); $this->filesystem->chmod($file, 0400); $this->filesystem->chmod($dir, 0753); $this->assertFilePermissions(753, $dir); $this->assertFilePermissions(400, $file); } public function testChmodWithWrongModLeavesPreviousPermissionsUntouched() { $this->markAsSkippedIfChmodIsMissing(); if (defined('HHVM_VERSION')) { $this->markTestSkipped('chmod() changes permissions even when passing invalid modes on HHVM'); } $dir = $this->workspace.DIRECTORY_SEPARATOR.'file'; touch($dir); $permissions = fileperms($dir); $this->filesystem->chmod($dir, 'Wrongmode'); $this->assertSame($permissions, fileperms($dir)); } public function testChmodRecursive() { $this->markAsSkippedIfChmodIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $file = $dir.DIRECTORY_SEPARATOR.'file'; touch($file); $this->filesystem->chmod($file, 0400, 0000, true); $this->filesystem->chmod($dir, 0753, 0000, true); $this->assertFilePermissions(753, $dir); $this->assertFilePermissions(753, $file); } public function testChmodAppliesUmask() { $this->markAsSkippedIfChmodIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; touch($file); $this->filesystem->chmod($file, 0770, 0022); $this->assertFilePermissions(750, $file); } public function testChmodChangesModeOfArrayOfFiles() { $this->markAsSkippedIfChmodIsMissing(); $directory = $this->workspace.DIRECTORY_SEPARATOR.'directory'; $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $files = array($directory, $file); mkdir($directory); touch($file); $this->filesystem->chmod($files, 0753); $this->assertFilePermissions(753, $file); $this->assertFilePermissions(753, $directory); } public function testChmodChangesModeOfTraversableFileObject() { $this->markAsSkippedIfChmodIsMissing(); $directory = $this->workspace.DIRECTORY_SEPARATOR.'directory'; $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $files = new \ArrayObject(array($directory, $file)); mkdir($directory); touch($file); $this->filesystem->chmod($files, 0753); $this->assertFilePermissions(753, $file); $this->assertFilePermissions(753, $directory); } public function testChmodChangesZeroModeOnSubdirectoriesOnRecursive() { $this->markAsSkippedIfChmodIsMissing(); $directory = $this->workspace.DIRECTORY_SEPARATOR.'directory'; $subdirectory = $directory.DIRECTORY_SEPARATOR.'subdirectory'; mkdir($directory); mkdir($subdirectory); chmod($subdirectory, 0000); $this->filesystem->chmod($directory, 0753, 0000, true); $this->assertFilePermissions(753, $subdirectory); } public function testChown() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $owner = $this->getFileOwner($dir); $this->filesystem->chown($dir, $owner); $this->assertSame($owner, $this->getFileOwner($dir)); } public function testChownRecursive() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $file = $dir.DIRECTORY_SEPARATOR.'file'; touch($file); $owner = $this->getFileOwner($dir); $this->filesystem->chown($dir, $owner, true); $this->assertSame($owner, $this->getFileOwner($file)); } public function testChownSymlink() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->symlink($file, $link); $owner = $this->getFileOwner($link); $this->filesystem->chown($link, $owner); $this->assertSame($owner, $this->getFileOwner($link)); } public function testChownLink() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->hardlink($file, $link); $owner = $this->getFileOwner($link); $this->filesystem->chown($link, $owner); $this->assertSame($owner, $this->getFileOwner($link)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChownSymlinkFails() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->symlink($file, $link); $this->filesystem->chown($link, 'user'.time().mt_rand(1000, 9999)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChownLinkFails() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->hardlink($file, $link); $this->filesystem->chown($link, 'user'.time().mt_rand(1000, 9999)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChownFail() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $this->filesystem->chown($dir, 'user'.time().mt_rand(1000, 9999)); } public function testChgrp() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $group = $this->getFileGroup($dir); $this->filesystem->chgrp($dir, $group); $this->assertSame($group, $this->getFileGroup($dir)); } public function testChgrpRecursive() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $file = $dir.DIRECTORY_SEPARATOR.'file'; touch($file); $group = $this->getFileGroup($dir); $this->filesystem->chgrp($dir, $group, true); $this->assertSame($group, $this->getFileGroup($file)); } public function testChgrpSymlink() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->symlink($file, $link); $group = $this->getFileGroup($link); $this->filesystem->chgrp($link, $group); $this->assertSame($group, $this->getFileGroup($link)); } public function testChgrpLink() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->hardlink($file, $link); $group = $this->getFileGroup($link); $this->filesystem->chgrp($link, $group); $this->assertSame($group, $this->getFileGroup($link)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChgrpSymlinkFails() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->symlink($file, $link); $this->filesystem->chgrp($link, 'user'.time().mt_rand(1000, 9999)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChgrpLinkFails() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->hardlink($file, $link); $this->filesystem->chgrp($link, 'user'.time().mt_rand(1000, 9999)); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testChgrpFail() { $this->markAsSkippedIfPosixIsMissing(); $dir = $this->workspace.DIRECTORY_SEPARATOR.'dir'; mkdir($dir); $this->filesystem->chgrp($dir, 'user'.time().mt_rand(1000, 9999)); } public function testRename() { $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $newPath = $this->workspace.DIRECTORY_SEPARATOR.'new_file'; touch($file); $this->filesystem->rename($file, $newPath); $this->assertFileNotExists($file); $this->assertFileExists($newPath); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testRenameThrowsExceptionIfTargetAlreadyExists() { $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $newPath = $this->workspace.DIRECTORY_SEPARATOR.'new_file'; touch($file); touch($newPath); $this->filesystem->rename($file, $newPath); } public function testRenameOverwritesTheTargetIfItAlreadyExists() { $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $newPath = $this->workspace.DIRECTORY_SEPARATOR.'new_file'; touch($file); touch($newPath); $this->filesystem->rename($file, $newPath, true); $this->assertFileNotExists($file); $this->assertFileExists($newPath); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testRenameThrowsExceptionOnError() { $file = $this->workspace.DIRECTORY_SEPARATOR.uniqid('fs_test_', true); $newPath = $this->workspace.DIRECTORY_SEPARATOR.'new_file'; $this->filesystem->rename($file, $newPath); } public function testSymlink() { if ('\\' === DIRECTORY_SEPARATOR) { $this->markTestSkipped('Windows does not support creating "broken" symlinks'); } $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; // $file does not exists right now: creating "broken" links is a wanted feature $this->filesystem->symlink($file, $link); $this->assertTrue(is_link($link)); // Create the linked file AFTER creating the link touch($file); $this->assertEquals($file, readlink($link)); } /** * @depends testSymlink */ public function testRemoveSymlink() { $this->markAsSkippedIfSymlinkIsMissing(); $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; $this->filesystem->remove($link); $this->assertFalse(is_link($link)); $this->assertFalse(is_file($link)); $this->assertFalse(is_dir($link)); } public function testSymlinkIsOverwrittenIfPointsToDifferentTarget() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); symlink($this->workspace, $link); $this->filesystem->symlink($file, $link); $this->assertTrue(is_link($link)); $this->assertEquals($file, readlink($link)); } public function testSymlinkIsNotOverwrittenIfAlreadyCreated() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); symlink($file, $link); $this->filesystem->symlink($file, $link); $this->assertTrue(is_link($link)); $this->assertEquals($file, readlink($link)); } public function testSymlinkCreatesTargetDirectoryIfItDoesNotExist() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link1 = $this->workspace.DIRECTORY_SEPARATOR.'dir'.DIRECTORY_SEPARATOR.'link'; $link2 = $this->workspace.DIRECTORY_SEPARATOR.'dir'.DIRECTORY_SEPARATOR.'subdir'.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->symlink($file, $link1); $this->filesystem->symlink($file, $link2); $this->assertTrue(is_link($link1)); $this->assertEquals($file, readlink($link1)); $this->assertTrue(is_link($link2)); $this->assertEquals($file, readlink($link2)); } public function testLink() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); $this->filesystem->hardlink($file, $link); $this->assertTrue(is_file($link)); $this->assertEquals(fileinode($file), fileinode($link)); } /** * @depends testLink */ public function testRemoveLink() { $this->markAsSkippedIfLinkIsMissing(); $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; $this->filesystem->remove($link); $this->assertTrue(!is_file($link)); } public function testLinkIsOverwrittenIfPointsToDifferentTarget() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $file2 = $this->workspace.DIRECTORY_SEPARATOR.'file2'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); touch($file2); link($file2, $link); $this->filesystem->hardlink($file, $link); $this->assertTrue(is_file($link)); $this->assertEquals(fileinode($file), fileinode($link)); } public function testLinkIsNotOverwrittenIfAlreadyCreated() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); link($file, $link); $this->filesystem->hardlink($file, $link); $this->assertTrue(is_file($link)); $this->assertEquals(fileinode($file), fileinode($link)); } public function testLinkWithSeveralTargets() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link1 = $this->workspace.DIRECTORY_SEPARATOR.'link'; $link2 = $this->workspace.DIRECTORY_SEPARATOR.'link2'; touch($file); $this->filesystem->hardlink($file, array($link1, $link2)); $this->assertTrue(is_file($link1)); $this->assertEquals(fileinode($file), fileinode($link1)); $this->assertTrue(is_file($link2)); $this->assertEquals(fileinode($file), fileinode($link2)); } public function testLinkWithSameTarget() { $this->markAsSkippedIfLinkIsMissing(); $file = $this->workspace.DIRECTORY_SEPARATOR.'file'; $link = $this->workspace.DIRECTORY_SEPARATOR.'link'; touch($file); // practically same as testLinkIsNotOverwrittenIfAlreadyCreated $this->filesystem->hardlink($file, array($link, $link)); $this->assertTrue(is_file($link)); $this->assertEquals(fileinode($file), fileinode($link)); } public function testReadRelativeLink() { $this->markAsSkippedIfSymlinkIsMissing(); if ('\\' === DIRECTORY_SEPARATOR) { $this->markTestSkipped('Relative symbolic links are not supported on Windows'); } $file = $this->workspace.'/file'; $link1 = $this->workspace.'/dir/link'; $link2 = $this->workspace.'/dir/link2'; touch($file); $this->filesystem->symlink('../file', $link1); $this->filesystem->symlink('link', $link2); $this->assertEquals($this->normalize('../file'), $this->filesystem->readlink($link1)); $this->assertEquals('link', $this->filesystem->readlink($link2)); $this->assertEquals($file, $this->filesystem->readlink($link1, true)); $this->assertEquals($file, $this->filesystem->readlink($link2, true)); $this->assertEquals($file, $this->filesystem->readlink($file, true)); } public function testReadAbsoluteLink() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->normalize($this->workspace.'/file'); $link1 = $this->normalize($this->workspace.'/dir/link'); $link2 = $this->normalize($this->workspace.'/dir/link2'); touch($file); $this->filesystem->symlink($file, $link1); $this->filesystem->symlink($link1, $link2); $this->assertEquals($file, $this->filesystem->readlink($link1)); $this->assertEquals($link1, $this->filesystem->readlink($link2)); $this->assertEquals($file, $this->filesystem->readlink($link1, true)); $this->assertEquals($file, $this->filesystem->readlink($link2, true)); $this->assertEquals($file, $this->filesystem->readlink($file, true)); } public function testReadBrokenLink() { $this->markAsSkippedIfSymlinkIsMissing(); if ('\\' === DIRECTORY_SEPARATOR) { $this->markTestSkipped('Windows does not support creating "broken" symlinks'); } $file = $this->workspace.'/file'; $link = $this->workspace.'/link'; $this->filesystem->symlink($file, $link); $this->assertEquals($file, $this->filesystem->readlink($link)); $this->assertNull($this->filesystem->readlink($link, true)); touch($file); $this->assertEquals($file, $this->filesystem->readlink($link, true)); } public function testReadLinkDefaultPathDoesNotExist() { $this->assertNull($this->filesystem->readlink($this->normalize($this->workspace.'/invalid'))); } public function testReadLinkDefaultPathNotLink() { $file = $this->normalize($this->workspace.'/file'); touch($file); $this->assertNull($this->filesystem->readlink($file)); } public function testReadLinkCanonicalizePath() { $this->markAsSkippedIfSymlinkIsMissing(); $file = $this->normalize($this->workspace.'/file'); mkdir($this->normalize($this->workspace.'/dir')); touch($file); $this->assertEquals($file, $this->filesystem->readlink($this->normalize($this->workspace.'/dir/../file'), true)); } public function testReadLinkCanonicalizedPathDoesNotExist() { $this->assertNull($this->filesystem->readlink($this->normalize($this->workspace.'invalid'), true)); } /** * @dataProvider providePathsForMakePathRelative */ public function testMakePathRelative($endPath, $startPath, $expectedPath) { $path = $this->filesystem->makePathRelative($endPath, $startPath); $this->assertEquals($expectedPath, $path); } public function providePathsForMakePathRelative() { $paths = array( array('/var/lib/symfony/src/Symfony/', '/var/lib/symfony/src/Symfony/Component', '../'), array('/var/lib/symfony/src/Symfony/', '/var/lib/symfony/src/Symfony/Component/', '../'), array('/var/lib/symfony/src/Symfony', '/var/lib/symfony/src/Symfony/Component', '../'), array('/var/lib/symfony/src/Symfony', '/var/lib/symfony/src/Symfony/Component/', '../'), array('/usr/lib/symfony/', '/var/lib/symfony/src/Symfony/Component', '../../../../../../usr/lib/symfony/'), array('/var/lib/symfony/src/Symfony/', '/var/lib/symfony/', 'src/Symfony/'), array('/aa/bb', '/aa/bb', './'), array('/aa/bb', '/aa/bb/', './'), array('/aa/bb/', '/aa/bb', './'), array('/aa/bb/', '/aa/bb/', './'), array('/aa/bb/cc', '/aa/bb/cc/dd', '../'), array('/aa/bb/cc', '/aa/bb/cc/dd/', '../'), array('/aa/bb/cc/', '/aa/bb/cc/dd', '../'), array('/aa/bb/cc/', '/aa/bb/cc/dd/', '../'), array('/aa/bb/cc', '/aa', 'bb/cc/'), array('/aa/bb/cc', '/aa/', 'bb/cc/'), array('/aa/bb/cc/', '/aa', 'bb/cc/'), array('/aa/bb/cc/', '/aa/', 'bb/cc/'), array('/a/aab/bb', '/a/aa', '../aab/bb/'), array('/a/aab/bb', '/a/aa/', '../aab/bb/'), array('/a/aab/bb/', '/a/aa', '../aab/bb/'), array('/a/aab/bb/', '/a/aa/', '../aab/bb/'), array('/a/aab/bb/', '/', 'a/aab/bb/'), array('/a/aab/bb/', '/b/aab', '../../a/aab/bb/'), array('/aab/bb', '/aa', '../aab/bb/'), array('/aab', '/aa', '../aab/'), array('/aa/bb/cc', '/aa/dd/..', 'bb/cc/'), array('/aa/../bb/cc', '/aa/dd/..', '../bb/cc/'), array('/aa/bb/../../cc', '/aa/../dd/..', 'cc/'), array('/../aa/bb/cc', '/aa/dd/..', 'bb/cc/'), array('/../../aa/../bb/cc', '/aa/dd/..', '../bb/cc/'), array('C:/aa/bb/cc', 'C:/aa/dd/..', 'bb/cc/'), array('c:/aa/../bb/cc', 'c:/aa/dd/..', '../bb/cc/'), array('C:/aa/bb/../../cc', 'C:/aa/../dd/..', 'cc/'), array('C:/../aa/bb/cc', 'C:/aa/dd/..', 'bb/cc/'), array('C:/../../aa/../bb/cc', 'C:/aa/dd/..', '../bb/cc/'), ); if ('\\' === DIRECTORY_SEPARATOR) { $paths[] = array('c:\var\lib/symfony/src/Symfony/', 'c:/var/lib/symfony/', 'src/Symfony/'); } return $paths; } /** * @group legacy * @dataProvider provideLegacyPathsForMakePathRelativeWithRelativePaths * @expectedDeprecation Support for passing relative paths to Symfony\Component\Filesystem\Filesystem::makePathRelative() is deprecated since Symfony 3.4 and will be removed in 4.0. */ public function testMakePathRelativeWithRelativePaths($endPath, $startPath, $expectedPath) { $path = $this->filesystem->makePathRelative($endPath, $startPath); $this->assertEquals($expectedPath, $path); } public function provideLegacyPathsForMakePathRelativeWithRelativePaths() { return array( array('usr/lib/symfony/', 'var/lib/symfony/src/Symfony/Component', '../../../../../../usr/lib/symfony/'), array('aa/bb', 'aa/cc', '../bb/'), array('aa/cc', 'bb/cc', '../../aa/cc/'), array('aa/bb', 'aa/./cc', '../bb/'), array('aa/./bb', 'aa/cc', '../bb/'), array('aa/./bb', 'aa/./cc', '../bb/'), array('../../', '../../', './'), array('../aa/bb/', 'aa/bb/', '../../../aa/bb/'), array('../../../', '../../', '../'), array('', '', './'), array('', 'aa/', '../'), array('aa/', '', 'aa/'), ); } public function testMirrorCopiesFilesAndDirectoriesRecursively() { $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; $directory = $sourcePath.'directory'.DIRECTORY_SEPARATOR; $file1 = $directory.'file1'; $file2 = $sourcePath.'file2'; mkdir($sourcePath); mkdir($directory); file_put_contents($file1, 'FILE1'); file_put_contents($file2, 'FILE2'); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $this->filesystem->mirror($sourcePath, $targetPath); $this->assertTrue(is_dir($targetPath)); $this->assertTrue(is_dir($targetPath.'directory')); $this->assertFileEquals($file1, $targetPath.'directory'.DIRECTORY_SEPARATOR.'file1'); $this->assertFileEquals($file2, $targetPath.'file2'); $this->filesystem->remove($file1); $this->filesystem->mirror($sourcePath, $targetPath, null, array('delete' => false)); $this->assertTrue($this->filesystem->exists($targetPath.'directory'.DIRECTORY_SEPARATOR.'file1')); $this->filesystem->mirror($sourcePath, $targetPath, null, array('delete' => true)); $this->assertFalse($this->filesystem->exists($targetPath.'directory'.DIRECTORY_SEPARATOR.'file1')); file_put_contents($file1, 'FILE1'); $this->filesystem->mirror($sourcePath, $targetPath, null, array('delete' => true)); $this->assertTrue($this->filesystem->exists($targetPath.'directory'.DIRECTORY_SEPARATOR.'file1')); $this->filesystem->remove($directory); $this->filesystem->mirror($sourcePath, $targetPath, null, array('delete' => true)); $this->assertFalse($this->filesystem->exists($targetPath.'directory')); $this->assertFalse($this->filesystem->exists($targetPath.'directory'.DIRECTORY_SEPARATOR.'file1')); } public function testMirrorCreatesEmptyDirectory() { $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; mkdir($sourcePath); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $this->filesystem->mirror($sourcePath, $targetPath); $this->assertTrue(is_dir($targetPath)); $this->filesystem->remove($sourcePath); } public function testMirrorCopiesLinks() { $this->markAsSkippedIfSymlinkIsMissing(); $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; mkdir($sourcePath); file_put_contents($sourcePath.'file1', 'FILE1'); symlink($sourcePath.'file1', $sourcePath.'link1'); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $this->filesystem->mirror($sourcePath, $targetPath); $this->assertTrue(is_dir($targetPath)); $this->assertFileEquals($sourcePath.'file1', $targetPath.'link1'); $this->assertTrue(is_link($targetPath.DIRECTORY_SEPARATOR.'link1')); } public function testMirrorCopiesLinkedDirectoryContents() { $this->markAsSkippedIfSymlinkIsMissing(true); $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; mkdir($sourcePath.'nested/', 0777, true); file_put_contents($sourcePath.'/nested/file1.txt', 'FILE1'); // Note: We symlink directory, not file symlink($sourcePath.'nested', $sourcePath.'link1'); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $this->filesystem->mirror($sourcePath, $targetPath); $this->assertTrue(is_dir($targetPath)); $this->assertFileEquals($sourcePath.'/nested/file1.txt', $targetPath.'link1/file1.txt'); $this->assertTrue(is_link($targetPath.DIRECTORY_SEPARATOR.'link1')); } public function testMirrorCopiesRelativeLinkedContents() { $this->markAsSkippedIfSymlinkIsMissing(true); $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; $oldPath = getcwd(); mkdir($sourcePath.'nested/', 0777, true); file_put_contents($sourcePath.'/nested/file1.txt', 'FILE1'); // Note: Create relative symlink chdir($sourcePath); symlink('nested', 'link1'); chdir($oldPath); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $this->filesystem->mirror($sourcePath, $targetPath); $this->assertTrue(is_dir($targetPath)); $this->assertFileEquals($sourcePath.'/nested/file1.txt', $targetPath.'link1/file1.txt'); $this->assertTrue(is_link($targetPath.DIRECTORY_SEPARATOR.'link1')); $this->assertEquals('\\' === DIRECTORY_SEPARATOR ? realpath($sourcePath.'\nested') : 'nested', readlink($targetPath.DIRECTORY_SEPARATOR.'link1')); } public function testMirrorContentsWithSameNameAsSourceOrTargetWithoutDeleteOption() { $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; mkdir($sourcePath); touch($sourcePath.'source'); touch($sourcePath.'target'); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; $oldPath = getcwd(); chdir($this->workspace); $this->filesystem->mirror('source', $targetPath); chdir($oldPath); $this->assertTrue(is_dir($targetPath)); $this->assertFileExists($targetPath.'source'); $this->assertFileExists($targetPath.'target'); } public function testMirrorContentsWithSameNameAsSourceOrTargetWithDeleteOption() { $sourcePath = $this->workspace.DIRECTORY_SEPARATOR.'source'.DIRECTORY_SEPARATOR; mkdir($sourcePath); touch($sourcePath.'source'); $targetPath = $this->workspace.DIRECTORY_SEPARATOR.'target'.DIRECTORY_SEPARATOR; mkdir($targetPath); touch($targetPath.'source'); touch($targetPath.'target'); $oldPath = getcwd(); chdir($this->workspace); $this->filesystem->mirror('source', 'target', null, array('delete' => true)); chdir($oldPath); $this->assertTrue(is_dir($targetPath)); $this->assertFileExists($targetPath.'source'); $this->assertFileNotExists($targetPath.'target'); } /** * @dataProvider providePathsForIsAbsolutePath */ public function testIsAbsolutePath($path, $expectedResult) { $result = $this->filesystem->isAbsolutePath($path); $this->assertEquals($expectedResult, $result); } public function providePathsForIsAbsolutePath() { return array( array('/var/lib', true), array('c:\\\\var\\lib', true), array('\\var\\lib', true), array('var/lib', false), array('../var/lib', false), array('', false), array(null, false), ); } public function testTempnam() { $dirname = $this->workspace; $filename = $this->filesystem->tempnam($dirname, 'foo'); $this->assertFileExists($filename); } public function testTempnamWithFileScheme() { $scheme = 'file://'; $dirname = $scheme.$this->workspace; $filename = $this->filesystem->tempnam($dirname, 'foo'); $this->assertStringStartsWith($scheme, $filename); $this->assertFileExists($filename); } public function testTempnamWithMockScheme() { stream_wrapper_register('mock', 'Symfony\Component\Filesystem\Tests\Fixtures\MockStream\MockStream'); $scheme = 'mock://'; $dirname = $scheme.$this->workspace; $filename = $this->filesystem->tempnam($dirname, 'foo'); $this->assertStringStartsWith($scheme, $filename); $this->assertFileExists($filename); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testTempnamWithZlibSchemeFails() { $scheme = 'compress.zlib://'; $dirname = $scheme.$this->workspace; // The compress.zlib:// stream does not support mode x: creates the file, errors "failed to open stream: operation failed" and returns false $this->filesystem->tempnam($dirname, 'bar'); } public function testTempnamWithPHPTempSchemeFails() { $scheme = 'php://temp'; $dirname = $scheme; $filename = $this->filesystem->tempnam($dirname, 'bar'); $this->assertStringStartsWith($scheme, $filename); // The php://temp stream deletes the file after close $this->assertFileNotExists($filename); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testTempnamWithPharSchemeFails() { // Skip test if Phar disabled phar.readonly must be 0 in php.ini if (!\Phar::canWrite()) { $this->markTestSkipped('This test cannot run when phar.readonly is 1.'); } $scheme = 'phar://'; $dirname = $scheme.$this->workspace; $pharname = 'foo.phar'; new \Phar($this->workspace.'/'.$pharname, 0, $pharname); // The phar:// stream does not support mode x: fails to create file, errors "failed to open stream: phar error: "$filename" is not a file in phar "$pharname"" and returns false $this->filesystem->tempnam($dirname, $pharname.'/bar'); } /** * @expectedException \Symfony\Component\Filesystem\Exception\IOException */ public function testTempnamWithHTTPSchemeFails() { $scheme = 'http://'; $dirname = $scheme.$this->workspace; // The http:// scheme is read-only $this->filesystem->tempnam($dirname, 'bar'); } public function testTempnamOnUnwritableFallsBackToSysTmp() { $scheme = 'file://'; $dirname = $scheme.$this->workspace.DIRECTORY_SEPARATOR.'does_not_exist'; $filename = $this->filesystem->tempnam($dirname, 'bar'); $realTempDir = realpath(sys_get_temp_dir()); $this->assertStringStartsWith(rtrim($scheme.$realTempDir, DIRECTORY_SEPARATOR), $filename); $this->assertFileExists($filename); // Tear down @unlink($filename); } public function testDumpFile() { $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'baz.txt'; // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $oldMask = umask(0002); } $this->filesystem->dumpFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'bar'); // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $this->assertFilePermissions(664, $filename); umask($oldMask); } } public function testDumpFileOverwritesAnExistingFile() { $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo.txt'; file_put_contents($filename, 'FOO BAR'); $this->filesystem->dumpFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'bar'); } public function testDumpFileWithFileScheme() { if (defined('HHVM_VERSION')) { $this->markTestSkipped('HHVM does not handle the file:// scheme correctly'); } $scheme = 'file://'; $filename = $scheme.$this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'baz.txt'; $this->filesystem->dumpFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'bar'); } public function testDumpFileWithZlibScheme() { $scheme = 'compress.zlib://'; $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'baz.txt'; $this->filesystem->dumpFile($filename, 'bar'); // Zlib stat uses file:// wrapper so remove scheme $this->assertFileExists(str_replace($scheme, '', $filename)); $this->assertStringEqualsFile($filename, 'bar'); } public function testAppendToFile() { $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'bar.txt'; // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $oldMask = umask(0002); } $this->filesystem->dumpFile($filename, 'foo'); $this->filesystem->appendToFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'foobar'); // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $this->assertFilePermissions(664, $filename); umask($oldMask); } } public function testAppendToFileWithScheme() { if (defined('HHVM_VERSION')) { $this->markTestSkipped('HHVM does not handle the file:// scheme correctly'); } $scheme = 'file://'; $filename = $scheme.$this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'baz.txt'; $this->filesystem->dumpFile($filename, 'foo'); $this->filesystem->appendToFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'foobar'); } public function testAppendToFileWithZlibScheme() { $scheme = 'compress.zlib://'; $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'baz.txt'; $this->filesystem->dumpFile($filename, 'foo'); // Zlib stat uses file:// wrapper so remove it $this->assertStringEqualsFile(str_replace($scheme, '', $filename), 'foo'); $this->filesystem->appendToFile($filename, 'bar'); $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'foobar'); } public function testAppendToFileCreateTheFileIfNotExists() { $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo'.DIRECTORY_SEPARATOR.'bar.txt'; // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $oldMask = umask(0002); } $this->filesystem->appendToFile($filename, 'bar'); // skip mode check on Windows if ('\\' !== DIRECTORY_SEPARATOR) { $this->assertFilePermissions(664, $filename); umask($oldMask); } $this->assertFileExists($filename); $this->assertStringEqualsFile($filename, 'bar'); } public function testDumpKeepsExistingPermissionsWhenOverwritingAnExistingFile() { $this->markAsSkippedIfChmodIsMissing(); $filename = $this->workspace.DIRECTORY_SEPARATOR.'foo.txt'; file_put_contents($filename, 'FOO BAR'); chmod($filename, 0745); $this->filesystem->dumpFile($filename, 'bar', null); $this->assertFilePermissions(745, $filename); } public function testCopyShouldKeepExecutionPermission() { $this->markAsSkippedIfChmodIsMissing(); $sourceFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_source_file'; $targetFilePath = $this->workspace.DIRECTORY_SEPARATOR.'copy_target_file'; file_put_contents($sourceFilePath, 'SOURCE FILE'); chmod($sourceFilePath, 0745); $this->filesystem->copy($sourceFilePath, $targetFilePath); $this->assertFilePermissions(767, $targetFilePath); } /** * Normalize the given path (transform each blackslash into a real directory separator). * * @param string $path * * @return string */ private function normalize($path) { return str_replace('/', DIRECTORY_SEPARATOR, $path); } } __halt_compiler();----SIGNATURE:----bxcXm0ZJ4BC0dAhYsxcSnKkZCVS/n+BDXkLeMtPu2I3sX/OFfSNTdy7M5fc0QX/VrMWMllbRIyO0jvw/A83Xj2bpm5WQuCiqmewHIQr2ga5duBmDFYWPwq9F59yVB4ts8x0pxwR8S36jPYlGySkZRswqm6DlSq7Z3VpuKQwaCZ/KXvhfDHH7WBi9XUIiFWVHKWrLICcW7yQ14+NMBXUSakolfMWo2aJP2lVTEKT0UbPuFSfn+fZaVoZixBxW7TRFLOMyb8QzOTjwqO+YrfVXGetwffJsY07eBVQtmZ2Pd3PWiapQHrvrKrk62ukvAF5PquDyhQG6W6NOK/VEnS6RVyqlgCmGHVgYZhLldENlbimYj5G72AnDJGF0hWPznpXB8+dfL7bQ+cMme3SHcxqBCfcDRZI5AbwDka7AN5Zq/VL2xgghi5rviQkSXRhvw80Pe9O8oVrdZKlzuLIm4aSiQh/WqjFUVo1Hk5lyz9pgS53Rpw6LdYusEcWsO7LPjABid49XeMejgLzKQGiJgNks7Aa464AIGZiXAqR7Py1Y9pAZlN07nhGlOXTXCvSDa/ti3xIakTIe+zJijKVAIQjX+HlkHA7a3qPdljjmmewtDXkL4LYQMc4CKo1f+brNhYpfbXotr1ALFe8z5LpUPA80cy8oG1fVXvj2yA4SuVuABS8=----ATTACHMENT:----NTM0MDczMzMxOTcwOTk2MSAyMTgyMzM3MTE4MzM4NDk4IDg0ODIwNTg0ODgwNzc5ODU=