级联验证不在表单的第三层上工作

m0c*_*m0c 12 php symfony-forms symfony symfony-2.1

我有一个3层的表单:First Layer是游戏的容器:

class GameListType extends AbstractType {

    public function buildForm(FormBuilderInterface $builder, array $options) {
        $builder->add('games', 'collection', array(
            'required' => false,
            'allow_add' => true,
            'prototype' => true,
            'by_reference' => false,
            'type' => new GameBetType(),
        ))
        ;
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver) {
        parent::setDefaultOptions($resolver);
        $resolver->setDefaults(array(
            'cascade_validation' => true,
        ));
    }  
}
Run Code Online (Sandbox Code Playgroud)

第二层是游戏本身,因为我不打算改变游戏,但是赌注,它只包括赌注的形式:

class GameBetType extends AbstractType {

    public function buildForm(FormBuilderInterface $builder, array $options) {   
        $builder->add('bet', new BetType());
    }

    public function setDefaultOptions(\Symfony\Component\OptionsResolver\OptionsResolverInterface $resolver) {
        parent::setDefaultOptions($resolver);
        $resolver->setDefaults(array(
            'data_class' => 'Strego\TippBundle\Entity\Game',
            'cascade_validation' => true,
        ));
    }  
}
Run Code Online (Sandbox Code Playgroud)

第三层是投注的形式:

class BetType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {

        $builder->add('scoreT1' , 'text')
                ->add('scoreT2' , 'text');
        ;
    }


        public function setDefaultOptions(\Symfony\Component\OptionsResolver\OptionsResolverInterface $resolver) {
        parent::setDefaultOptions($resolver);
        $resolver->setDefaults(array(
            'data_class' => 'Strego\TippBundle\Entity\Bet',
        ));
    }

}
Run Code Online (Sandbox Code Playgroud)

问题是,如果在第三级存在验证约束,则根表单始终有效,但如果我专门验证投注实体,则在控制器中获取正确的违规列表:

// Form processing
$form = $this->createForm(new GameListType(), $entity);
$request = $this->getRequest();

if ($request->getMethod() == 'POST') {
    $form->bind($request);
    $entity = $form->getData();

    if ($form->isValid()) {
        foreach ($entity->getGames() as $game) {                  
            if($game->hasBet()){
                $bet = $game->getBet();
                $bet->setUser($user);
                $validator = $this->container->get('validator');
                $errors = $validator->validate($bet);
                var_dump($errors)  //<-- i see there are errors
                if(count($errors) == 0){
                    print($bet. ' gets persisted<br>');
                    $em->persist($bet);
                } 
            }             
        }
    $em->flush();
    }
}
Run Code Online (Sandbox Code Playgroud)

另一个问题是即使我不调用$em->persist($bet)实体也会持久存在.我没有看到该行中描述的输出

print($bet.'gets persisted<br>'); 
Run Code Online (Sandbox Code Playgroud)

但表单输入仍然保留在数据库中.

所以我的两个问题:

  1. 如果验证失败,如果投注失败,我如何得到错误(我不希望整个表格无效,因为3个投注中只有1个无效).

  2. 即使我不打电话$em-persist($bet),为什么赌注仍然存在,这是绑定时发生的一些魔力吗?

Mun*_*Das 3

为了回答你的第二个问题,你可能已经声明了和实体之间的双向OneToMany关系,并设置在关系的一侧。因此实体管理器会自动标记实体对象以便在下一个操作中插入/更新。还将线路移出循环。当方法发出数据库查询时,每次代码进入循环时它都会连接到数据库。而如果在循环之外,实体将查询一次,并将所有更改的 sql 包装在单个事务中。UserBetcascade={"persist"}UserBetflush$em->flush();flush$em->flush();