Как сохранить исходное имя файла После загрузки в Symfony
пользователи из бэкэнд-приложение может загружать файлы и публиковать их на сайте. Используя sfWidgetFormInputFile и sfValidatorFile, я хотел бы сохранить исходное имя файла вместо функциональности по умолчанию случайной строки (т. е. Meaningful_filename.docx вместо a4b25e9f48cfb6268f34b691fc18cd76fefe96b5.docx-номера могут быть добавлены к дубликатам имен). Это может быть полезно в сценариях, где пользователь загружает несколько файлов и не сможет отличить их друг от друга из имени файла.
$this->widgetSchema['file_name'] = new sfWidgetFormInputFile(array('label' => 'File'));
$this->validatorSchema['file_name'] = new sfValidatorFile(array(
'required' => true,
'path' => sfConfig::get('sf_upload_dir').DIRECTORY_SEPARATOR.sfConfig::get('app_dir_file_sharing').DIRECTORY_SEPARATOR,
'mime_types' => array('application/msword',
'application/vnd.ms-word',
'application/msword',
'application/msword; charset=binary')
), array(
'invalid' => 'Invalid file.',
'required' => 'Select a file to upload.',
'mime_types' => 'The file must be a supported type.'
));
есть ли собственные функции в виджете sfWidgetFormInputFile или есть другое решение для этого?
3 ответов
вы получаете файл, вызвав $form["file_name"]->getValue()
. Это дает вам объект класса sfValidatedFile
где вы можете вызвать метод getOriginalName()
.
чтобы определить, как файл должен быть сохранен, Вы можете сделать это:
The sfValidatorFile
класс принимает параметр, который sfValidatedFile
класс:
validated_file_class: имя класса, который управляет очищенным загруженным файлом (необязательно)
на sfValidatedFile
класса метод save
это вызывает метод generateFileName
. Подкласс этого класса и перезаписать этот метод:
class CustomValidatedFile extends sfValidatedFile {
/**
* Generates a random filename for the current file.
*
* @return string A random name to represent the current file
*/
public function generateFilename()
{
return 'foo bar'// your custom generated file name;
}
}
вот функция из исходного класса:
public function generateFilename()
{
return sha1($this->getOriginalName().rand(11111, 99999)).$this->getExtension($this->getOriginalExtension());
}
затем вы настраиваете валидатор следующим образом:
$this->validatorSchema['file_name'] = new sfValidatorFile(array(
'required' => true,
'path' => 'yourpath',
'validated_file_class' => 'CustomValidatedFile',
'mime_types' => array('application/msword',
'application/vnd.ms-word',
'application/msword',
'application/msword; charset=binary')
),
array('invalid' => 'Invalid file.',
'required' => 'Select a file to upload.',
'mime_types' => 'The file must be a supported type.')
);
надеюсь, что это поможет!
после некоторых исследований:
хотя вы можете расширить sfValidatedFile и переопределить generateFilename, я обнаружил, что sfFormPropel проверяет наличие метода на основе имени столбца для модели, чтобы назвать файл.
из symfony / плагинов / sfPropelPlugin/lib / form line 292:
$method = sprintf('generate%sFilename', $column);
if (null !== $filename)
{
return $file->save($filename);
}
else if (method_exists($this, $method))
{
return $file->save($this->$method($file));
}
поэтому, если ваш столбец называется file_name, метод ищет существование generateFileNameFilename в классе формы. Таким образом, вам нужно только добавить один метод для класса формы, а не расширение виджета sfValidatedFile. Например, моя функция использует исходное имя, если оно не принято, иначе добавляет последовательный номер (один из методов-рекурсивно проверять сгенерированное имя файла):
public function generateFileNameFilename($file = null)
{
if (null === $file) {
// use a random filename instead
return null;
}
if (file_exists($file->getpath().$file->getOriginalName())) {
return $this->appendToName($file);
}
return $file->getOriginalName();
}
public function appendToName($file, $index = 0)
{
$newname = pathinfo($file->getOriginalName(), PATHINFO_FILENAME).$index.$file->getExtension();
if (file_exists($file->getpath().$newname)) {
return $this->appendToName($file, ++$index);
} else {
return $newname;
}
}
Я не могу найти это задокументировано в Symfony API в любом месте, поэтому потребовался поиск в базе кода. Если вы используете этот метод во многих местах, расширение sfValidatedFile может быть хорошим вариантом тоже.
согласно документации Symfony " валидатор sfValidatorFile проверяет загруженный файл. Валидатор преобразует загруженный файл в экземпляр класса sfValidatedFile или параметра validated_file_class, если он установлен."(Источник: http://www.symfony-project.org/forms/1_4/en/B-Validators#chapter_b_sub_sfvalidatorfile)
хотя класс sfValidatedFile переименовывает файлы прямо из коробки, вы можете переопределить эту функцию, установив validated_file_class в пользовательский класс и расширение sfValidatedFile.
в вашем пользовательском проверенном классе файлов передайте свое пользовательское имя файла методу save (). "Если вы не передадите имя файла,оно будет сгенерировано методом generateFilename." (Источник: http://www.symfony-project.org/api/1_4/sfValidatedFile#method_save)
вот один из способов вы могли бы это сделать (Источник: http://forum.symfony-project.org/index.php/m/90887/#msg_90887)...
пользовательский файл проверен класс:
// lib/validator/myValidatedFile.php
class myValidatedFile extends sfValidatedFile {
private $savedFilename = null;
// Override sfValidatedFile's save method
public function save($file = null, $fileMode = 0666, $create = true, $dirMode = 0777) {
// This makes sure we use only one savedFilename (it will be the first)
if ($this->savedFilename === null) $this->savedFilename = $file;
// Let the original save method do its magic :)
return parent::save($this->savedFilename, $fileMode, $create, $dirMode);
}
}
обязательно установите 'validated_file_class' => 'myValidatedFile'
для sfWidgetFormInputFile. И установить логику для того, что имя файла будет в методе сохранения формы.