`

WF中的跟踪服务(2):使用SqlTrackingService

阅读更多

WF提供了可插拔的跟踪服务,运行时引擎可以在其生存期过程中添加多个运行时服务,因此可以同时启用多个不同类型的跟踪服务。WF框架提供了一个现成的SqlTrackingService跟踪服务,该服务将可配置的跟踪信息写到Sql Server数据库中,下面我们来看看如何使用SqlTrackingService跟踪服务。

跟踪事件类型

工作流运行时在执行工作流时会引发某些事件。 跟踪服务通过捕获这些事件来记录跟踪信息,工作流可以引发以下三种事件:
1.工作流事件(workflow Events):工作流事件(workflow events)表示工作流状态改变的事件,每次工作流状态的改变都会引发跟踪服务中的跟踪事件,工作流事件定义在TrackingWorkflowEvent枚举中,包括以下值:Created ,Completed,Idle,Suspended,Resumed,Persisted,Unloaded,Loaded,Exception,Terminated,Aborted,Changed,Started。

2.活动事件(Activity Events):活动事件表示活动状态的改变,活动的状态被定义在ActivityExecutionStatus枚举中,如下:Initialized,Executing,Canceling,Closed,Compensating,Faulting

3.用户事件(User Events): 在工作流事件(workflow events)和活动事件(Activity Events)中我们只能跟踪已经定义好的事件,有的时候我们需要工作流中特定位置的信息,此时,我们就可以使用自定义跟踪点,用户跟踪事件可以在整个工作流生命周期的任何位置。我们使用Activity的TrackData()方法来创建用户跟踪点。

SqlTrackingService跟踪服务应用举例

1.首先我们要建立跟踪数据,建立数据库的脚本在C:\WINDOWS\Microsoft.NET\Framework\v3.0\Windows Workflow Foundation\SQL\ZH-CHS目录中,Tracking_Logic.sql和Tracking_Schema.sql两个文件。建立完成的跟踪数据库包含表,视图和一些存储过程,详细信息,请查看WF中的跟踪服务(1):Sql跟踪数据库表,视图,存储过程等相关说明这文。

2.跟踪数据库建立完成后我们新建一个顺序工作流控制台程序,在工作流设计器中我们只要拖入一个CodeActivity活动即可,工作流的代码如下:

namespace CarySqlTrackingService
{
    public sealed partial class CarySqlTrackingServiceWorkflow: SequentialWorkflowActivity
    {
        public CarySqlTrackingServiceWorkflow()
        {
            InitializeComponent();
        }

        private void codeActivity1_ExecuteCode(object sender, EventArgs e)
        {
            Console.WriteLine("CodeActivity执行了哦!");
         }
    }
}

3.在宿主程序中,我们给workflow runtime添加SqlTrackingService跟踪服务,代码如下:
namespace CarySqlTrackingService
{
   class Program
   {
      private static String strConn = String.Format("Initial Catalog={0};Data Source={1};
Integrated Security={2};"
,"WorkflowTracking", @"localhost\SQLEXPRESS", "SSPI"); static void Main(string[] args) { using(WorkflowRuntime workflowRuntime = new WorkflowRuntime()) { SqlTrackingService sts = new SqlTrackingService(strConn); AutoResetEvent waitHandle = new AutoResetEvent(false); workflowRuntime.WorkflowCompleted += delegate(object sender,
WorkflowCompletedEventArgs e) {waitHandle.Set();}; workflowRuntime.WorkflowTerminated += delegate(object sender,
WorkflowTerminatedEventArgs e) { Console.WriteLine(e.Exception.Message); waitHandle.Set(); }; workflowRuntime.AddService(sts); WorkflowInstance instance = workflowRuntime.CreateWorkflow(typeof
(CarySqlTrackingService.CarySqlTrackingServiceWorkflow)); Console.WriteLine("---工作流执行开始---"); instance.Start(); waitHandle.WaitOne(); Console.WriteLine("---工作流执行结束---"); } } } }

4.我们还可以使用配置文件的方式来使用SqlTrackingService,app.config代码如下:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <section name="WorkflowRuntime" 
      type="System.Workflow.Runtime.Configuration.WorkflowRuntimeSection,
        System.Workflow.Runtime, Version=3.0.00000.0, Culture=neutral,
        PublicKeyToken=31bf3856ad364e35" />
  </configSections>
  <WorkflowRuntime>
    <CommonParameters>
      <!--Add parameters common to all services-->
      <add name="ConnectionString" 
           value="Initial Catalog=WorkflowTracking;
             Data Source=localhost\SQLEXPRESS;
             Integrated Security=SSPI;" />
    </CommonParameters>
    <Services>
      <!--Add core services here-->
      <add type="System.Workflow.Runtime.Tracking.SqlTrackingService, 
        System.Workflow.Runtime, Version=3.0.00000.0, Culture=neutral, 
        PublicKeyToken=31bf3856ad364e35"/>
    </Services>
  </WorkflowRuntime>
</configuration>

5.在宿主程序中我们需要使用重载的WorkflowRuntime构造函数来传递配置节名就可以了。
using(WorkflowRuntimeworkflowRuntime = newWorkflowRuntime("WorkflowRuntime")

6.现在我们就使用了SqlTrackingService跟踪服务,运行工作流我们会发现执行的结果只显示CodeActivity被执行了,并没有看到我们的跟踪信息,实际上现在跟踪数据已经存到的Sql Server的跟踪数据库了,不信你可以打开Sql Server数据库中去查看相应的表内容。

使用 SqlTrackingQuery 查询跟踪数据

1.上面我们看到了使用SqlTrackingService跟踪服务后,跟踪的信息已经保存到了数据库中,但是我们直接去数据库中查看很不方便,SqlTrackingQuery 类提供了一些方法和属性,可用于访问存储在 SqlTrackingService 跟踪数据库中的跟踪数据。 对此数据的访问是通过 SqlTrackingWorkflowInstance 对象提供的。 可以调用 TryGetWorkflow 方法来获取特定工作流实例的 SqlTrackingWorkflowInstance。 还可以选择调用 GetWorkflows 方法来获取 SqlTrackingWorkflowInstance 对象的集合,这些对象对应于具有跟踪数据的工作流实例,其中的跟踪数据与一组包含在作为参数传递到方法的SqlTrackingQueryOptions 对象中的查询参数相匹配。

通过传递跟踪数据库的连接字符串,创建 SqlTrackingQuery 类的实例。 然后,可以调用TryGetWorkflow 或GetWorkflows 方法来得到SqlTrackingWorkflowInstance 对象或 SqlTrackingWorkflowInstance 对象的集合。 ActivityEvents 属性包含 ActivityTrackingRecord 对象的集合,这些对象包含工作流中的活动的跟踪信息。 此外,还可以通过检查包含在WorkflowEvents 集合中的 WorkflowTrackingRecord 对象,使用相同的过程来跟踪工作流实例事件。 对于通过调用 TrackData 方法插入的用户事件,UserTrackingRecord 对象将包含在 UserEvents 集合中,该集合也是在SqlTrackingWorkflowInstance 类中定义的。

2.下面我们对SqlTrackingQuery 自己封装了一个类来将跟踪数据查询出来写到控制台上,TrackingConsoleWriter.cs,该类我们会在以后的文章也会使用,现在虽然用不了该类的全部功能,但是先给出类的全部代码:代码如下:

namespace CarySqlTrackingService
{
    public class TrackingConsoleWriter
    {
        private String connectionString = String.Empty;
        public TrackingConsoleWriter(String connString)
        {
            connectionString = connString;
        }
public void DisplayTrackingData(Guid instanceId) { SortedList<Int32, TrackingRecord> records = QueryTrackingData(instanceId); WriteSingleInstanceToConsole(instanceId, records); }
private SortedList<Int32, TrackingRecord> QueryTrackingData(Guid instanceId) { SortedList<Int32, TrackingRecord> records = new SortedList<int, TrackingRecord>(); try
{ SqlTrackingQuery query = new SqlTrackingQuery(connectionString); SqlTrackingWorkflowInstance instance = null; query.TryGetWorkflow(instanceId, out instance); BuildSortedList(records, instance); } catch (System.Data.SqlClient.SqlException e) { Console.WriteLine("SqlException in QueryTrackingData: {0}", e.Message); } return records; }
private static void BuildSortedList(SortedList<Int32, TrackingRecord> records,
SqlTrackingWorkflowInstance instance) { if (instance != null) { foreach (TrackingRecord record in instance.WorkflowEvents) { records.Add(record.EventOrder, record); } foreach (TrackingRecord record in instance.ActivityEvents) { records.Add(record.EventOrder, record); } foreach (TrackingRecord record in instance.UserEvents) { records.Add(record.EventOrder, record); } } }
private void WriteSingleInstanceToConsole(Guid instanceId,
SortedList<Int32, TrackingRecord> records) { Console.WriteLine("工作流实例 {0}的跟踪数据", instanceId); foreach (TrackingRecord record in records.Values) { if (record is WorkflowTrackingRecord) { WorkflowTrackingRecord wfRecord = record as WorkflowTrackingRecord; Console.WriteLine("{0:HH:mm:ss.fff} Workflow {1}", wfRecord.EventDateTime, wfRecord.TrackingWorkflowEvent); } else if (record is ActivityTrackingRecord) { ActivityTrackingRecord actRecord = record as ActivityTrackingRecord; Console.WriteLine("{0:HH:mm:ss.fff} {1} {2}, Type={3}",
actRecord.EventDateTime,actRecord.ExecutionStatus, actRecord.QualifiedName,actRecord.ActivityType.Name); WriteBodyToConsole(actRecord); } else if (record is UserTrackingRecord) { UserTrackingRecord userRecord = record as UserTrackingRecord; if (userRecord.UserData is RuleActionTrackingEvent) { WriteRuleData(userRecord); } else { Console.WriteLine("{0:HH:mm:ss.fff} UserData from {1} {2}:{3}", userRecord.EventDateTime,userRecord.QualifiedName, userRecord.UserDataKey,userRecord.UserData);
} } } Console.WriteLine("工作流实例{0}跟踪数据完成\n\r", instanceId); } private void WriteBodyToConsole(ActivityTrackingRecord record) { if (record.Annotations.Count > 0) { foreach (String annotation in record.Annotations) { Console.WriteLine("{0}", annotation); } } if (record.Body.Count > 0) { foreach (TrackingDataItem data in record.Body) { Console.WriteLine("{0}={1}",data.FieldName, data.Data); } } } private static void WriteRuleData(UserTrackingRecord userRecord) { RuleActionTrackingEvent ruleAction = userRecord.UserData as RuleActionTrackingEvent; Console.WriteLine( "{0:HH:mm:ss.fff} RuleAction from {1} Rule:{2} Result:{3}", userRecord.EventDateTime,userRecord.QualifiedName, ruleAction.RuleName,ruleAction.ConditionResult); } public void DisplayAllTrackingData(SqlTrackingQueryOptions options) { IList<SqlTrackingWorkflowInstance> workflows = QueryWorkflowList(options); SortedList<Int32, TrackingRecord> records = new SortedList<int, TrackingRecord>(); foreach (SqlTrackingWorkflowInstance wf in workflows) { records.Clear(); BuildSortedList(records, wf); WriteSingleInstanceToConsole(wf.WorkflowInstanceId, records); } } private IList<SqlTrackingWorkflowInstance> QueryWorkflowList(SqlTrackingQueryOptions options) { IList<SqlTrackingWorkflowInstance> workflows = new List<SqlTrackingWorkflowInstance>(); try { SqlTrackingQuery query = new SqlTrackingQuery(connectionString); workflows = query.GetWorkflows(options); } catch (System.Data.SqlClient.SqlException e) { Console.WriteLine("SqlException in QueryWorkflowList: {0}", e.Message); } return workflows; } } }

3.然后我们使用我们封装好的类来查询跟踪数据,需要添加如下代码就可以了:
TrackingConsoleWriter trackingWriter= new TrackingConsoleWriter(strConn);
trackingWriter.DisplayTrackingData(instance.InstanceId);

4.下面是执行结果:

SqlTrackingService1

用户跟踪的事件

1.一个User Track Point(用户跟踪点)是我们在工作流或是自定义活动中加入的。你可以加在任意的位置。我们如何创建用户跟踪点呢,我只要调用Activity的TrackData方法就可以了。这样你设定的用户跟踪点就会被跟踪服务记录下来,同样是一条UserTrackingRecord。我们在CodeActivity中添加TrackData方法,代码如下:

private void codeActivity1_ExecuteCode(object sender, EventArgs e)
{
   Console.WriteLine("CodeActivity执行了哦!");
   TrackData("CaryUserPoint", "CaryUserData");
}

2.再次运行工作流,结果如下:

SqlTrackingService2

3.从结果中我们可以看到红色线部分就是我们自己加入的跟踪点。

分享到:
评论

相关推荐

    workflow资料

    WF资料 ├─Activity │ │ 1_Activity 类.doc │ │ 2_状态机与顺序工作流的继承结构.doc │ │ 3_顺序工作流容器 SequentialWorkflowActivity .doc │ │ EventDriven绑定容器 ...

    网络编程网络编程网络编程

    网络编程网络编程网络编程网络编程

    setuptools-5.4.zip

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    基于树莓派智能小车(H5页面操作移动+实时显示摄像头内容+各类传感器)源码+详细文档+全部资料齐全 高分项目.zip

    【资源说明】 基于树莓派智能小车(H5页面操作移动+实时显示摄像头内容+各类传感器)源码+详细文档+全部资料齐全 高分项目.zip基于树莓派智能小车(H5页面操作移动+实时显示摄像头内容+各类传感器)源码+详细文档+全部资料齐全 高分项目.zip 【备注】 1、该项目是个人高分项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    2024-01-03-【办公自动化】Python执行Windows命令.md

    2024-01-03-【办公自动化】Python执行Windows命令

    基于FPGA的FS-FBMC调制器的设计源码+全部资料齐全.zip

    【资源说明】 基于FPGA的FS-FBMC调制器的设计源码+全部资料齐全.zip基于FPGA的FS-FBMC调制器的设计源码+全部资料齐全.zip 【备注】 1、该项目是高分课程设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过mac/window10/11/linux测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    MySQL进阶篇学习笔记

    黑马MySQL课程总结的学习笔记

    setuptools-41.1.0.zip

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    张祖豪-软件工程师-职业规划.pdf

    张祖豪-软件工程师-职业规划.pdf

    智慧工地整体解决方案qy.pptx

    智慧工地整体解决方案qy.pptx

    setuptools-49.1.1-py3-none-any.whl

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    setuptools-40.1.1.zip

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    基于FPGA设计的电梯控制电路系统源码+全部资料齐全.zip

    【资源说明】 基于FPGA设计的电梯控制电路系统源码+全部资料齐全.zip基于FPGA设计的电梯控制电路系统源码+全部资料齐全.zip 【备注】 1、该项目是高分课程设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过mac/window10/11/linux测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    全系统实战营销课,从小白到营销高手

    课程内容: 将成功建立在问题基层上 不做第- 就做唯一 新切割营销 品牌4s战略-构建高效强势品牌 智慧底牌-成功背后的10大思维方式 切割营销将对手通向一侧的营销策略 营销绝对竞争力 品牌突破 切割营销 企业家修炼-第九届学习型中国世纪成功论坛 如何创造七种动力进行整合营销 营销品牌教材 林伟贤董进宇姜岚昕演讲-第八届学习型中国世纪成功论坛 内部报告:处理人生10种关系智慧 高效构建强势品牌 营销纲领(首部谋定未来的营销大典)1~4

    基于SpringBoot+Vue的酒店(预约)客房管理系统的设计与实现+毕业论文(包运行成功)

    酒店客房管理系统为酒店管理者和用户、清洁人员提供一个在线管理酒店客房的系统。在网站的设计中,一共分为了两个模块设计,一个是前台模块,一个是后台模块,前台主要用于提供查看客房信息,酒店资讯,留言反馈,个人中心,在线客服等一系列的功能,后台会根据等于角色的不同分配不同的权限,如果登录的是管理员角色的话,则有管理员个人信息管理,用户管理,客房管理,清洁管理,系统管理等,如果登录的是用户角色的话,则有用户个人信息管理,预约管理,入住管理,收藏管理等,如果登录的是清洁人员角色的话,,则有清洁人员个人信息管理,退房管理,清洁管理等。 整个后台系统的大致功能如图1所示,整个后台系统分为两个部分,一部分为用户端,一部分为管理端,用户端的功能主要是用户来进行房屋预约,房屋入住,房屋收藏,浏览反馈,在线咨询。管理端也分三类角色的管理,管理员角色,用户角色,清洁人员角色,对应的角色不同,相应的对应的管理端也不同。

    IDC智能机房整体解决方案.ppt

    IDC智能机房整体解决方案.ppt

    setuptools-25.4.0.zip

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    setuptools-0.8.zip

    Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。

    电子行业周报:华为召开鸿蒙生态春季沟通会,智界S7与MateBook X Pro焕新亮相.pdf

    电子元件 电子行业 行业分析 数据分析 数据报告 行业报告

    PasteSpider的管理端静态页面V24.5.12.1(主服务端和文件同步器版本至少要24.5.12.1及以上版本)

    当前文件需要和PasteSpider配套使用, 配套的还有PasteSpiderFile(文件同步管理器), 和PasteSpider的主端(一款类似K8S的容器管理工具)。 当前版本为大版本升级,需要和配套的其他端的版本使用,否则出现访问错误等! 把当前文件解压后,放于服务器上,或者是方入PasteSpider的解压缩里面的wwwroot文件夹下。 个人建议独立存放,因为PasteSpider是需要打包到docker的镜像里面的,存放于宿主服务器上便于修改! 当前版本修改内容主要如下: 1.数据的获取修改为get模式,后续会基于http的method做一些特定的日志记录,大致的思路是get只做简单的校验,post做强校验并记录日志等。 2.定时任务添加任务串的支持,比如需要发布一个项目中的几个服务,他们有执行顺序,只需要把上一个任务的ID作为当前任务的父级ID即可, 3.静态服务(一般是web静态端不需要构建的),支持暂存模式,配合定时发布使用! 更多PasteSpider资料访问 https://blog.csdn.net/apeart/category_12291787.html

Global site tag (gtag.js) - Google Analytics