我在symfony2中使用UploadedFile
类来上传图像文件。我允许使用mime类型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 Web服务文件上传,如下所示:
curl -X POST myapi.com/app_dev.php/api/v1/files.json --form "media=@/home/myuser/img.png"
答案 0 :(得分:5)
最后我得到了我的问题的解决方案,将在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
,这给了我线索。
:
所以我开始查看我的代码并发现我使用的是$file->getClientMimeType()
,这不是很好我们必须使用$file->getMimeType()
使用fileinfo
来获取mime类型并且会给出正确的哑剧类型。
在getMimeType()
中使用MimeTypeGuesser
实例猜测mime类型,该实例使用finfo()
,mime_content_type()
和系统二进制“文件”(按此顺序),具体取决于哪些可用。
感谢大家的帮助:D