您不一定需要修改验证规则,您可以选择使用实现\Psr\Http\Message\UploadedFileInterface 的对象。 CakePHP 的默认上传文件验证支持此类对象。
CakePHP 需要zendframework/zend-diactoros,因此您可以使用\Zend\Diactoros\UploadedFile 并在您的测试中执行以下操作:
$data = [
// ...
'file' => new \Zend\Diactoros\UploadedFile([
'/path/to/the/temporary/file/on/disk',
1234, // filesize in bytes
\UPLOAD_ERR_OK, // upload (error) status
'filename.jpg', // upload filename
'image/jpeg' // upload mime type
])
];
uploadedFile 规则会自动将此类对象视为上传的文件。
当然,您处理文件上传的代码也必须支持该接口,但这并不复杂,您只需确保将常规文件上传数组转换为UploadedFileInterface 实现,以便您的上传处理程序可以把它作为一个要求。
它当然可以在上传处理程序本身中完成,因此验证将使用常规文件上传数组以及UploadedFile 对象。另一种方法是在创建实体时更早地转换它们,使用 beforeMarshal 处理程序/事件,类似于以下内容:
public function beforeMarshal(\Cake\Event\Event $event, \ArrayObject $data, \ArrayObject $options)
{
$file = \Cake\Utility\Hash::get($data, 'file');
if ($file === null) {
return;
}
if (!($file instanceof \Psr\Http\Message\UploadedFileInterface)) {
if (!is_uploaded_file(\Cake\Utility\Hash::get($file, 'tmp_name'))) {
$file = new \Zend\Diactoros\UploadedFile(
null,
0,
UPLOAD_ERR_NO_FILE,
null,
null
);
} else {
$file = new \Zend\Diactoros\UploadedFile(
\Cake\Utility\Hash::get($file, 'tmp_name'),
\Cake\Utility\Hash::get($file, 'size'),
\Cake\Utility\Hash::get($file, 'error'),
\Cake\Utility\Hash::get($file, 'name'),
\Cake\Utility\Hash::get($file, 'type')
);
}
$data['file'] = $file;
}
}
这会将数据转换为UploadedFile 对象,以防它是实际上传的文件。添加此额外检查是因为 CakePHP 将文件数据与 POST 数据合并的行为,使得无法确定(除非可以访问请求对象或 $_FILES 超全局)来确定用户是否发布了该数据,或者 PHP 是否生成了该数据实际文件上传的数据。
如果您随后使用\Psr\Http\Message\UploadedFileInterface::moveTo() 移动文件,它将在 SAPI(基于浏览器)以及非 SAPI (CLI) 环境中工作:
try {
$file->moveTo($targetPath);
} catch (\Exception $exception) {
$entity->setError(
'file', [__('The file could not be moved to its destination.')]
);
}
另见