如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

通过第一部分的介绍,我们对这个Audit Logging解决方案的原理有了较为全面的了解,接下来我们将通过一个简单的Sample来进一步介绍如何在一个具体的应用中如何实现这样一个Audit Logging。

我们沿用在Part I提出的Order management 场景,为了简单起见,我们通过一个Console application来模拟。在这个Application中,你将会看到如何以一种离线的模式操作数据库,如何Log这些操作。

我们首先来介绍Sample程序的结构(如下图)。整个Solution 一共包括两个Project,一个Console application,另一个是用以管理和定义Stored Procedure和Trigger的Database project。我们现在就来一步步实现这样一个Order management的简单的应用。Source Code从Artech.WCFService.zip下载。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
一.定义一个轻型的Data access 的Helper class。

在Part I中我提到过这样一个Helper class,虽然这不是本篇所要介绍的重点,但是为了让读者能够较为全面地了解整个处理流程,在这里我对她作一个简单的介绍。

我在前一阵子,写过一篇叫做[原创]我的ORM: 开发自己的Data Access Application Block 的文章,在这片文章中我开发了一个自定义的DAB。我这个Data access helper便是采用里面提出的思想,实现了其中一小部分功能:Data retrieval, Update Dataset和Transaction。力求简洁,我剔出掉其中可配置的data mapping部分采用hard coding的方式实现Dataset和Stored procedure的Mapping。

通过这个Helper class,你可以调用UpdateData方法把对一个Table作的修改向数据库提交。这个方法的思路是这样的:我们对该Table的增加、修改和删除均定义了一个Stored procedure,我们假设Table name和Stored procedure name之间,Stored procedure的每个Parameter 的名称和Table中的Field name和DataRowVersion存在一个Mapping。比如T_ORDER对应的增加、修改和删除stored procedure分别为sp_order_i,sp_order_u和sp_order_d;stored procedure的参数@p_order_id对应的Source column为ORDER_ID, SourceVersion为DataRowVersion.Current,而参数@o_order_id对应的SourceVersion为DataRowVersion.Original。有了这样一个Mapping为前提,相信大家对Helper class的实现原理应该想象得到。当然要使我们的Stored procedure满足这样一个Mapping,靠我们手工的方式来定义每个stored procedure在一个真正的application是不现实的,一般地这些都是通过我们根据具体的Mapping关系开发的生成器生成的。像本Sample的所有stored procedure和trigger也都是通过Generator生成的。

下面是整个Helper class的实现,不算太复杂,有兴趣的话可以看看。否则敬请略过。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Collections.Generic;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Text;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Configuration;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Data;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Data.Common;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Data.SqlClient;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]namespace Artech.AuditLogging.ConsoleApp
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]{
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    public class DataAccessHelper:IDisposable
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Private Fields
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Public Properties
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Transaction Operations
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Data Mapping
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Discovery Parameter
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        Public Methods
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        IDisposable Members
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]}

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

二.定义Dataset

我们根据数据库中Table的结构定义一个具有相同结构的strongly typed dataset:OrderDataSet.

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
为了保证数据的完整性,我们必须为两表的relation做出如下的设置:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
同时我们为Log的数据定义下面的一个strongly typed dataset:AuditLoggingDataSet。该Dataset中只包含一个Table: T_AUDIT_LOG。我们之所以没有定义T_AUDIT_LOG_DETAIL是因为T_AUDIT_LOG_DETAIL中的数据是通过trigger添加的,我们同过程序只需要在主表中添加总体信息就可以了。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
三、定义用于Audit log的helper class:AuditLoggingHelper

