我正在使用Symfony v3.4
我正在尝试从数据库中删除商品。为此,我使用方法described here。
简而言之 - 可以使用两个表单提交按钮来决定 - 删除项目或返回列表(通过重定向)。检查表单有效性后,可以知道按下了哪个按钮,以满足需求。 (这种方法不依赖于JavaScript)。
出于此目的,我使用2个不同名称的提交按钮创建了表单(没有其他字段)。我在控制器中创建表单并将其传递给视图,它按预期显示。
问题是 - 当我提交表格时,它无效!错误是:The CSRF token is invalid. Please try to resubmit the form.
重新提交不会更改内容,表单仍无效...
我认为Symfony Forms
会自动处理CSRF令牌!在这种情况下不是吗?我错过了什么?
在这种情况下,最佳做法是什么?
我的FormType
namespace AppBundle\Form\Type;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolver;
class ReservationActionType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('delete', SubmitType::class, array(
'attr' => array('class' => 'btn btn-danger'),
'label' => 'Yes, delete an item'
))
->add('return', SubmitType::class, array(
'attr' => array('class' => 'btn btn-default'),
'label' => 'No, return to list'
));
;
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(
array(
'csrf_protection' => true,
'csrf_field_name' => '_token',
)
);
}
}
我的Controller
public function reservationDelete($reservation_id = 0, Request $request)
{
// get $reservation that corresponds $reservation_id from database
$form = $this->createForm(ReservationActionType::class);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid())
{
if ($form->get('delete')->isClicked())
{
dump('replace this with item delete code');
}
else if ($form->get('return')->isClicked())
{
return $this->redirectToRoute('admin_reservations');
}
}
return $this->render('admin/reservations_use_code.html.twig',
[
'reservation_id' => $reservation_id,
'reservation' => $reservation,
'form' => $form->createView()
]
);
}
表单所在的template
部分
{% if reservation['id'] is defined %}
{{ form_start(form, {'attr': {'novalidate': 'novalidate', 'method': 'POST'}}) }}
{{ form_widget(form.delete) }}
{{ form_widget(form.return) }}
{{ form_end(form) }}
{% endif %}
答案 0 :(得分:0)
尝试直接在表单中添加CSRF:
...
{{ form_row(form._token) }}
{{ form_end(form) }}
答案 1 :(得分:0)
实际上,是的,它可以!
(渲染形式的差异很小,特定于Twig,而不是Symfony)
但是你呈现表单的方式,实际上(我猜错了)你说twig 不呈现其他字段而不是"删除"并且"返回"。这就是找不到您的CSRF令牌的原因。
我建议您不要手动渲染字段,而是建议您使用主题(https://symfony.com/doc/3.4/form/form_customization.html)来自定义特定的行和字段。
然后您可以通过以下方式呈现表单:
{{ form(form) }}
要为表单附加费用,自动呈现所有字段(包括CSRF):注意form_widget(form)
(呈现所有字段)和form_widget(form.my_field)
之间的差异(渲染" my_field"仅限)
{{ form_start(form, { attr: { novalidate: true }}) }}
{{ form_widget(form) }}
{{ form_end(form) }}
要在不使用主题的情况下对特定字段进行附加费(写入IMO要稍微重一点),请使用form_rest(form)
呈现所有其他字段(包括CSRF)
{{ form_start(form) }}
{{ form_row(form.delete, { attr: { class: 'whatever' }}) }}
{{ form_rest(form) }}
{{ form_end(form) }}