DoctrineExtensions 实现全局日志记录与选择性忽略的实践
2025-06-16 14:20:26作者:柏廷章Berta
背景介绍
在基于Symfony框架和Doctrine ORM的开发中,Gedmo的DoctrineExtensions提供了Loggable行为,能够自动记录实体对象的变更历史。标准的Loggable实现需要显式地为每个需要记录的实体添加注解或属性标记。但在实际业务场景中,我们往往希望记录所有实体的变更,只排除少数不需要记录的实体或字段。
解决方案设计
通过继承LoggableListener并重写关键方法,我们可以实现以下功能:
- 默认记录所有实体变更
- 通过NotLoggable属性标记排除特定实体或字段
- 增强日志记录功能,包括对象标题、变更前后数据等
核心实现代码
配置扩展
首先需要在Symfony的配置文件中替换默认的LoggableListener:
services:
Gedmo\Loggable\LoggableListener:
class: App\EventSubscriber\Loggable
calls:
- [ setSecurity, ['@security.helper'] ]
tags:
- { name: doctrine.event_listener, event: 'onFlush' }
- { name: doctrine.event_listener, event: 'loadClassMetadata' }
- { name: doctrine.event_listener, event: 'postPersist' }
自定义Loggable监听器
创建自定义的Loggable监听器类,继承并扩展原始功能:
class Loggable extends LoggableListener
{
private Security $security;
private string $lastAction;
private array $lastChanges;
public function setSecurity(Security $security)
{
$this->security = $security;
}
// 关键方法:获取实体配置
public function getConfiguration(ObjectManager $objectManager, $class): array
{
$config = parent::getConfiguration($objectManager, $class);
$meta = $objectManager->getClassMetadata($class);
// 检查类级别NotLoggable属性
if ($meta->reflClass->getAttributes(NotLoggable::class)) {
return $config;
}
// 过滤掉带有NotLoggable属性的字段
$versioned = array_filter(
array_merge($meta->getFieldNames(), $meta->getAssociationNames()),
fn ($field) => !$meta->reflFields[$field]->getAttributes(NotLoggable::class)
);
// 设置默认记录配置
$config = array_merge($config, [
'loggable' => true,
'logEntryClass' => AuditLogEntry::class,
'useObjectClass' => $class,
'versioned' => $versioned,
]);
self::$configurations[$this->name][$meta->getName()] = $config;
return $config;
}
// 日志条目预处理
protected function prePersistLogEntry($logEntry, $object): void
{
if (method_exists($object, '__toString')) {
$logEntry->setObjectTitle((string)$object);
}
if (LogEntryInterface::ACTION_UPDATE !== $this->lastAction) {
return;
}
// 记录变更前数据
$prevData = [];
foreach ($this->lastChanges as $field => $changes) {
$value = $changes[0];
if (is_object($value) && method_exists($value, 'getId')) {
$value = ['id' => $value->getId()];
}
$prevData[$field] = $value;
}
$logEntry->setPrevData($prevData);
}
// 创建日志条目
protected function createLogEntry($action, $object, LoggableAdapter $ea): ?LogEntryInterface
{
$user = $this->security->getUser();
if ($user) {
$this->setUsername($user->getFullNameShort());
}
$this->lastAction = $action;
$om = $ea->getObjectManager();
$this->lastChanges = $ea->getObjectChangeSet($om->getUnitOfWork(), $object);
return parent::createLogEntry($action, $object, $ea);
}
}
使用示例
排除整个实体
#[\NotLoggable]
#[ORM\Entity(repositoryClass: RepairStatusLogRepository::class)]
class RepairStatusLog
{
// 实体定义
}
排除特定字段
#[ORM\Entity]
class User
{
#[ORM\Column]
private string $name;
#[\NotLoggable]
#[ORM\Column]
private string $password;
}
技术要点解析
-
配置覆盖机制:通过重写getConfiguration方法,实现了默认记录所有实体的逻辑,同时保留通过NotLoggable属性排除特定实体或字段的能力。
-
变更数据捕获:在createLogEntry方法中捕获变更数据集,并在prePersistLogEntry中将变更前数据格式化后存入日志条目。
-
用户上下文集成:通过Security组件自动获取当前用户信息并记录到日志中。
-
对象标识处理:对于关联对象,自动记录其ID而非整个对象,避免数据冗余。
实际应用价值
这种实现方式特别适合以下场景:
- 需要全面审计的系统,如金融、医疗等合规要求高的领域
- 开发初期阶段,尚未完全确定哪些实体需要记录
- 需要细粒度控制记录字段的系统
相比标准的Loggable实现,这种"记录所有,排除例外"的模式可以显著减少配置工作量,同时保证系统的审计完整性。
注意事项
- 性能考虑:全量记录可能对高并发系统产生性能影响,需合理设计日志存储方案
- 存储规划:日志数据可能快速增长,需要考虑归档策略
- 敏感数据处理:确保排除密码等敏感字段的记录
- 关联对象处理:对于复杂对象图,可能需要定制更精细的记录策略
通过这种定制化的Loggable实现,开发者可以在保证系统可审计性的同时,保持代码的简洁性和可维护性。
登录后查看全文
热门项目推荐
相关项目推荐
atomcodeClaude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed. Get StartedRust0153- DDeepSeek-V4-ProDeepSeek-V4-Pro(总参数 1.6 万亿,激活 49B)面向复杂推理和高级编程任务,在代码竞赛、数学推理、Agent 工作流等场景表现优异,性能接近国际前沿闭源模型。Python00
LongCat-Video-Avatar-1.5最新开源LongCat-Video-Avatar 1.5 版本,这是一款经过升级的开源框架,专注于音频驱动人物视频生成的极致实证优化与生产级就绪能力。该版本在 LongCat-Video 基础模型之上构建,可生成高度稳定的商用级虚拟人视频,支持音频-文本转视频(AT2V)、音频-文本-图像转视频(ATI2V)以及视频续播等原生任务,并能无缝兼容单流与多流音频输入。00
auto-devAutoDev 是一个 AI 驱动的辅助编程插件。AutoDev 支持一键生成测试、代码、提交信息等,还能够与您的需求管理系统(例如Jira、Trello、Github Issue 等)直接对接。 在IDE 中,您只需简单点击,AutoDev 会根据您的需求自动为您生成代码。Kotlin03
Intern-S2-PreviewIntern-S2-Preview,这是一款高效的350亿参数科学多模态基础模型。除了常规的参数与数据规模扩展外,Intern-S2-Preview探索了任务扩展:通过提升科学任务的难度、多样性与覆盖范围,进一步释放模型能力。Python00
skillhubopenJiuwen 生态的 Skill 托管与分发开源方案,支持自建与可选 ClawHub 兼容。Python0112
热门内容推荐
最新内容推荐
项目优选
收起
暂无描述
Dockerfile
733
4.76 K
deepin linux kernel
C
31
16
Ascend Extension for PyTorch
Python
652
797
Claude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed.
Get Started
Rust
1.25 K
153
旨在打造算法先进、性能卓越、高效敏捷、安全可靠的密码套件,通过轻量级、可剪裁的软件技术架构满足各行业不同场景的多样化要求,让密码技术应用更简单,同时探索后量子等先进算法创新实践,构建密码前沿技术底座!
C
1.1 K
611
本项目是CANN提供的数学类基础计算算子库,实现网络在NPU上加速计算。
C++
1.01 K
1.01 K
华为昇腾面向大规模分布式训练的多模态大模型套件,支撑多模态生成、多模态理解。
Python
147
237
昇腾LLM分布式训练框架
Python
168
200
openEuler内核是openEuler操作系统的核心,既是系统性能与稳定性的基石,也是连接处理器、设备与服务的桥梁。
C
434
395
暂无简介
Dart
987
253