下面是所有AuditLoggingHelper所有的Code,很简单。我分别定义了一个public的property:AuditLoggingData。其类型为我们上面定义的strongly typed dataset:AuditLoggingDataSet。还定义了一个Public 的方法AuditLog向AuditLoggingData中添加Log信息,并返回一个Guid用以标识将要执行的transaction。我管这个Guid为Transaction no。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Collections.Generic;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Text;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]namespace Artech.AuditLogging.ConsoleApp
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]{
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    public class AuditLoggingHelper
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        private AuditLoggingDataSet _auditLoggingData;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        /// <summary>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
/// A strongly typed dataset to used to store the general auditoing inforamtion. 
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
/// </summary>

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        public AuditLoggingDataSet AuditLoggingData
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            get return _auditLoggingData; }
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            set { _auditLoggingData = value; }
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        /// <summary>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
/// Log the general auditoing information according with the current transaction.
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
/// </summary>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
/// <returns>A guid which identifies uniquely a transaction</returns>

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        public Guid AuditLog()
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            if (this._auditLoggingData == null)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                this._auditLoggingData = new AuditLoggingDataSet();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Guid transactionNo = Guid.NewGuid();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            AuditLoggingDataSet.T_AUDIT_LOGRow auditRow= this._auditLoggingData.T_AUDIT_LOG.NewT_AUDIT_LOGRow();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            auditRow.BeginEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            auditRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            //TODO: The user id is generally the account of the current login user.
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
            auditRow.USER_ID = "testUser";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            auditRow.OPERATION_DATE = DateTime.Now;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            auditRow.EndEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            this._auditLoggingData.T_AUDIT_LOG.AddT_AUDIT_LOGRow(auditRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            return transactionNo;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]}

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

四、定义stored procedure和trigger

为了较为真实地贴近我们现实的开发, 本Sample的所有Data access操作(除了data retrieval外)均采用stored procedure。通过所有需要进行Log的详细地信息都是通过Trigger来添加的。所有的stored procedure通过这里来查看,所有的trigger通过这里查看。

五:模拟Insert操作

