Symfony2 没有实体的简单文件上传编辑

Symfony2 simple file upload edit without entity

请帮帮我,因为我不敢相信自己的眼睛。

我拒绝使用某些第 3 方插件进行文件上传,并拒绝为 file/document 创建单独的实体。我只想要在 Zend/Laravel 等中执行的简单文件上传。

我有一张发票 table,最后一个 collumb 名称 "attachment",我想在这里存储它经过清理的名称(例如:123421_filename.jpg),添加表格和上传都很顺利.

代码在这里:

//AddAction
     $file=$form['attachment']->getData();
        $fileName=$file->getClientOriginalName();
        $sanitizedFilename=rand(1, 99999).'_'.$fileName;
        $dir='files/'.$userId.'/';
        $file->move($dir, $sanitizedFilename);
    ...
        $invoice->setAttachment($sanitizedFilename);
        $em = $this->getDoctrine()->getManager();
        $em->persist($invoice);

我遇到但不知道如何解决的第一个问题是使用编辑表单。我有一个 formbuilder

   $builder->add('attachment', 'file',array('data_class'=>null))

因为我的文件没有 "Object" 并且因为在我的发票 table 中我将名称存储为字符串,所以这里出现错误,我需要强制 data_class => null .. 这很糟糕,因为如果我在前端为上传字段编辑发票,我会得到 NULL,而不是链接到发票的当前文件的文件名。

有:

$builder->add('attachment', 'text')

我没有得到文件输入,只是一个愚蠢的文本框,但这次 -- 里面有名字 -- .. 那么我该如何解决这个问题?没有文档对象的文件输入小部件和其中的文件名?!

第二个问题(我只是想放弃到目前为止开发的所有应用程序并转向 laravel 因为我有很多 "these kind of problems" ..在symfony 几乎总是比其他框架更复杂..或者我不想遵循指南并创建文档实体可能是我的错?!如果我想要灵活性并想以另一种方式管理我的数据库怎么办不符合所有这些准则?)

所以我正在进行编辑表单操作,我上传了一个新文件(不要忘记我将文件设置为 $builder->add('attachment',file',array('data_class'=>空))

我无法从我正在编辑的当前发票中获取附件字符串名称? !

public function editAction($id)
   ....
   $invoice = $em->getRepository('AppBundle:Invoice')->find($id);
   ..
    if ($form->isValid()) {

            $oldFileName=$invoice->getAttachment();  //this is null
            $oldFileName=$invoice->getId();   //this returns the invoice id
            $oldFileName=$invoice->getValue(); //returns invoice value

                echo 'old: '.$oldFileName.'<br/>';
                exit();
     }

所以请有人告诉我为什么我无法访问我的发票 属性?哪个是字符串?

我尝试创建一个新实例,但不知何故,如果我使用 $invoice 对象创建表单,它会以某种方式将他链接到编辑表单中的附件

 if ($form->isValid()) {
    $sameInvoice= $em->getRepository('AppBundle:Invoice')->find(20); //hardcoded ID

                   $oldFileName=$sameInvoice->getAttachment();   //still null
                   $oldFileName=$sameInvoice->getId();   //returns 20
                    echo 'old: '.$oldFileName.'<br/>';
                    exit();

我唯一想要的是,在我的发票中有一个文件名字符串 table,然后自己测试该文件是否存在于具有该文件名的路径中,如果存在,则将其删除并上传新的一个等等..为什么要这么难??

为什么我必须创建一个实体?为什么我必须改变我的数据库结构(这里不是这种情况..但是如果客户不希望更改数据库怎么办..所以我不能插入这个 "Document" table ).. 为什么表单生成器不能显示发票-> 附件中的数据,我知道是因为他需要文件数据并且他不能接受字符串,但为什么没有这些简单任务的指南? !

保持冷静。你让这件事变得更难了。

myControllerAction()
{
    // No need for an object, an array works fine
    $model = array(
        'invoice' => $invoice,
        'attachment' => null
    );

    $builder = $this->createFormBuilder($model);

    $builder->setAction($this->generateUrl('cerad_game_schedule_import'));
    $builder->setMethod('POST');

    $builder->add('attachment', 'file');
    $builder->add('invoice', new InvoiceType());

    $builder->add('import', 'submit', array(
        'label' => 'Import From File',
        'attr' => array('class' => 'import'),
    ));        
    $form = $builder->getForm();

    $form->handleRequest($request);

    if ($form->isValid())
    {   
        $model = $form->getData();
        $file = $model['attachment']; // The file object is built by the form processor

        if (!$file->isValid())
        {
            $model['results'] = sprintf("Max file size %d %d Valid: %d, Error: %d<br />\n",
            $file->getMaxFilesize(), // Returns null?
            $file->getClientSize(),
            $file->isValid(), 
            $file->getError());
            die($model['results'];
        }
        $importFilePath = $file->getPathname();
        $clientFileName = $file->getClientOriginalName();

        // Xfer info to the invoice object and update

    }

好的,我终于成功了:

所以它为 NULL 的问题是因为它确实将 $invoice 实例绑定到表单:

$invoice = $em->getRepository('AppBundle:Invoice')->find($id);
$form = $this->createForm(new InvoiceType($id), $invoice);

在此代码块之后,所有引用如:$invoice->getAttachment();

被称为表单发票对象而不是实际对象,所以基本上在此之后我只能调用类似的东西:"getClientOriginalName()" 并会显示我在编辑表单中上传的新文件。

为了解决这个问题,我创建了一个变量,该变量最初在 $invoice 对象之前存储该名称以绑定到表单

    $invoice = $em->getRepository('AppBundle:Invoice')->find($id);
    $oldFileName=$invoice->getAttachment();  //this is stil the object here I can see the record from the database
    $form = $this->createForm(new InvoiceType($id), $invoice);
  //here the $invoice->getAttachment(); returns null

所以现在我有了旧名称,我可以像这样测试它:

 if ($form->isValid()) {

                $fs=new Filesystem();
                $newFile=$form['attachment']->getData();
                $newFileName=$newFile->getClientOriginalName();

                if($newFileName != $oldFileName)
                {
                  //  if($fs->exists('files/'.$this->getUser()->getId().'/'.$oldFileName))
                  //      $fs->remove('files/'.$this->getUser()->getId().'/'.$oldFileName);

                    $sanitizedFilename=rand(1, 99999).'_'.$newFileName;
                    $dir='files/'.$this->getUser()->getId().'/';
                    $newFile->move($dir, $sanitizedFilename);
                    $invoice->setAttachment($sanitizedFilename);
                }

                $em->persist($invoice);
                $em->flush();

                return $this->redirect($this->generateUrl('my-invoices'));
            }

对于旧文件的文件名未出现在“编辑发票”页面中的另一个问题,我将我的变量发送到视图:

return $this->render('AppBundle:Invoices:edit.html.twig', array(
            'oldFileName' => $oldFileName) 

并使用 twig

将该值添加到输入文件小部件的标签中