2, 1 => 2, 2 => 2, 3 => 2, 4 => 2, ]; protected $fileCountByDepth = [ 0 => 10, 1 => 8, 2 => 5, 3 => 5, 4 => 5, ]; protected $fixtureFileBaseUrl = "https://s3-ap-southeast-2.amazonaws.com/silverstripe-frameworktest-assets/"; protected $defaultImageFileName = 'image-huge-tall.jpg'; protected $fixtureFileNames = [ 'archive.zip', 'animated.gif', 'document.docx', 'document.pdf', 'image-huge-tall.jpg', 'image-huge-wide.jpg', 'image-large.jpg', 'image-large.png', 'image-large.gif', 'image-medium.jpg', 'image-small.jpg', 'image-tiny.jpg', 'image-medium.bmp', 'spreadsheet.xlsx', 'video.m4v' ]; protected $fixtureFileTypes = [ 'archive.zip' => 'SilverStripe\Assets\File', 'animated.gif' => 'SilverStripe\Assets\Image', 'document.docx' => 'SilverStripe\Assets\File', 'document.pdf' => 'SilverStripe\Assets\File', 'image-huge-tall.jpg' => 'SilverStripe\Assets\Image', 'image-huge-wide.jpg' => 'SilverStripe\Assets\Image', 'image-large.jpg' => 'SilverStripe\Assets\Image', 'image-large.png' => 'SilverStripe\Assets\Image', 'image-large.gif' => 'SilverStripe\Assets\Image', 'image-medium.jpg' => 'SilverStripe\Assets\Image', 'image-small.jpg' => 'SilverStripe\Assets\Image', 'image-tiny.jpg' => 'SilverStripe\Assets\Image', 'image-medium.bmp' => 'SilverStripe\Assets\File', 'spreadsheet.xlsx' => 'SilverStripe\Assets\File', 'video.m4v' => 'SilverStripe\Assets\File', ]; protected $lineBreak = "\n
"; /** @var Member */ protected $anonymousMember = null; public function run($request) { set_time_limit(0); // used to test canView() permissions $this->anonymousMember = Member::get()->find('Email', 'frameworktestuser'); if (!$this->anonymousMember) { $this->anonymousMember = Member::create(); $this->anonymousMember->Email = 'frameworktestuser'; $this->anonymousMember->write(); } if (!$this->anonymousMember->inGroup('content-authors')) { $this->anonymousMember->addToGroupByCode('content-authors'); } Security::setCurrentUser($this->anonymousMember); if (php_sapi_name() == "cli") { $this->lineBreak = "\n"; } if ($request->getVar('reset')) { $this->reset(); } echo "Downloading fixtures" . $this->lineBreak; $fixtureFilePaths = $this->downloadFixtureFiles(); if (!self::config()->get('documentsOnly')) { echo "Generate thumbnails" . $this->lineBreak; $this->generateThumbnails($fixtureFilePaths); } echo "Generate files" . $this->lineBreak; $this->generateFiles($fixtureFilePaths); if (!self::config()->get('doPutProtectedFilesInPublicStore')) { echo "Incorrectly putting protected files into public asset store on purpose" . $this->lineBreak; $this->putProtectedFilesInPublicAssetStore(); } } protected function reset() { echo "Resetting assets" . $this->lineBreak; DB::query('TRUNCATE "File"'); DB::query('TRUNCATE "File_Live"'); DB::query('TRUNCATE "File_Versions"'); if (file_exists(ASSETS_PATH) && ASSETS_PATH && ASSETS_PATH !== '/') { exec("rm -rf " . ASSETS_PATH); } } protected function downloadFixtureFiles() { $client = new Client(['base_uri' => $this->fixtureFileBaseUrl]); $fixtureFileNames = $this->fixtureFileNames; if (self::config()->get('documentsOnly')) { $fixtureFileNames = array_filter($fixtureFileNames, function($v) { return (bool) preg_match('%\.(docx|xlsx|pdf)$%', $v); }); } // Initiate each request but do not block $promises = []; $paths = []; foreach ($fixtureFileNames as $filename) { $path = TEMP_FOLDER . '/' . $filename; $paths[$filename] = $path; $url = "{$this->fixtureFileBaseUrl}/{$filename}"; if (!file_exists($path)) { $promises[$filename] = $client->getAsync($filename, [ 'sink' => $path ]); echo "Downloading $url" . $this->lineBreak; } } // Wait on all of the requests to complete. Throws a ConnectException // if any of the requests fail Promise\unwrap($promises); return $paths; } /** * Creates thumbnails of sample images * * @param array $fixtureFilePaths */ protected function generateThumbnails($fixtureFilePaths) { $folder = Folder::find_or_make('testfolder-thumbnail'); $fileName = $this->defaultImageFileName; foreach(['draft', 'published'] as $state) { $file = new Image([ 'ParentID' => $folder->ID, 'Title' => "{$fileName} {$state}", 'Name' => $fileName, ]); $file->File->setFromLocalFile($fixtureFilePaths[$fileName], $folder->getFilename() . $fileName); $file->write(); if ($state === 'published') { $file->publishFile(); } $file->Pad(60, 60)->CropHeight(30); } } protected function generateFiles($fixtureFilePaths, $depth = 0, $prefix = "0", $parentID = 0) { $folderCount = self::config()->get('folderCountByDepth')[$depth]; $fileCount = self::config()->get('fileCountByDepth')[$depth]; $doSetFolderPermissions = (bool) self::config()->get('doSetFolderPermissions'); $doSetOldCreationDate = (bool) self::config()->get('doSetOldCreationDate'); $doRandomlyPublish = (bool) self::config()->get('doRandomlyPublish'); for ($i = 1; $i <= $folderCount; $i++) { $folder = new Folder([ 'ParentID' => $parentID, 'Title' => "testfolder-{$prefix}{$i}", 'Name' => "testfolder-{$prefix}{$i}", ]); if ($doSetFolderPermissions) { if ($i == 1) { // the first folder should always be public to ensure there's some public folders $folder->CanViewType = 'Inherit'; } elseif ($i == $folderCount) { // the last folder should always be protected to ensure there's some protected folders $folder->CanViewType = 'OnlyTheseUsers'; } else { // all the other folder have a 33% chance of being a protected folder $folder->CanViewType = rand(0, 2) == 0 ? 'OnlyTheseUsers' : 'Inherit'; } } $folder->write(); for ($j = 1; $j <= $fileCount; $j++) { $randomFileName = array_keys($fixtureFilePaths)[rand(0, count($fixtureFilePaths) - 1)]; $randomFilePath = $fixtureFilePaths[$randomFileName]; $fileName = pathinfo($randomFilePath, PATHINFO_FILENAME) . "-{$prefix}-{$j}" . "." . pathinfo($randomFilePath, PATHINFO_EXTENSION); // Add a random prefix to avoid all types of files showing up on a single screen page $fileName = substr(md5($fileName), 0, 5) . '-' . $fileName; $class = $this->fixtureFileTypes[$randomFileName]; $file = new $class([ 'ParentID' => $folder->ID, 'Title' => $fileName, 'Name' => $fileName, ]); $file->File->setFromLocalFile($randomFilePath, $folder->getFilename() . $fileName); // Randomly set old created date (for testing) if ($doSetOldCreationDate) { if (rand(0, 10) == 0) { $file->Created = '2010-01-01 00:00:00'; $file->Title = '[old] ' . $file->Title; } } $file->write(); if ($doRandomlyPublish) { if (rand(0, 1) == 0) { $file->publishFile(); } } else { // publish files that should be viewable if ($file->canView($this->anonymousMember)) { $url = $file->getAbsoluteURL(); $file->publishFile(); } } } if ($depth < self::config()->get('depth') - 1) { $this->generateFiles($fixtureFilePaths, $depth + 1, "{$prefix}-{$i}", $folder->ID); } } } protected function putProtectedFilesInPublicAssetStore() { /** @var File $file */ foreach (File::get()->exclude(['ClassName' => Folder::class]) as $file) { // file is already in public asset store if ($file->canView($this->anonymousMember)) { continue; } // randomly move 50% of the files that should be in the protected store to the public store if (rand(0, 1) == 0) { continue; } // this will move the file into the public asset store, even it it should be protected // i.e. the parent folder CanViewType = OnlyTheseUsers $file->publishFile(); $url = $file->getAbsoluteURL(); } } }