博客园  :: 首页  :: 联系 :: 订阅 订阅  :: 管理

WF与WebService

Posted on 2008-12-14 16:29  生鱼片  阅读(3003)  评论(5编辑  收藏  举报

WF与WebService交互

WF中提供了WebServiceInputActivity,WebServiceOutputActivity,InvokeWebServiceActivity,WebServiceFaultActivity四个活动来完成与Webservice
的交互,下面简要介绍下这四个活动:

1.WebServiceInputActivity活动:使工作流能够从 Web 服务接收数据。 在启动 Web 服务时将自身作为 Web 服务方法发布,然后在随后调用 Web 服务
方法时接收数据。
2.WebServiceOutputActivity活动:用于响应对工作流所做的 Web 服务请求,该活动必须与WebServiceInputActivity 活动关联。
3.WebServiceFaultActivity活动:为所出现的 Web 服务错误建立模型。
4.InvokeWebServiceActivity活动:通过代理类调用 Web 服务,并传递和接收指定的参数。

应用举例

1.下面我们举例说明,由于WebServiceInputActivity活动所接收的方法必须定义在接口中,所以我们先建立一个接口IAddService ,代码如下:

namespace CaryWFLib
{
    interface IAddService
    {
        double Add(double number1, double number2);
    }
}

2.工作流设计器如下图:

wfws1

3.在工作流设计器中我们首先拖入了一个WebServiceInputActivity活动,设置其InterfaceType属性,然后MethodName,然后将IsActivating设为true,
默认为false,当选了MethodName属性后属性窗口会自动多出参数的属性。在CodeActivity活动中完成加法运算逻辑,最后是WebServiceOutputActivity
活动,工作流的代码如下:

public sealed partial class AddWorkflow: SequentialWorkflowActivity
{
     public double number1;
     public double number2;
     public double result;

     public AddWorkflow()
     {
        InitializeComponent();
     }
     private void codeActivity1_ExecuteCode(object sender, EventArgs e)
     {
         result = number1 + number2;
     }        
}
4.这个时候你就可以用项目的右键菜单里的"作为Web服务发布"来将工作流发布为WebService,发布完成后会在当前的解决方案里多出一个WebService
的项目,我们就可以运行测试了,如下图:
wfws2 
只有工作流库才可以发布为WebService,新生成的WebService项目会包含一个CaryWFLib.AddWorkflow_WebService.asmx文件和一个Web.config
配置文件,web.config文件的内容如下:
<?xml version="1.0"?>
<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 Name="CaryWFLib">
    <CommonParameters>
      <add name="ConnectionString"
           value="Initial Catalog=WorkflowPersistence;Data Source=localhost\SQLEXPRESS;Integrated Security=SSPI;" />
    </CommonParameters>
    <Services>
      <add type="System.Workflow.Runtime.Hosting.ManualWorkflowSchedulerService, System.Workflow.Runtime, 
           Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" />
      <add type="System.Workflow.Runtime.Hosting.DefaultWorkflowCommitWorkBatchService, System.Workflow.Runtime, 
           Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" />
      <add type="System.Workflow.Runtime.Hosting.SqlWorkflowPersistenceService, 
        System.Workflow.Runtime, Version=3.0.00000.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" 
UnloadOnIdle="true"/> </Services> </WorkflowRuntime> <appSettings/> <connectionStrings/> <system.web> <compilation debug="false"/> <authentication mode="Windows"/> <httpModules> <add type="System.Workflow.Runtime.Hosting.WorkflowWebHostingModule, System.Workflow.Runtime, Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" name="WorkflowHost"/> </httpModules> </system.web> </configuration> web.config中的SqlWorkflowPersistenceService是我们手动添加的,其他的都是自动生成的。
5.我们第一次调用WebService是正常的,如果你再次调用就会出现下错误:
System.InvalidOperationException: 在状态持久性存储中找不到 ID 为“21cf8af3-885d-47dd-acbc-530eeba794ed”的工作流。
这是由于在web.config中使用了WorkflowWebHostingModule,我们第二次调用的时候WebService会去装载之前的工作流实例,而第一次调用完成后该实
例就销毁了,如果你关掉浏览器在打开就又正常了,这是因为新的工作流实例又产生了。
 
WorkflowWebHostingModule

1.WorkflowWebHostingModule类是默认的路由机制,用于通过使用ASP.NET cookie将web服务请求路由到相应的工作流,发出这些请求的客户端要支持
cookie。这个主要目的是为了可以多次的调用一个工作流实例的方法。我们从新设计一个工作流用于解决上面的问题,将原接口扩展一下,代码如下:

namespace CaryWFLib
{
    interface IAddStatefulService
    {
        void StartWorkflow();
        double Add(double number1, double number2);
        void StopWorkflow();
    }
}

2.工作流设计如下:

wfws3

我们使用ListenActivity来监听,WebServiceInputActivity1绑定StartWorkflow方法,WebServiceInputActivity2绑定Add方法,WebServiceInputActivity3
绑定StopWorkflow方法,如果调用StopWorkflow方法,我们就在WebServiceInputActivity3的调用完成的事件中将While的条件设为false。我们在web.config
中装载了持久化服务,所以在工作流Idle的时候就进行持久化的存储,当第二次调用的时候会根据cookie中保存的工作流实例id从持久化数据库中装载相同的
工作流实例。

3.如果在其他程序中,比如winform中,我们需要手动装载持久化中的工作流实例,那我们如何知道工作流实例的id呢,我们reflector一下WorkflowWebHostingModule的代码如下:

public sealed class WorkflowWebHostingModule : IHttpModule
{
    // Fields
    private HttpApplication currentApplication;

    // Methods
    public WorkflowWebHostingModule()
    {
        WorkflowTrace.Host.TraceEvent(8, 0, "Workflow Web Hosting Module Created");
    }

    private void OnAcquireRequestState(object sender, EventArgs e)
    {
        WorkflowTrace.Host.TraceEvent(8, 0, "WebHost Module Routing Begin");
        HttpCookie cookie = HttpContext.Current.Request.Cookies.Get("WF_WorkflowInstanceId");
        if (cookie != null)
        {
            HttpContext.Current.Items.Add("__WorkflowInstanceId__", new Guid(cookie.Value));
        }
    }

    private void OnReleaseRequestState(object sender, EventArgs e)
    {
        if (HttpContext.Current.Request.Cookies.Get("WF_WorkflowInstanceId") == null)
        {
            HttpCookie cookie = new HttpCookie("WF_WorkflowInstanceId");
            object obj2 = HttpContext.Current.Items["__WorkflowInstanceId__"];
            if (obj2 != null)
            {
                cookie.Value = obj2.ToString();
                HttpContext.Current.Response.Cookies.Add(cookie);
            }
        }
    }

    void IHttpModule.Dispose()
    {
    }

    void IHttpModule.Init(HttpApplication application)
    {
        WorkflowTrace.Host.TraceEvent(8, 0, "Workflow Web Hosting Module Initialized");
        this.currentApplication = application;
        application.ReleaseRequestState += new EventHandler(this.OnReleaseRequestState);
        application.AcquireRequestState += new EventHandler(this.OnAcquireRequestState);
    }
}
4.从reflector出的源代码我们可以发现,工作流实例的id存储在名称为WF_WorkflowInstanceId的cookie中,Item为__WorkflowInstanceId__。
我们通过如下代码就可以得到工作流实例的id了,如下:
localhost.AddWorkflow_WebService ws = new localhost.AddWorkflow_WebService();
ws.CookieContainer = new System.Net.CookieContainer();
ws.Url = "http://localhost:6411/CaryWFLib_WebService1/CaryWFLib.AddWorkflow_WebService.asmx";
ws.Add(1, 2);
Uri uri = new Uri(ws.Url);
CookieCollection cc = ws.CookieContainer.GetCookies(uri);
foreach (Cookie c in cc)
{
   if (c.Name == "WF_WorkflowInstanceId")
   {
       this.Label3.Text = c.Value;
   }
}
5.由于WorkflowWebHostingModule必须要求客户端启用cookie,当不能在客户端中启用 Cookie 时我们可以从新写一个WorkflowWebHostingModule。
6.上面的发布WebService是使用WF本身提供功能,我们完全可以自己来实现,代码如下:
namespace CaryWFLib
{
    [WebService(Namespace = "http://CaryWFLibWebService.com/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    public class CaryWFLibWebService:System.Web.Services.WebService
    {
        private static WorkflowRuntime workflowRuntime = new WorkflowRuntime();

        [WebMethod]
        public double Add(double number1,double number2)
        {
            double result=0;
            AutoResetEvent waitHandle = new AutoResetEvent(false);
            workflowRuntime.WorkflowCompleted +=
               delegate(object sender, WorkflowCompletedEventArgs e)
               {
                   result =Convert.ToDouble( e.OutputParameters["result"]);
                   waitHandle.Set();
               };

            workflowRuntime.WorkflowTerminated +=
               delegate(object sender, WorkflowTerminatedEventArgs e)
               {
                   waitHandle.Set();
               };           
            Dictionary<string, object> wfParas = new Dictionary<string, object>();
            wfParas.Add("number1", number1);
            wfParas.Add("number2", number2);
            WorkflowInstance instance =
               workflowRuntime.CreateWorkflow(typeof(ManualAddWorkflow),wfParas);
            instance.Start();
            waitHandle.WaitOne();
            return result;
        }    
    }
}

当然这个时候,ManualAddWorkflow工作流的设计就只需要一个CodeActivity来完成加法的逻辑就可以了,WebServiceInputActivity等这些活动就不需要了。