我正在使用带有StofDoctrineExtensionsBundle的DoctrineExtensions来获得软删除行为。
它在我的应用程序的前端运行良好。
在后端,我需要"硬"删除实体的选项。
我已经在我的管理控制器中禁用了过滤器(我使用SonataAdmin):
$filters = $this->getModelManager()->getEntityManager($this->getClass())->getFilters();
if (array_key_exists('softdeleteable', $filters->getEnabledFilters())) {
$filters->disable('softdeleteable');
}
这有效(软删除的实体显示在列表中),但是当我尝试删除它时,该实体确实会再次被软删除。如何强制"硬"删除?
您不需要禁用过滤器 - 它仅用于过滤选择的记录。您必须改为禁用侦听器:
// $em is your EntityManager
foreach ($em->getEventManager()->getListeners() as $eventName => $listeners) {
foreach ($listeners as $listener) {
if ($listener instanceof GedmoSoftDeleteableSoftDeleteableListener) {
$em->getEventManager()->removeEventListener($eventName, $listener);
}
}
}
然后致电
$em->remove($entity);
$em->flush();
无需创建侦听器或任何要硬删除的内容,并启用了软删除。
原始软删除事件具有以下行:
$reflProp = $meta->getReflectionProperty($config['fieldName']);
$oldValue = $reflProp->getValue($object);
if ($oldValue instanceof Datetime) {
continue; // want to hard delete
}
所有这些都意味着,如果您:
$entity->setDeletedAt(new Datetime());
$em->flush();
然后:
$em->remove($entity);
$em->flush();
届时,它将很难删除。
如果您在 ->remove($entity) 后调用 ->flush() 时在 deletedAt 字段中有一个有效日期,您的实体将被硬删除
不是最优雅的方式:你总是可以用SQL做一个真正的删除,它会绕过softdeletable
$em->createQuery("DELETE MyEntity e WHERE e = :et")->setParameter('et',$entity)->execute();
虽然这个问题有点旧,也许它对某人有用:
创建自己的事件侦听器可能是更好的解决方案:
class SoftDeleteableListener extends BaseSoftDeleteableListener
{
/**
* @inheritdoc
*/
public function onFlush(EventArgs $args)
{
$ea = $this->getEventAdapter($args);
$om = $ea->getObjectManager();
//return from event listener if you disabled filter: $em->getFilters()->disable('softdeleteable');
if (!$om->getFilters()->isEnabled('softdeleteable')) {
return;
}
parent::onFlush($args);
}
}
并添加您的配置:
gedmo.listener.softdeleteable:
class: AppBundleEventListenerSoftDeleteableListener
tags:
- { name: doctrine.event_subscriber, connection: default }
calls:
- [ setAnnotationReader, [ @annotation_reader ] ]
来源: https://github.com/Atlantic18/DoctrineExtensions/issues/1175