2008-08-25 01:42:27 +00:00
|
|
|
<?php
|
|
|
|
|
2016-10-14 14:30:05 +13:00
|
|
|
namespace SilverStripe\Assets\Tests;
|
|
|
|
|
2016-11-23 18:09:10 +13:00
|
|
|
use SilverStripe\Assets\Image;
|
2016-08-19 10:51:35 +12:00
|
|
|
use SilverStripe\Assets\Storage\AssetStore;
|
2016-10-14 14:30:05 +13:00
|
|
|
use SilverStripe\Assets\Tests\FileTest\MyCustomFile;
|
|
|
|
use SilverStripe\ORM\ValidationException;
|
2016-06-15 16:03:16 +12:00
|
|
|
use SilverStripe\ORM\Versioning\Versioned;
|
|
|
|
use SilverStripe\ORM\DataObject;
|
2016-06-23 11:37:22 +12:00
|
|
|
use SilverStripe\Security\Member;
|
2016-08-05 10:32:13 +12:00
|
|
|
use SilverStripe\CMS\Model\ErrorPage;
|
2016-08-19 10:51:35 +12:00
|
|
|
use SilverStripe\Assets\Filesystem;
|
|
|
|
use SilverStripe\Assets\Folder;
|
|
|
|
use SilverStripe\Assets\File;
|
|
|
|
use SilverStripe\Core\Config\Config;
|
|
|
|
use SilverStripe\Core\Injector\Injector;
|
|
|
|
use SilverStripe\Dev\SapphireTest;
|
|
|
|
use SilverStripe\Control\Session;
|
|
|
|
use SilverStripe\Control\Director;
|
|
|
|
use SilverStripe\View\Parsers\ShortcodeParser;
|
2016-10-14 14:30:05 +13:00
|
|
|
use SilverStripe\Assets\Tests\Storage\AssetStoreTest\TestAssetStore;
|
2015-09-15 14:52:02 +12:00
|
|
|
|
2008-08-25 01:42:27 +00:00
|
|
|
/**
|
|
|
|
* Tests for the File class
|
|
|
|
*/
|
2016-12-16 17:34:21 +13:00
|
|
|
class FileTest extends SapphireTest
|
|
|
|
{
|
|
|
|
|
|
|
|
protected static $fixture_file = 'FileTest.yml';
|
|
|
|
|
|
|
|
protected $extraDataObjects = array(
|
|
|
|
MyCustomFile::class
|
|
|
|
);
|
|
|
|
|
|
|
|
public function setUp()
|
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
|
|
|
|
|
|
// Set backend root to /ImageTest
|
|
|
|
TestAssetStore::activate('FileTest');
|
|
|
|
|
|
|
|
// Create a test folders for each of the fixture references
|
|
|
|
$folderIDs = $this->allFixtureIDs(Folder::class);
|
|
|
|
foreach ($folderIDs as $folderID) {
|
|
|
|
$folder = DataObject::get_by_id(Folder::class, $folderID);
|
|
|
|
$filePath = ASSETS_PATH . '/FileTest/' . $folder->getFilename();
|
|
|
|
Filesystem::makeFolder($filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a test files for each of the fixture references
|
|
|
|
$fileIDs = $this->allFixtureIDs(File::class);
|
|
|
|
foreach ($fileIDs as $fileID) {
|
|
|
|
/**
|
2016-12-29 21:21:00 +13:00
|
|
|
* @var File $file
|
|
|
|
*/
|
2016-12-16 17:34:21 +13:00
|
|
|
$file = DataObject::get_by_id(File::class, $fileID);
|
|
|
|
$root = ASSETS_PATH . '/FileTest/';
|
|
|
|
if ($folder = $file->Parent()) {
|
|
|
|
$root .= $folder->getFilename();
|
|
|
|
}
|
|
|
|
$path = $root . substr($file->getHash(), 0, 10) . '/' . basename($file->getFilename());
|
|
|
|
Filesystem::makeFolder(dirname($path));
|
|
|
|
$fh = fopen($path, "w+");
|
|
|
|
fwrite($fh, str_repeat('x', 1000000));
|
|
|
|
fclose($fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Conditional fixture creation in case the 'cms' module is installed
|
|
|
|
if (class_exists('SilverStripe\\CMS\\Model\\ErrorPage')) {
|
|
|
|
$page = new ErrorPage(
|
|
|
|
array(
|
|
|
|
'Title' => 'Page not Found',
|
|
|
|
'ErrorCode' => 404
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$page->write();
|
|
|
|
$page->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function tearDown()
|
|
|
|
{
|
|
|
|
TestAssetStore::reset();
|
|
|
|
parent::tearDown();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLinkShortcodeHandler()
|
|
|
|
{
|
|
|
|
$testFile = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
|
|
|
|
$parser = new ShortcodeParser();
|
|
|
|
$parser->register('file_link', array(File::class, 'handle_shortcode'));
|
|
|
|
|
|
|
|
$fileShortcode = sprintf('[file_link,id=%d]', $testFile->ID);
|
|
|
|
$fileEnclosed = sprintf('[file_link,id=%d]Example Content[/file_link]', $testFile->ID);
|
|
|
|
|
|
|
|
$fileShortcodeExpected = $testFile->Link();
|
|
|
|
$fileEnclosedExpected = sprintf(
|
|
|
|
'<a href="%s" class="file" data-type="txt" data-size="977 KB">Example Content</a>',
|
|
|
|
$testFile->Link()
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertEquals($fileShortcodeExpected, $parser->parse($fileShortcode), 'Test that simple linking works.');
|
|
|
|
$this->assertEquals($fileEnclosedExpected, $parser->parse($fileEnclosed), 'Test enclosed content is linked.');
|
|
|
|
|
|
|
|
$testFile->delete();
|
|
|
|
|
|
|
|
$fileShortcode = '[file_link,id="-1"]';
|
|
|
|
$fileEnclosed = '[file_link,id="-1"]Example Content[/file_link]';
|
|
|
|
|
|
|
|
$this->assertEquals('', $parser->parse('[file_link]'), 'Test that invalid ID attributes are not parsed.');
|
|
|
|
$this->assertEquals('', $parser->parse('[file_link,id="text"]'));
|
|
|
|
$this->assertEquals('', $parser->parse('[file_link]Example Content[/file_link]'));
|
|
|
|
|
|
|
|
if (class_exists('SilverStripe\\CMS\\Model\\ErrorPage')) {
|
|
|
|
$errorPage = ErrorPage::get()->filter('ErrorCode', 404)->first();
|
|
|
|
$this->assertEquals(
|
|
|
|
$errorPage->Link(),
|
|
|
|
$parser->parse($fileShortcode),
|
|
|
|
'Test link to 404 page if no suitable matches.'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
sprintf('<a href="%s">Example Content</a>', $errorPage->Link()),
|
|
|
|
$parser->parse($fileEnclosed)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$this->assertEquals(
|
|
|
|
'',
|
|
|
|
$parser->parse($fileShortcode),
|
|
|
|
'Short code is removed if file record is not present.'
|
|
|
|
);
|
|
|
|
$this->assertEquals('', $parser->parse($fileEnclosed));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testCreateWithFilenameWithSubfolder()
|
|
|
|
{
|
|
|
|
// Note: We can't use fixtures/setUp() for this, as we want to create the db record manually.
|
|
|
|
// Creating the folder is necessary to avoid having "Filename" overwritten by setName()/setRelativePath(),
|
|
|
|
// because the parent folders don't exist in the database
|
|
|
|
$folder = Folder::find_or_make('/FileTest/');
|
|
|
|
$testfilePath = BASE_PATH . '/assets/FileTest/CreateWithFilenameHasCorrectPath.txt'; // Important: No leading slash
|
2016-12-29 21:21:00 +13:00
|
|
|
$fh = fopen($testfilePath, 'w');
|
2016-12-16 17:34:21 +13:00
|
|
|
fwrite($fh, str_repeat('x', 1000000));
|
|
|
|
fclose($fh);
|
|
|
|
|
|
|
|
$file = new File();
|
|
|
|
$file->setFromLocalFile($testfilePath);
|
|
|
|
$file->ParentID = $folder->ID;
|
|
|
|
$file->write();
|
|
|
|
|
|
|
|
$this->assertEquals(
|
|
|
|
'CreateWithFilenameHasCorrectPath.txt',
|
|
|
|
$file->Name,
|
|
|
|
'"Name" property is automatically set from "Filename"'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
'FileTest/CreateWithFilenameHasCorrectPath.txt',
|
|
|
|
$file->Filename,
|
|
|
|
'"Filename" property remains unchanged'
|
|
|
|
);
|
|
|
|
|
|
|
|
// TODO This should be auto-detected, see File->updateFilesystem()
|
|
|
|
// $this->assertInstanceOf('Folder', $file->Parent(), 'Parent folder is created in database');
|
|
|
|
// $this->assertFileExists($file->Parent()->getURL(), 'Parent folder is created on filesystem');
|
|
|
|
// $this->assertEquals('FileTest', $file->Parent()->Name);
|
|
|
|
// $this->assertInstanceOf('Folder', $file->Parent()->Parent(), 'Grandparent folder is created in database');
|
|
|
|
// $this->assertFileExists($file->Parent()->Parent()->getURL(),
|
|
|
|
// 'Grandparent folder is created on filesystem');
|
|
|
|
// $this->assertEquals('assets', $file->Parent()->Parent()->Name);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetExtension()
|
|
|
|
{
|
|
|
|
$this->assertEquals(
|
|
|
|
'',
|
|
|
|
File::get_file_extension('myfile'),
|
|
|
|
'No extension'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
'txt',
|
|
|
|
File::get_file_extension('myfile.txt'),
|
|
|
|
'Simple extension'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
'gz',
|
|
|
|
File::get_file_extension('myfile.tar.gz'),
|
|
|
|
'Double-barrelled extension only returns last bit'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testValidateExtension()
|
|
|
|
{
|
|
|
|
Session::set('loggedInAs', null);
|
|
|
|
|
|
|
|
$orig = Config::inst()->get(File::class, 'allowed_extensions');
|
|
|
|
Config::inst()->remove(File::class, 'allowed_extensions');
|
|
|
|
Config::inst()->update(File::class, 'allowed_extensions', array('txt'));
|
|
|
|
|
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
|
|
|
|
// Invalid ext
|
|
|
|
$file->Name = 'asdf.php';
|
|
|
|
$result = $file->validate();
|
|
|
|
$this->assertFalse($result->isValid());
|
2016-11-23 18:09:10 +13:00
|
|
|
$messages = $result->getMessages();
|
|
|
|
$this->assertEquals(1, count($messages));
|
2016-12-16 17:34:21 +13:00
|
|
|
$this->assertEquals('Extension is not allowed', $messages[0]['message']);
|
|
|
|
|
|
|
|
// Valid ext
|
|
|
|
$file->Name = 'asdf.txt';
|
|
|
|
$result = $file->validate();
|
|
|
|
$this->assertTrue($result->isValid());
|
|
|
|
|
|
|
|
// Capital extension is valid as well
|
|
|
|
$file->Name = 'asdf.TXT';
|
|
|
|
$result = $file->validate();
|
|
|
|
$this->assertTrue($result->isValid());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAppCategory()
|
|
|
|
{
|
|
|
|
// Test various categories
|
|
|
|
$this->assertEquals('image', File::get_app_category('jpg'));
|
|
|
|
$this->assertEquals('image', File::get_app_category('JPG'));
|
|
|
|
$this->assertEquals('image', File::get_app_category('JPEG'));
|
|
|
|
$this->assertEquals('image', File::get_app_category('png'));
|
|
|
|
$this->assertEquals('image', File::get_app_category('tif'));
|
|
|
|
$this->assertEquals('document', File::get_app_category('pdf'));
|
|
|
|
$this->assertEquals('video', File::get_app_category('mov'));
|
|
|
|
$this->assertEquals('audio', File::get_app_category('OGG'));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetCategoryExtensions()
|
|
|
|
{
|
|
|
|
// Test specific categories
|
|
|
|
$images = array(
|
|
|
|
'alpha', 'als', 'bmp', 'cel', 'gif', 'ico', 'icon', 'jpeg', 'jpg', 'pcx', 'png', 'ps', 'tif', 'tiff'
|
|
|
|
);
|
|
|
|
$this->assertEquals($images, File::get_category_extensions('image'));
|
|
|
|
$this->assertEquals(array('gif', 'jpeg', 'jpg', 'png'), File::get_category_extensions('image/supported'));
|
|
|
|
$this->assertEquals($images, File::get_category_extensions(array('image', 'image/supported')));
|
|
|
|
$this->assertEquals(
|
|
|
|
array('fla', 'gif', 'jpeg', 'jpg', 'png', 'swf'),
|
|
|
|
File::get_category_extensions(array('flash', 'image/supported'))
|
|
|
|
);
|
|
|
|
|
|
|
|
// Test other categories have at least one item
|
|
|
|
$this->assertNotEmpty(File::get_category_extensions('archive'));
|
|
|
|
$this->assertNotEmpty(File::get_category_extensions('audio'));
|
|
|
|
$this->assertNotEmpty(File::get_category_extensions('document'));
|
|
|
|
$this->assertNotEmpty(File::get_category_extensions('flash'));
|
|
|
|
$this->assertNotEmpty(File::get_category_extensions('video'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider allowedExtensions
|
|
|
|
* @param string $extension
|
|
|
|
*/
|
|
|
|
public function testAllFilesHaveCategory($extension)
|
|
|
|
{
|
|
|
|
$this->assertNotEmpty(
|
|
|
|
File::get_app_category($extension),
|
|
|
|
"Assert that extension {$extension} has a valid category"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the list of all extensions for testing
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function allowedExtensions()
|
|
|
|
{
|
|
|
|
$args = array();
|
|
|
|
foreach (array_filter(File::config()->allowed_extensions) as $ext) {
|
|
|
|
$args[] = array($ext);
|
|
|
|
}
|
|
|
|
return $args;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetNameChangesFilesystemOnWrite()
|
|
|
|
{
|
|
|
|
/**
|
2016-12-29 21:21:00 +13:00
|
|
|
* @var File $file
|
|
|
|
*/
|
2016-12-16 17:34:21 +13:00
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$file->publishRecursive();
|
|
|
|
$oldTuple = $file->File->getValue();
|
|
|
|
|
|
|
|
// Rename
|
|
|
|
$file->Name = 'renamed.txt';
|
|
|
|
$newTuple = $oldTuple;
|
|
|
|
$newTuple['Filename'] = $file->generateFilename();
|
|
|
|
|
|
|
|
// Before write()
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old path is still present'
|
|
|
|
);
|
|
|
|
$this->assertFalse(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is updated in memory, not written before write() is called'
|
|
|
|
);
|
|
|
|
|
|
|
|
// After write()
|
|
|
|
$file->write();
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old path exists after draft change'
|
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is created after write()'
|
|
|
|
);
|
|
|
|
|
|
|
|
// After publish
|
|
|
|
$file->publishRecursive();
|
|
|
|
$this->assertFalse(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old file is finally removed after publishing new file'
|
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is created after write()'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetParentIDChangesFilesystemOnWrite()
|
|
|
|
{
|
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$file->publishRecursive();
|
|
|
|
$subfolder = $this->objFromFixture(Folder::class, 'subfolder');
|
|
|
|
$oldTuple = $file->File->getValue();
|
|
|
|
|
|
|
|
// set ParentID
|
|
|
|
$file->ParentID = $subfolder->ID;
|
|
|
|
$newTuple = $oldTuple;
|
|
|
|
$newTuple['Filename'] = $file->generateFilename();
|
|
|
|
|
|
|
|
// Before write()
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old path is still present'
|
|
|
|
);
|
|
|
|
$this->assertFalse(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is updated in memory, not written before write() is called'
|
|
|
|
);
|
|
|
|
$file->write();
|
|
|
|
|
|
|
|
// After write()
|
|
|
|
$file->write();
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old path exists after draft change'
|
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is created after write()'
|
|
|
|
);
|
|
|
|
|
|
|
|
// After publish
|
|
|
|
$file->publishSingle();
|
|
|
|
$this->assertFalse(
|
|
|
|
$this->getAssetStore()->exists($oldTuple['Filename'], $oldTuple['Hash']),
|
|
|
|
'Old file is finally removed after publishing new file'
|
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($newTuple['Filename'], $newTuple['Hash']),
|
|
|
|
'New path is created after write()'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @see http://open.silverstripe.org/ticket/5693
|
|
|
|
*/
|
|
|
|
public function testSetNameWithInvalidExtensionDoesntChangeFilesystem()
|
|
|
|
{
|
|
|
|
Config::inst()->remove(File::class, 'allowed_extensions');
|
|
|
|
Config::inst()->update(File::class, 'allowed_extensions', array('txt'));
|
|
|
|
$this->setExpectedException(ValidationException::class);
|
|
|
|
|
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$file->Name = 'renamed.php'; // evil extension
|
|
|
|
$file->write();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetURL()
|
|
|
|
{
|
|
|
|
$rootfile = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->assertEquals('/assets/FileTest/55b443b601/FileTest.txt', $rootfile->getURL());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetAbsoluteURL()
|
|
|
|
{
|
|
|
|
$rootfile = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->assertEquals(
|
|
|
|
Director::absoluteBaseURL() . 'assets/FileTest/55b443b601/FileTest.txt',
|
|
|
|
$rootfile->getAbsoluteURL()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testNameAndTitleGeneration()
|
|
|
|
{
|
|
|
|
// When name is assigned, title is automatically assigned
|
|
|
|
$file = $this->objFromFixture(Image::class, 'setfromname');
|
|
|
|
$this->assertEquals('FileTest', $file->Title);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSizeAndAbsoluteSizeParameters()
|
|
|
|
{
|
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
|
|
|
|
/* AbsoluteSize will give the integer number */
|
|
|
|
$this->assertEquals(1000000, $file->AbsoluteSize);
|
|
|
|
/* Size will give a humanised number */
|
|
|
|
$this->assertEquals('977 KB', $file->Size);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFileType()
|
|
|
|
{
|
|
|
|
$file = $this->objFromFixture(Image::class, 'gif');
|
|
|
|
$this->assertEquals("GIF image - good for diagrams", $file->FileType);
|
|
|
|
|
|
|
|
$file = $this->objFromFixture(File::class, 'pdf');
|
|
|
|
$this->assertEquals("Adobe Acrobat PDF file", $file->FileType);
|
|
|
|
|
|
|
|
$file = $this->objFromFixture(Image::class, 'gifupper');
|
|
|
|
$this->assertEquals("GIF image - good for diagrams", $file->FileType);
|
|
|
|
|
|
|
|
/* Only a few file types are given special descriptions; the rest are unknown */
|
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->assertEquals("unknown", $file->FileType);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test the File::format_size() method
|
|
|
|
*/
|
|
|
|
public function testFormatSize()
|
|
|
|
{
|
|
|
|
$this->assertEquals("1000 bytes", File::format_size(1000));
|
|
|
|
$this->assertEquals("1023 bytes", File::format_size(1023));
|
|
|
|
$this->assertEquals("1 KB", File::format_size(1025));
|
|
|
|
$this->assertEquals("9.8 KB", File::format_size(10000));
|
|
|
|
$this->assertEquals("49 KB", File::format_size(50000));
|
|
|
|
$this->assertEquals("977 KB", File::format_size(1000000));
|
|
|
|
$this->assertEquals("1 MB", File::format_size(1024*1024));
|
|
|
|
$this->assertEquals("954 MB", File::format_size(1000000000));
|
|
|
|
$this->assertEquals("1 GB", File::format_size(1024*1024*1024));
|
|
|
|
$this->assertEquals("9.3 GB", File::format_size(10000000000));
|
|
|
|
// It use any denomination higher than GB. It also doesn't overflow with >32 bit integers
|
|
|
|
$this->assertEquals("93132.3 GB", File::format_size(100000000000000));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDeleteFile()
|
|
|
|
{
|
|
|
|
/**
|
2016-12-29 21:21:00 +13:00
|
|
|
* @var File $file
|
|
|
|
*/
|
2016-12-16 17:34:21 +13:00
|
|
|
$file = $this->objFromFixture(File::class, 'asdf');
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$file->publishSingle();
|
|
|
|
$tuple = $file->File->getValue();
|
|
|
|
|
|
|
|
// Before delete
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($tuple['Filename'], $tuple['Hash']),
|
|
|
|
'File is still present'
|
|
|
|
);
|
|
|
|
|
|
|
|
// after unpublish
|
|
|
|
$file->doUnpublish();
|
|
|
|
$this->assertTrue(
|
|
|
|
$this->getAssetStore()->exists($tuple['Filename'], $tuple['Hash']),
|
|
|
|
'File is still present after unpublish'
|
|
|
|
);
|
|
|
|
|
|
|
|
// after delete
|
|
|
|
$file->delete();
|
|
|
|
$this->assertFalse(
|
|
|
|
$this->getAssetStore()->exists($tuple['Filename'], $tuple['Hash']),
|
|
|
|
'File is deleted after unpublish and delete'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testRenameFolder()
|
|
|
|
{
|
|
|
|
$newTitle = "FileTest-folder-renamed";
|
|
|
|
|
|
|
|
//rename a folder's title
|
|
|
|
$folderID = $this->objFromFixture(Folder::class, "folder2")->ID;
|
|
|
|
$folder = DataObject::get_by_id(Folder::class, $folderID);
|
|
|
|
$folder->Title = $newTitle;
|
|
|
|
$folder->write();
|
|
|
|
|
|
|
|
//get folder again and see if the filename has changed
|
|
|
|
$folder = DataObject::get_by_id(Folder::class, $folderID);
|
|
|
|
$this->assertEquals(
|
2016-12-29 21:21:00 +13:00
|
|
|
$newTitle . '/',
|
2016-12-16 17:34:21 +13:00
|
|
|
$folder->Filename,
|
|
|
|
"Folder Filename updated after rename of Title"
|
|
|
|
);
|
|
|
|
|
|
|
|
//rename a folder's name
|
|
|
|
$newTitle2 = "FileTest-folder-renamed2";
|
|
|
|
$folder->Name = $newTitle2;
|
|
|
|
$folder->write();
|
|
|
|
|
|
|
|
//get folder again and see if the Title has changed
|
|
|
|
$folder = DataObject::get_by_id(Folder::class, $folderID);
|
|
|
|
$this->assertEquals(
|
|
|
|
$folder->Title,
|
|
|
|
$newTitle2,
|
|
|
|
"Folder Title updated after rename of Name"
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
//rename a folder's Filename
|
|
|
|
$newTitle3 = "FileTest-folder-renamed3";
|
|
|
|
$folder->Filename = $newTitle3;
|
|
|
|
$folder->write();
|
|
|
|
|
|
|
|
//get folder again and see if the Title has changed
|
|
|
|
$folder = DataObject::get_by_id(Folder::class, $folderID);
|
|
|
|
$this->assertEquals(
|
|
|
|
$folder->Title,
|
|
|
|
$newTitle3,
|
|
|
|
"Folder Title updated after rename of Filename"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-01-11 16:58:59 +13:00
|
|
|
public function testRenamesDuplicateFilesInSameFolder()
|
|
|
|
{
|
|
|
|
$original = new File();
|
|
|
|
$original->update([
|
|
|
|
'Name' => 'file1.txt',
|
|
|
|
'ParentID' => 0
|
|
|
|
]);
|
|
|
|
$original->write();
|
|
|
|
|
|
|
|
$duplicate = new File();
|
|
|
|
$duplicate->update([
|
|
|
|
'Name' => 'file1.txt',
|
|
|
|
'ParentID' => 0
|
|
|
|
]);
|
|
|
|
$duplicate->write();
|
|
|
|
|
|
|
|
$original = File::get()->byID($original->ID);
|
|
|
|
|
|
|
|
$this->assertEquals($original->Name, 'file1.txt');
|
|
|
|
$this->assertEquals($original->Title, 'file1');
|
|
|
|
$this->assertEquals($duplicate->Name, 'file1-v2.txt');
|
|
|
|
$this->assertEquals($duplicate->Title, 'file1 v2');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetsEmptyTitleToNameWithoutExtensionAndSpecialCharacters()
|
|
|
|
{
|
|
|
|
$fileWithTitle = new File();
|
|
|
|
$fileWithTitle->update([
|
|
|
|
'Name' => 'file1-with-title.txt',
|
|
|
|
'Title' => 'Some Title'
|
|
|
|
]);
|
|
|
|
$fileWithTitle->write();
|
|
|
|
|
|
|
|
$this->assertEquals($fileWithTitle->Name, 'file1-with-title.txt');
|
|
|
|
$this->assertEquals($fileWithTitle->Title, 'Some Title');
|
|
|
|
|
|
|
|
$fileWithoutTitle = new File();
|
|
|
|
$fileWithoutTitle->update([
|
|
|
|
'Name' => 'file1-without-title.txt',
|
|
|
|
]);
|
|
|
|
$fileWithoutTitle->write();
|
|
|
|
|
|
|
|
$this->assertEquals($fileWithoutTitle->Name, 'file1-without-title.txt');
|
|
|
|
$this->assertEquals($fileWithoutTitle->Title, 'file1 without title');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetsEmptyNameToSingularNameWithoutTitle()
|
|
|
|
{
|
|
|
|
$fileWithTitle = new File();
|
|
|
|
$fileWithTitle->update([
|
|
|
|
'Name' => '',
|
|
|
|
'Title' => 'Some Title',
|
|
|
|
]);
|
|
|
|
$fileWithTitle->write();
|
|
|
|
|
|
|
|
$this->assertEquals($fileWithTitle->Name, 'Some-Title');
|
|
|
|
$this->assertEquals($fileWithTitle->Title, 'Some Title');
|
|
|
|
|
|
|
|
$fileWithoutTitle = new File();
|
|
|
|
$fileWithoutTitle->update([
|
|
|
|
'Name' => '',
|
|
|
|
'Title' => '',
|
|
|
|
]);
|
|
|
|
$fileWithoutTitle->write();
|
|
|
|
|
|
|
|
$this->assertEquals($fileWithoutTitle->Name, $fileWithoutTitle->i18n_singular_name());
|
|
|
|
$this->assertEquals($fileWithoutTitle->Title, $fileWithoutTitle->i18n_singular_name());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetsEmptyNameToTitleIfPresent()
|
|
|
|
{
|
|
|
|
$file = new File();
|
|
|
|
$file->update([
|
|
|
|
'Name' => '',
|
|
|
|
'Title' => 'file1',
|
|
|
|
]);
|
|
|
|
$file->write();
|
|
|
|
|
|
|
|
$this->assertEquals($file->Name, 'file1');
|
|
|
|
$this->assertEquals($file->Title, 'file1');
|
|
|
|
}
|
|
|
|
|
2016-12-16 17:34:21 +13:00
|
|
|
public function testSetsOwnerOnFirstWrite()
|
|
|
|
{
|
|
|
|
Session::set('loggedInAs', null);
|
|
|
|
$member1 = new Member();
|
|
|
|
$member1->write();
|
|
|
|
$member2 = new Member();
|
|
|
|
$member2->write();
|
|
|
|
|
|
|
|
$file1 = new File();
|
|
|
|
$file1->write();
|
|
|
|
$this->assertEquals(0, $file1->OwnerID, 'Owner not written when no user is logged in');
|
|
|
|
|
|
|
|
$member1->logIn();
|
|
|
|
$file2 = new File();
|
|
|
|
$file2->write();
|
|
|
|
$this->assertEquals($member1->ID, $file2->OwnerID, 'Owner written when user is logged in');
|
|
|
|
|
|
|
|
$member2->logIn();
|
|
|
|
$file2->forceChange();
|
|
|
|
$file2->write();
|
|
|
|
$this->assertEquals($member1->ID, $file2->OwnerID, 'Owner not overwritten on existing files');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testCanEdit()
|
|
|
|
{
|
|
|
|
$file = $this->objFromFixture(Image::class, 'gif');
|
|
|
|
|
|
|
|
// Test anonymous permissions
|
|
|
|
Session::set('loggedInAs', null);
|
|
|
|
$this->assertFalse($file->canEdit(), "Anonymous users can't edit files");
|
|
|
|
|
|
|
|
// Test permissionless user
|
|
|
|
$this->objFromFixture(Member::class, 'frontend')->logIn();
|
|
|
|
$this->assertFalse($file->canEdit(), "Permissionless users can't edit files");
|
|
|
|
|
|
|
|
// Test global CMS section users
|
|
|
|
$this->objFromFixture(Member::class, 'cms')->logIn();
|
|
|
|
$this->assertTrue($file->canEdit(), "Users with all CMS section access can edit files");
|
|
|
|
|
|
|
|
// Test cms access users without file access
|
|
|
|
$this->objFromFixture(Member::class, 'security')->logIn();
|
|
|
|
$this->assertFalse($file->canEdit(), "Security CMS users can't edit files");
|
|
|
|
|
|
|
|
// Test asset-admin user
|
|
|
|
$this->objFromFixture(Member::class, 'assetadmin')->logIn();
|
|
|
|
$this->assertTrue($file->canEdit(), "Asset admin users can edit files");
|
|
|
|
|
|
|
|
// Test admin
|
|
|
|
$this->objFromFixture(Member::class, 'admin')->logIn();
|
|
|
|
$this->assertTrue($file->canEdit(), "Admins can edit files");
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testJoinPaths()
|
|
|
|
{
|
|
|
|
$this->assertEquals('name/file.jpg', File::join_paths('/name', 'file.jpg'));
|
|
|
|
$this->assertEquals('name/file.jpg', File::join_paths('name', 'file.jpg'));
|
|
|
|
$this->assertEquals('name/file.jpg', File::join_paths('/name', '/file.jpg'));
|
|
|
|
$this->assertEquals('name/file.jpg', File::join_paths('name/', '/', 'file.jpg'));
|
|
|
|
$this->assertEquals('file.jpg', File::join_paths('/', '/', 'file.jpg'));
|
|
|
|
$this->assertEquals('', File::join_paths('/', '/'));
|
|
|
|
}
|
|
|
|
|
2016-12-29 21:21:00 +13:00
|
|
|
/**
|
|
|
|
* Test that ini2bytes returns the number of bytes for a PHP ini style size declaration
|
|
|
|
*
|
|
|
|
* @param string $iniValue
|
|
|
|
* @param int $expected
|
|
|
|
* @dataProvider ini2BytesProvider
|
|
|
|
*/
|
|
|
|
public function testIni2Bytes($iniValue, $expected)
|
|
|
|
{
|
|
|
|
$this->assertSame($expected, File::ini2bytes($iniValue));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function ini2BytesProvider()
|
|
|
|
{
|
|
|
|
return [
|
2016-12-29 22:21:28 +13:00
|
|
|
['2k', 2 * 1024],
|
|
|
|
['512M', 512 * 1024 * 1024],
|
|
|
|
['1024g', 1024 * 1024 * 1024 * 1024],
|
|
|
|
['1024G', 1024 * 1024 * 1024 * 1024]
|
2016-12-29 21:21:00 +13:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2016-12-16 17:34:21 +13:00
|
|
|
/**
|
|
|
|
* @return AssetStore
|
|
|
|
*/
|
|
|
|
protected function getAssetStore()
|
|
|
|
{
|
|
|
|
return Injector::inst()->get('AssetStore');
|
|
|
|
}
|
2011-08-22 13:04:15 +02:00
|
|
|
}
|