2015-09-15 14:52:02 +12:00
|
|
|
<?php
|
|
|
|
|
2016-08-19 10:51:35 +12:00
|
|
|
namespace SilverStripe\Assets\Storage;
|
|
|
|
|
|
|
|
use SilverStripe\Assets\File;
|
|
|
|
use SilverStripe\Assets\Thumbnail;
|
|
|
|
use SilverStripe\Assets\ImageManipulation;
|
|
|
|
use SilverStripe\Core\Injector\Injector;
|
|
|
|
use SilverStripe\Control\Director;
|
|
|
|
use SilverStripe\Forms\AssetField;
|
2016-06-15 16:03:16 +12:00
|
|
|
use SilverStripe\ORM\ValidationResult;
|
|
|
|
use SilverStripe\ORM\ValidationException;
|
|
|
|
use SilverStripe\ORM\FieldType\DBComposite;
|
2016-06-23 11:37:22 +12:00
|
|
|
use SilverStripe\Security\Permission;
|
2016-09-07 15:35:47 +12:00
|
|
|
use SilverStripe\Core\Convert;
|
2016-06-23 11:37:22 +12:00
|
|
|
|
2015-09-15 14:52:02 +12:00
|
|
|
/**
|
|
|
|
* Represents a file reference stored in a database
|
|
|
|
*
|
|
|
|
* @property string $Hash SHA of the file
|
|
|
|
* @property string $Filename Name of the file, including directory
|
|
|
|
* @property string $Variant Variant of the file
|
|
|
|
*/
|
2016-04-12 10:24:16 +12:00
|
|
|
class DBFile extends DBComposite implements AssetContainer, Thumbnail {
|
2015-09-15 14:52:02 +12:00
|
|
|
|
|
|
|
use ImageManipulation;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of allowed file categories.
|
|
|
|
*
|
|
|
|
* {@see File::$app_categories}
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $allowedCategories = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of image mime types supported by the image manipulations API
|
|
|
|
*
|
|
|
|
* {@see File::app_categories} for matching extensions.
|
|
|
|
*
|
|
|
|
* @config
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $supported_images = array(
|
|
|
|
'image/jpeg',
|
|
|
|
'image/gif',
|
|
|
|
'image/png'
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new image manipulation
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
* @param array|string $allowed List of allowed file categories (not extensions), as per File::$app_categories
|
|
|
|
*/
|
|
|
|
public function __construct($name = null, $allowed = array()) {
|
|
|
|
parent::__construct($name);
|
|
|
|
$this->setAllowedCategories($allowed);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if a valid non-empty image exists behind this asset, which is a format
|
|
|
|
* compatible with image manipulations
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function getIsImage() {
|
|
|
|
// Check file type
|
|
|
|
$mime = $this->getMimeType();
|
|
|
|
return $mime && in_array($mime, $this->config()->supported_images);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return AssetStore
|
|
|
|
*/
|
|
|
|
protected function getStore() {
|
|
|
|
return Injector::inst()->get('AssetStore');
|
|
|
|
}
|
|
|
|
|
|
|
|
private static $composite_db = array(
|
|
|
|
"Hash" => "Varchar(255)", // SHA of the base content
|
|
|
|
"Filename" => "Varchar(255)", // Path identifier of the base content
|
|
|
|
"Variant" => "Varchar(255)", // Identifier of the variant to the base, if given
|
|
|
|
);
|
|
|
|
|
|
|
|
private static $casting = array(
|
|
|
|
'URL' => 'Varchar',
|
|
|
|
'AbsoluteURL' => 'Varchar',
|
|
|
|
'Basename' => 'Varchar',
|
|
|
|
'Title' => 'Varchar',
|
|
|
|
'MimeType' => 'Varchar',
|
|
|
|
'String' => 'Text',
|
2016-06-03 20:51:02 +12:00
|
|
|
'Tag' => 'HTMLFragment',
|
2015-10-19 17:27:27 +13:00
|
|
|
'Size' => 'Varchar'
|
2015-09-15 14:52:02 +12:00
|
|
|
);
|
|
|
|
|
|
|
|
public function scaffoldFormField($title = null, $params = null) {
|
2015-10-19 17:27:27 +13:00
|
|
|
return AssetField::create($this->getName(), $title);
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a html5 tag of the appropriate for this file (normally img or a)
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2016-06-03 20:51:02 +12:00
|
|
|
public function XML() {
|
2015-09-15 14:52:02 +12:00
|
|
|
return $this->getTag() ?: '';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a html5 tag of the appropriate for this file (normally img or a)
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getTag() {
|
|
|
|
$template = $this->getFrontendTemplate();
|
|
|
|
if(empty($template)) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
return (string)$this->renderWith($template);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine the template to render as on the frontend
|
|
|
|
*
|
|
|
|
* @return string Name of template
|
|
|
|
*/
|
|
|
|
public function getFrontendTemplate() {
|
|
|
|
// Check that path is available
|
|
|
|
$url = $this->getURL();
|
|
|
|
if(empty($url)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Image template for supported images
|
|
|
|
if($this->getIsImage()) {
|
|
|
|
return 'DBFile_image';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Default download
|
|
|
|
return 'DBFile_download';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get trailing part of filename
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getBasename() {
|
2015-12-10 10:19:23 +13:00
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
return basename($this->getSourceURL());
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get file extension
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getExtension() {
|
2015-12-10 10:19:23 +13:00
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
return pathinfo($this->Filename, PATHINFO_EXTENSION);
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Alt title for this
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getTitle() {
|
|
|
|
// If customised, use the customised title
|
|
|
|
if($this->failover && ($title = $this->failover->Title)) {
|
|
|
|
return $title;
|
|
|
|
}
|
|
|
|
// fallback to using base name
|
|
|
|
return $this->getBasename();
|
|
|
|
}
|
|
|
|
|
2015-12-10 10:19:23 +13:00
|
|
|
public function setFromLocalFile($path, $filename = null, $hash = null, $variant = null, $config = array()) {
|
2015-09-15 14:52:02 +12:00
|
|
|
$this->assertFilenameValid($filename ?: $path);
|
|
|
|
$result = $this
|
|
|
|
->getStore()
|
2015-12-10 10:19:23 +13:00
|
|
|
->setFromLocalFile($path, $filename, $hash, $variant, $config);
|
2015-09-15 14:52:02 +12:00
|
|
|
// Update from result
|
|
|
|
if($result) {
|
|
|
|
$this->setValue($result);
|
|
|
|
}
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2015-12-10 10:19:23 +13:00
|
|
|
public function setFromStream($stream, $filename, $hash = null, $variant = null, $config = array()) {
|
2015-09-15 14:52:02 +12:00
|
|
|
$this->assertFilenameValid($filename);
|
|
|
|
$result = $this
|
|
|
|
->getStore()
|
2015-12-10 10:19:23 +13:00
|
|
|
->setFromStream($stream, $filename, $hash, $variant, $config);
|
2015-09-15 14:52:02 +12:00
|
|
|
// Update from result
|
|
|
|
if($result) {
|
|
|
|
$this->setValue($result);
|
|
|
|
}
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2015-12-10 10:19:23 +13:00
|
|
|
public function setFromString($data, $filename, $hash = null, $variant = null, $config = array()) {
|
2015-09-15 14:52:02 +12:00
|
|
|
$this->assertFilenameValid($filename);
|
|
|
|
$result = $this
|
|
|
|
->getStore()
|
2015-12-10 10:19:23 +13:00
|
|
|
->setFromString($data, $filename, $hash, $variant, $config);
|
2015-09-15 14:52:02 +12:00
|
|
|
// Update from result
|
|
|
|
if($result) {
|
|
|
|
$this->setValue($result);
|
|
|
|
}
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getStream() {
|
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->getAsStream($this->Filename, $this->Hash, $this->Variant);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getString() {
|
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->getAsString($this->Filename, $this->Hash, $this->Variant);
|
|
|
|
}
|
|
|
|
|
2015-12-10 10:19:23 +13:00
|
|
|
public function getURL($grant = true) {
|
2015-09-15 14:52:02 +12:00
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
$url = $this->getSourceURL($grant);
|
2015-09-15 14:52:02 +12:00
|
|
|
$this->updateURL($url);
|
|
|
|
$this->extend('updateURL', $url);
|
|
|
|
return $url;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get URL, but without resampling.
|
2015-10-19 17:27:27 +13:00
|
|
|
* Note that this will return the url even if the file does not exist.
|
2015-09-15 14:52:02 +12:00
|
|
|
*
|
2015-12-10 10:19:23 +13:00
|
|
|
* @param bool $grant Ensures that the url for any protected assets is granted for the current user.
|
2015-09-15 14:52:02 +12:00
|
|
|
* @return string
|
|
|
|
*/
|
2015-12-10 10:19:23 +13:00
|
|
|
public function getSourceURL($grant = true) {
|
2015-09-15 14:52:02 +12:00
|
|
|
return $this
|
|
|
|
->getStore()
|
2015-12-10 10:19:23 +13:00
|
|
|
->getAsURL($this->Filename, $this->Hash, $this->Variant, $grant);
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the absolute URL to this resource
|
|
|
|
*
|
2015-12-10 10:19:23 +13:00
|
|
|
* @return string
|
2015-09-15 14:52:02 +12:00
|
|
|
*/
|
|
|
|
public function getAbsoluteURL() {
|
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return Director::absoluteURL($this->getURL());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getMetaData() {
|
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->getMetadata($this->Filename, $this->Hash, $this->Variant);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getMimeType() {
|
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->getMimeType($this->Filename, $this->Hash, $this->Variant);
|
|
|
|
}
|
2016-03-09 09:50:18 +13:00
|
|
|
|
2015-09-15 14:52:02 +12:00
|
|
|
public function getValue() {
|
2015-12-10 10:19:23 +13:00
|
|
|
if(!$this->exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return array(
|
|
|
|
'Filename' => $this->Filename,
|
|
|
|
'Hash' => $this->Hash,
|
|
|
|
'Variant' => $this->Variant
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getVisibility() {
|
|
|
|
if(empty($this->Filename)) {
|
|
|
|
return null;
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->getVisibility($this->Filename, $this->Hash);
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
public function exists() {
|
2015-10-19 17:27:27 +13:00
|
|
|
if(empty($this->Filename)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->exists($this->Filename, $this->Hash, $this->Variant);
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
2016-03-09 09:50:18 +13:00
|
|
|
|
2015-09-15 14:52:02 +12:00
|
|
|
public function getFilename() {
|
|
|
|
return $this->getField('Filename');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getHash() {
|
|
|
|
return $this->getField('Hash');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getVariant() {
|
|
|
|
return $this->getField('Variant');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return file size in bytes.
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public function getAbsoluteSize() {
|
|
|
|
$metadata = $this->getMetaData();
|
|
|
|
if(isset($metadata['size'])) {
|
|
|
|
return $metadata['size'];
|
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
return 0;
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Customise this object with an "original" record for getting other customised fields
|
|
|
|
*
|
|
|
|
* @param AssetContainer $original
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setOriginal($original) {
|
|
|
|
$this->failover = $original;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get list of allowed file categories
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getAllowedCategories() {
|
|
|
|
return $this->allowedCategories;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assign allowed categories
|
|
|
|
*
|
|
|
|
* @param array|string $categories
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setAllowedCategories($categories) {
|
|
|
|
if(is_string($categories)) {
|
|
|
|
$categories = preg_split('/\s*,\s*/', $categories);
|
|
|
|
}
|
|
|
|
$this->allowedCategories = (array)$categories;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the list of extensions (if limited) for this field. Empty list
|
|
|
|
* means there is no restriction on allowed types.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getAllowedExtensions() {
|
|
|
|
$categories = $this->getAllowedCategories();
|
|
|
|
return File::get_category_extensions($categories);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate that this DBFile accepts this filename as valid
|
|
|
|
*
|
|
|
|
* @param string $filename
|
|
|
|
* @throws ValidationException
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function isValidFilename($filename) {
|
|
|
|
$extension = strtolower(File::get_file_extension($filename));
|
|
|
|
|
|
|
|
// Validate true if within the list of allowed extensions
|
|
|
|
$allowed = $this->getAllowedExtensions();
|
|
|
|
if($allowed) {
|
|
|
|
return in_array($extension, $allowed);
|
|
|
|
}
|
2016-03-09 09:50:18 +13:00
|
|
|
|
2015-09-15 14:52:02 +12:00
|
|
|
// If no extensions are configured, fallback to global list
|
|
|
|
$globalList = File::config()->allowed_extensions;
|
|
|
|
if(in_array($extension, $globalList)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only admins can bypass global rules
|
|
|
|
return !File::config()->apply_restrictions_to_admin && Permission::check('ADMIN');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check filename, and raise a ValidationException if invalid
|
|
|
|
*
|
|
|
|
* @param string $filename
|
|
|
|
* @throws ValidationException
|
|
|
|
*/
|
|
|
|
protected function assertFilenameValid($filename) {
|
|
|
|
$result = new ValidationResult();
|
|
|
|
$this->validate($result, $filename);
|
|
|
|
if(!$result->valid()) {
|
|
|
|
throw new ValidationException($result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook to validate this record against a validation result
|
|
|
|
*
|
|
|
|
* @param ValidationResult $result
|
|
|
|
* @param string $filename Optional filename to validate. If omitted, the current value is validated.
|
|
|
|
* @return bool Valid flag
|
|
|
|
*/
|
|
|
|
public function validate(ValidationResult $result, $filename = null) {
|
|
|
|
if(empty($filename)) {
|
|
|
|
$filename = $this->getFilename();
|
|
|
|
}
|
|
|
|
if(empty($filename) || $this->isValidFilename($filename)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check allowed extensions
|
|
|
|
$extensions = $this->getAllowedExtensions();
|
|
|
|
if(empty($extensions)) {
|
|
|
|
$extensions = File::config()->allowed_extensions;
|
|
|
|
}
|
|
|
|
sort($extensions);
|
|
|
|
$message = _t(
|
|
|
|
'File.INVALIDEXTENSION',
|
|
|
|
'Extension is not allowed (valid: {extensions})',
|
|
|
|
'Argument 1: Comma-separated list of valid extensions',
|
|
|
|
array('extensions' => wordwrap(implode(', ',$extensions)))
|
|
|
|
);
|
|
|
|
$result->error($message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setField($field, $value, $markChanged = true) {
|
|
|
|
// Catch filename validation on direct assignment
|
|
|
|
if($field === 'Filename' && $value) {
|
|
|
|
$this->assertFilenameValid($value);
|
|
|
|
}
|
2016-03-09 09:50:18 +13:00
|
|
|
|
2015-09-15 14:52:02 +12:00
|
|
|
return parent::setField($field, $value, $markChanged);
|
|
|
|
}
|
2015-10-19 17:27:27 +13:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the size of the file type in an appropriate format.
|
|
|
|
*
|
|
|
|
* @return string|false String value, or false if doesn't exist
|
|
|
|
*/
|
|
|
|
public function getSize() {
|
|
|
|
$size = $this->getAbsoluteSize();
|
|
|
|
if($size) {
|
2016-08-19 10:51:35 +12:00
|
|
|
return File::format_size($size);
|
2015-10-19 17:27:27 +13:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2015-12-10 10:19:23 +13:00
|
|
|
|
|
|
|
public function deleteFile() {
|
|
|
|
if(!$this->Filename) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this
|
|
|
|
->getStore()
|
|
|
|
->delete($this->Filename, $this->Hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function publishFile() {
|
|
|
|
if($this->Filename) {
|
|
|
|
$this
|
|
|
|
->getStore()
|
|
|
|
->publish($this->Filename, $this->Hash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function protectFile() {
|
|
|
|
if($this->Filename) {
|
|
|
|
$this
|
|
|
|
->getStore()
|
|
|
|
->protect($this->Filename, $this->Hash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function grantFile() {
|
|
|
|
if($this->Filename) {
|
|
|
|
$this
|
|
|
|
->getStore()
|
|
|
|
->grant($this->Filename, $this->Hash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function revokeFile() {
|
|
|
|
if($this->Filename) {
|
|
|
|
$this
|
|
|
|
->getStore()
|
|
|
|
->revoke($this->Filename, $this->Hash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function canViewFile() {
|
|
|
|
return $this->Filename
|
|
|
|
&& $this
|
|
|
|
->getStore()
|
|
|
|
->canView($this->Filename, $this->Hash);
|
|
|
|
}
|
2016-09-07 15:35:47 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates the URL for this DBFile preview, this is particularly important for images that
|
|
|
|
* have been manipulated e.g. by {@link ImageManipulation}
|
|
|
|
* Use the 'updatePreviewLink' extension point to customise the link.
|
|
|
|
*
|
|
|
|
* @param null $action
|
|
|
|
* @return bool|string
|
|
|
|
*/
|
|
|
|
public function PreviewLink($action = null) {
|
|
|
|
// Since AbsoluteURL can whitelist protected assets,
|
|
|
|
// do permission check first
|
|
|
|
if (!$this->failover->canView()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if ($this->getIsImage()) {
|
|
|
|
$link = $this->getAbsoluteURL();
|
|
|
|
} else {
|
|
|
|
$link = Convert::raw2att($this->failover->getIcon());
|
|
|
|
}
|
|
|
|
$this->extend('updatePreviewLink', $link, $action);
|
|
|
|
return $link;
|
|
|
|
}
|
2015-09-15 14:52:02 +12:00
|
|
|
}
|