取消订单日志必须同步记录order_id、cancel_reason(校验枚举+code/text双字段)、operator_id(区分user_id/admin_id),且与订单状态更新置于同一PDO事务中,并为order_id及(operator_id, created_at)建立索引。
不记录 order_id、cancel_reason 和 operator_id,后续根本没法查清谁在什么时间因何原因取消了哪笔订单。尤其 cancel_reason 不能只存前端传来的字符串——得先校验是否在预设枚举里(如 'user_request'、'stock_shortage'、'fraud_risk'),否则容易被恶意注入或写入脏数据。
order_id 必须与订单主表一致,建议用数据库外键约束或事务内二次查询确认存在cancel_reason 建议用整型字段存 reason_code,同时冗余一个 reason_text 字段存原始描述(便于审计)operator_id 要区分是用户主动取消(填 user_id),还是客服后台操作(填 admin_id),不能统一写 0 或空常见错误是先更新订单表 status = 'cancelled',再单独 insert 日志表——万一 insert 失败,订单已变状态,日志却丢了,完全不可追溯。必须把两步包进同一 PDO::beginTransaction()。
$pdo->beginTransaction();
try {
$stmt = $pdo->prepare("UPDATE orders SET status = ?, updated_at = NOW() WHERE id = ? AND status = 'paid'");
$stmt->execute(['cancelled', $orderId]);
if ($stmt->rowCount() === 0) {
throw new Exception('Order not found or not in payable state');
}
$logStmt = $pdo->prepare("INSERT INTO order_logs (order_id, action, reason_code, reason_text, operator_id, created_at) VALUES (?, ?, ?, ?, ?, NOW())");
$logStmt->execute([$orderId, 'cancel', $reasonCode, $reasonText, $operatorId]);
$pdo->commit();} catch (Exception $e) {
$pdo->rollback();
throw $e;
}
日志表结构要支持快速按时间+订单号+操作人筛选
线上出问题时,DBA 最常跑的查询是:SELECT * FROM order_logs WHERE order_id = ? ORDER BY created_at DESC LIMIT 10 或 WHERE operator_id = ? AND created_at > DATE_SUB(NOW(), INTERVAL 1 HOUR)。没索引会直接拖垮慢查询。
order_id 单独建索引(operator_id, created_at) 能加速客服
自查操作流reason_text 上建全文索引——99% 的检索靠 reason_code 就够了有些团队为了“解耦”把日志写入扔给 Redis 队列再由消费者处理,结果消费者挂了、重试失败、消息堆积,日志就永远消失了。订单取消是强一致性操作,日志必须同步落库。异步只适合通知类动作(如发短信、推消息),不是日志。
如果真要异步,至少保证:日志先同步写入一张临时表(order_logs_buffer),再由定时任务捞取并转正——但这增加了复杂度,多数业务没必要。