如何处理 Symfony/PHP 中错误的 CSV 文件导入

问题描述 投票:0回答:2

我正在从 csv 上传大量数据,没有问题,但是我想保护可能的用户错误,从某种意义上说,他们可能在表单中放入了错误的 csv ...

问题是每次启动表单时我都会截断表......


ImportController.php

       $form = $this->createFormBuilder()
       ->add('form', FileType::class, [
           'attr' => ['accept' => '.csv',
           'class' => 'custom-file-input'],
           'label' => 'Import'
       ])
       ->getForm();
       
       $form->handleRequest($request);

           if ($form->isSubmitted() && $form->isValid()) 
           {
               /** 
                * @var UploadedFile
                */
               $file = $form->get('form')->getData();  
               $connection = $em->getConnection();
               $platform = $connection->getDatabasePlatform();
               $connection->beginTransaction();
               $connection->executeQuery($platform->getTruncateTableSQL('MyTable', true));
               $this->getDoctrine()->getManager()->getRepository(MyTable::class)->importMyData($file);           
       
               $this->addFlash('success',"The csv has been successfully imported");
               return $this->redirectToRoute('import');
           } 


MyTableRepository.php

public function importMyData($file)
    {
        $em = $this->entityManager;

        if (($handle = fopen($file->getPathname(), "r")) !== false) 
        {
            $count = 0;
            $batchSize = 1000;
            $data = fgetcsv($handle, 0, ","); 

            while (($data = fgetcsv($handle, 0, ",")) !== false) 
            {
                $count++;
                $entity = new MyTable();

                // 40 entity fields...
                $entity->setFieldOne($data[0]);                
                $entity->setFieldTwo($data[1]); 
                //....
                $entity->setFieldForty($data[39]); 

                $em->persist($entity);

                if (($count % $batchSize) === 0 )
                {
                    $em->flush();
                    $em->clear();
                }
            }
            fclose($handle);
            $em->flush();
            $em->clear();
        }
    }

我只是希望当启动错误的 CSV 文件时该表不会被截断

php csv symfony truncate
2个回答
0
投票

您可以在插入数据库之前清理和验证数据。您可以通过多种方式进行尝试。首先,我会使用 Symfony 文件验证器来确保检查已上传的有效文件。

Symfony 文件验证器

要验证每一行,您可以使用自定义回调验证器或数组的原始值 验证原始数据

//call the validator in the repository
$validator = Validation::createValidator();

// sample input
$input = [
'field_1' => 'hello',
'field_2' => '[email protected]',
'field_40' => 3
];

//define the constraints for each row
$constraint = new Assert\Collection([
// the keys correspond to the keys in the input array
'field_1' => new Assert\Collection([
    'first_name' => new Assert\Length(['min' => 101]),
    'last_name' => new Assert\Length(['min' => 1]),
]),
'field_2' => new Assert\Email(),
'field_40' => new Assert\Length(['min' => 102])
]);

while (($data = fgetcsv($handle, 0, ",")) !== false) {
  $violations = $validator->validate($data, $constraint);
  
  // you can skip the row or log the error
  if ($violations->count() > 0) {
    continue;
  }
}

0
投票

我只是放置了一个“try catch”,如果 importMyData 函数中发生错误,它会返回其他内容,然后我检查我的控制器是否返回了该值...如果是这样,我会重定向等。

© www.soinside.com 2019 - 2024. All rights reserved.