我正在尝试在symfony 3.4中使用lock组件,就像在 https://symfony.com/doc/3.4/components/lock.html
我想防止来自不同用户的多个数据更改。
例如,user1调用带有数据的同一公司表格,然后user2
我如何告诉user2,编辑数据已被user1(包括用户名)阻止?
更新: 它用于后端,许多员工在编辑客户,订单等数据。 此表格仅供编辑。这意味着,如果他们想更新一些数据,则单击“编辑”。在将数据加载到表单之前,应通知他们另一位员工何时更改此记录。员工有时需要花费一些时间来更改所有内容。如果员工在保存消息时收到消息,他们必须返回,重新加载数据,然后重新开始。
我的控制器中的一个示例:
public function CompanyEdit(Request $request)
{
$error = null;
$company_id = $request->get('id');
// if (!preg_match('/^\d+$/', $company_id)){
// return $this->showError();
// }
$store = new SemaphoreStore();
$factory = new Factory($store);
$lock = $factory->createLock('company-edit-'.$company_id, 30);
if(!$lock->acquire()) {
//send output with username
// this data is locked by user xy
return 0;
}
$company = $this->getDoctrine()->getRepository(Company::class)->find($company_id);
$payment = $this->getDoctrine()->getRepository(Companypay::class)->findOneBy(array('company_id' => $company_id));
$form = $this->createFormBuilder()
->add('company', CompanyFormType::class, array(
'data_class' => 'AppBundle\Entity\Company',
'data' => $company
))
->add('payment', CompanyPayFormType::class, array(
'data_class' => 'AppBundle\Entity\CompanyPay',
'data' => $payment
))
->getForm();
$form->handleRequest($request);
$company = $form->get('company')->getData();
$payment = $form->get('payment')->getData();
if ($form->isSubmitted() && $form->isValid()) {
$event = new FormEvent($form, $request);
if ($payment->getCompanyId() == null) {
$payment->setCompanyId($company->getId());
}
try {
$this->getDoctrine()->getManager()->persist($company);
$this->getDoctrine()->getManager()->persist($payment);
$this->getDoctrine()->getManager()->flush();
$this->container->get('app.logging')->write('Kundendaten geändert', $company->getId());
} catch (PDOException $e) {
$error = $e->getMessage();
}
if (null === $response = $event->getResponse()) {
return $this->render('customer/edit.html.twig', [
'form' => $form->createView(),
'company' => $company,
'error' => $error,
'success' => true
]);
}
$lock->release();
return $response;
}
答案 0 :(得分:1)
您不能(锁不能有任何元数据),但是您可能一开始就不希望这样做。
在这种情况下,您可以在用户打开编辑页面时创建一个锁,并在用户提交表单时释放它。但是,如果用户打开页面而不提交表单怎么办?用户为什么甚至不能查看表单?
这看起来像XY-problem。我认为您正在尝试阻止用户在不知情的情况下覆盖数据。相反,您可以将时间戳或哈希添加到更改实体后更改的表单。例如:
<form type="hidden" name="updatedAt" value="{{ company.updatedAt()|date('U') }}" />
并以您的形式:
<?php
if ($company->getUpdatedAt()->format('U') !== $form->get('updatedAt')->getData()) {
throw new \LogicException('The entity has been changed after you opened the page');
}
免责声明:代码未经测试,仅作为示例说明此解决方案的外观。