я использую UploadedFile
класс в symfony2 для загрузки файлов изображений. Я позволил изображения с типом пантомимы image/jpg
а также image/png
, Когда я загружаю файл png из моей системы на сервер, его тип mime изменяется автоматически. Я знаю, что это изменилось, потому что, когда я запускаю эту команду:
file --mime-type /home/mysuer/img.png
это дает мне это:
home/myuser/img.png: image/png
но когда я загружаю файл и печатаю UploadedFile
объект дает следующий вывод:
Array
(
[media] => Symfony\Component\HttpFoundation\File\UploadedFile Object
(
[test:Symfony\Component\HttpFoundation\File\UploadedFile:private] =>
[originalName:Symfony\Component\HttpFoundation\File\UploadedFile:private] => img.png
[mimeType:Symfony\Component\HttpFoundation\File\UploadedFile:private] => application/octet-stream
[size:Symfony\Component\HttpFoundation\File\UploadedFile:private] => 1246
[error:Symfony\Component\HttpFoundation\File\UploadedFile:private] => 0
[pathName:SplFileInfo:private] => /tmp/phpivgnne
[fileName:SplFileInfo:private] => phpivgnne
)
)
Кто-нибудь может подсказать, что не так?
Вот мой код:
public function postFileAction(Request $request)
{
print_r($request->files->all());//Prints All files$image = $request->files->get('media');
echo $image->getClientMimeType();exit; //Mime Type Information
if (empty($image)) {
throw new FileException('Invalid File, Please Select a Valid File.');
}
$uploader = $this->get('application.file_uploader');
if (!$this->container->get('security.context')->isGranted('IS_AUTHENTICATED_FULLY')) {
//authenticated (NON anonymous users)
throw new InvalidParameterException('Invalid User, Please use valid credentials');
}
return $uploader->upload($image, $this->get('security.context')->getToken()->getUser()->getId());
}
Тестовый код PHPUnit
public function testPostFile()
{
$file = tempnam(sys_get_temp_dir(), 'upl'); // create file
imagepng(imagecreatetruecolor(10, 10), $file); // create and write image/png to it
$image = new UploadedFile($file, 'new_image.png', 'image/png', 10, UPLOAD_ERR_OK);
$crawler = $this->client->request(
'POST', '/api/v1/files.json', array('submit' => 'Submit Form'), array('media' => $image)
);
$response = $this->client->getResponse();
$this->assertJsonResponse($response);
$this->assertContains('filename', $response->getContent());
print_r($response->getContent());exit;
$fileToken = json_decode($response->getContent());
$this->assertNotEmpty($fileToken->filename);
unlink($file);
unset($file, $image, $crawler, $response);
return $fileToken->filename;
}
я использую curl
из командной строки, чтобы проверить мою загрузку файла веб-службы REST следующим образом:
curl -X POST myapi.com/app_dev.php/api/v1/files.json --form "media=@/home/myuser/img.png"
Наконец, я получил решение моей проблемы и опубликую ответ на проблему как для core-php, так и для symfony2.
Основной PHP (найден на http://php.net/manual/en/features.file-upload.php):
// DO NOT TRUST $_FILES['upfile']['mime'] VALUE !!
// Check MIME Type by yourself.
$finfo = new finfo(FILEINFO_MIME_TYPE);
if (false === $ext = array_search(
$finfo->file($_FILES['upfile']['tmp_name']),
array(
'jpg' => 'image/jpeg',
'png' => 'image/png',
'gif' => 'image/gif',
),
true
)) {
throw new RuntimeException('Invalid file format.');
}
Так что никогда не верь $_FILES['upfile']['mime']
всегда использовать fileinfo
и это дало мне ключ к разгадке.
для Symfony2:
Итак, я начал изучать свой код и обнаружил, что я использую $file->getClientMimeType()
что не хорошо мы должны использовать $file->getMimeType()
который использует fileinfo
чтобы выбрать тип MIME и даст правильный тип MIME.
В getMimeType()
Тип MIME угадывается с помощью MimeTypeGuesser
экземпляр, который использует finfo()
, mime_content_type()
и системный двоичный «файл» (в этом порядке), в зависимости от того, какие из них доступны.
Спасибо всем за помощь: D
Других решений пока нет …