我们先清空所有的Log数据,通过下面的方法添加一个新的Order。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Collections.Generic;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Text;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]using System.Data;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]namespace Artech.AuditLogging.ConsoleApp
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]{
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    class Program
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        static string USER_ID = "testUser";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        static void Main(string[] args)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            AddOrderData();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Console.WriteLine("Operation completes!");
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Console.Read();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        static void UpdateCommonField(DataRow row)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            row["LAST_UPDATED_BY"] = USER_ID;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            row["LAST_UPDATED_ON"] = DateTime.Now;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            if (row.RowState == DataRowState.Detached || row.RowState == DataRowState.Added)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                row["CREATED_BY"] = USER_ID;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                row["CREATED_ON"] = DateTime.Now;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }
  
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        static void AddOrderData()
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            AuditLoggingHelper auditLoggingHelper = new AuditLoggingHelper();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Guid transactionNo = auditLoggingHelper.AuditLog();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            //Add an order item
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
            OrderDataSet orderData = new OrderDataSet();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            OrderDataSet.T_ORDERRow orderRow = orderData.T_ORDER.NewT_ORDERRow();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.BeginEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.ORDER_DATE = DateTime.Today;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.SUPPLIER = "HP Corporation";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            UpdateCommonField(orderRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.EndEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderData.T_ORDER.AddT_ORDERRow(orderRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            //Add two order detail items.
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
            OrderDataSet.T_ORDER_DETAILRow orderDetailRow = orderData.T_ORDER_DETAIL.NewT_ORDER_DETAILRow();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.BeginEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.ORDER_ID = orderRow.ORDER_ID;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_ID = 1;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_NAME = "HP Printer";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.UNIT_PRICE = 3000;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.QUANTITY = 2;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            UpdateCommonField(orderDetailRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.EndEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderData.T_ORDER_DETAIL.AddT_ORDER_DETAILRow(orderDetailRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow = orderData.T_ORDER_DETAIL.NewT_ORDER_DETAILRow();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.BeginEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.ORDER_ID = orderRow.ORDER_ID;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_ID = 2;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_NAME = "HP PC";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.UNIT_PRICE = 3400;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.QUANTITY = 22;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            UpdateCommonField(orderDetailRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.EndEdit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderData.T_ORDER_DETAIL.AddT_ORDER_DETAILRow(orderDetailRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            using (DataAccessHelper dataAccessHelper = new DataAccessHelper())
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                dataAccessHelper.BeginTransaction();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                try
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(auditLoggingHelper.AuditLoggingData.T_AUDIT_LOG);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER_DETAIL);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Commit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                catch(Exception ex)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Rollback();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    Console.WriteLine(ex.Message);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]}

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

我们来看看T_AUDIT_LOG的记录:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
T_AUDIT_LOG_DETAIL的记录

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
DATA_CHANGE的data分别为下面3段XML。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <after order_id="33" order_date="Apr 23 2007 12:00AM" supplier="HP Corporation" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <after order_id="33" product_id="1" product_name="HP Printer" unit_price="3000.00" quantity="2" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <after order_id="33" product_id="2" product_name="HP PC" unit_price="3400.00" quantity="22" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

六、模拟Update操作

定义新的方法UpdateOrderData修改我们刚刚添加的Order记录:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]    static void UpdateOrderData()
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            OrderDataSet orderData = GetAllOrderData();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            AuditLoggingHelper auditLoggingHelper = new AuditLoggingHelper();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Guid transactionNo = auditLoggingHelper.AuditLog();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            OrderDataSet.T_ORDERRow orderRow = orderData.T_ORDER[0];
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.ORDER_DATE = new DateTime(2005, 1, 1);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.SUPPLIER = "Dell Corporation";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.TRANSACTION_NO  = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            UpdateCommonField(orderRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            OrderDataSet.T_ORDER_DETAILRow orderDetailRow = orderData.T_ORDER_DETAIL[0];
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_ID = 3;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.PRODUCT_NAME = "Workstation";
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.UNIT_PRICE = 10000;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.QUANTITY = 1;
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.TRANSACTION_NO  = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            UpdateCommonField(orderDetailRow);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            using (DataAccessHelper dataAccessHelper = new DataAccessHelper())
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            {                
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                dataAccessHelper.BeginTransaction();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                try
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(auditLoggingHelper.AuditLoggingData.T_AUDIT_LOG);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER_DETAIL);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Commit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                catch (Exception ex)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Rollback();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    Console.WriteLine(ex.Message);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            }
            
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

执行上面的方法,然后查看我们Log表的记录。

 如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]


如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
两条Update记录对应的DATA_CHANGE字段的XML分别为:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <before order_id="33" order_date="Apr 23 2007 12:00AM" supplier="HP Corporation" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <after order_id="33" order_date="Jan  1 2005 12:00AM" supplier="Dell Corporation" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <before order_id="33" product_id="1" product_name="HP Printer" unit_price="3000.00" quantity="2" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <after order_id="33" product_id="1" product_name="Workstation" unit_price="10000.00" quantity="1" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

七、模拟Delete操作

定于DeleteOrderData方法delete掉我们添加的Order记录:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]static void DeleteOrderData()
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            OrderDataSet orderData = GetAllOrderData();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            AuditLoggingHelper auditLoggingHelper = new AuditLoggingHelper();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            Guid transactionNo = auditLoggingHelper.AuditLog();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            //Delete two order detail items.
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
            OrderDataSet.T_ORDER_DETAILRow orderDetailRow = orderData.T_ORDER_DETAIL[0];
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.AcceptChanges();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.Delete();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow = orderData.T_ORDER_DETAIL[1];
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.AcceptChanges();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderDetailRow.Delete();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            //Delete the order item.
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
            OrderDataSet.T_ORDERRow orderRow = orderData.T_ORDER[0];
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.TRANSACTION_NO = transactionNo.ToString();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.AcceptChanges();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            orderRow.Delete();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            using (DataAccessHelper dataAccessHelper = new DataAccessHelper())
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                dataAccessHelper.BeginTransaction();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                try
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(auditLoggingHelper.AuditLoggingData.T_AUDIT_LOG);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER_DETAIL);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.UpdateData(orderData.T_ORDER);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Commit();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                catch (Exception ex)
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                {
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    dataAccessHelper.Rollback();
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                    Console.WriteLine(ex.Message);
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]                }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]            }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]        }

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

执行上面的方法,然后查看我们Log表的记录。

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]
三条Delete记录对应的DATA_CHANGE字段的XML分别为:

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <before order_id="33" product_id="1" product_name="Workstation" unit_price="10000.00" quantity="1" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <before order_id="33" product_id="2" product_name="HP PC" unit_price="3400.00" quantity="22" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]<dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]  <before order_id="33" order_date="Jan  1 2005 12:00AM" supplier="Dell Corporation" />
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]</dataChange>
如何追踪每一笔记录的来龙去脉:一个完整的Audit Logging解决方案[下篇]

作者:蒋金楠
微信公众账号:大内老A
微博:www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号蒋金楠的自媒体将会停用)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。