我们可以将Workflow Service作为Windows服务托管吗?

时间:2010-03-30 19:44:29

标签: wcf workflow named-pipes

我正在开发一个日志记录应用程序,它要求我有一个作为服务公开的工作流程(工作流程服务)。我们希望将其作为Windows服务托管(不希望将工作流服务作为IIS中的.svc文件托管)。将其作为Windows服务的另一个原因是能够通过命名管道与服务进行通信。

我们是否可以通过命名管道公开工作流服务而无需在IIS中托管它?

2 个答案:

答案 0 :(得分:5)

是的,你确定可以。至少,我已经完成了Workflow 4 Release Candidate。

考虑,

// a generic self-hosted workflow service hosting thingy. Actual
// implementation should contain more logging and thread safety, this
// is an abbreviated version ;)
public class WorkflowHost
{

    // NOTE: with Workflow, it helps to maintain a concept of
    // Workflow definition [the Activity or WorkflowService from
    // a designer] and a Workflow instance [what is running within
    // WorkflowInvoker, WorkflowApplication, WorkflowServiceHost].
    // a definition may be used to generate an instance. an instance
    // contains run-time state and cannot be recycled into a new
    // instance. therefore, to repeatedly re-host a WorkflowService
    // we need to maintain references to original definitions and
    // actual instances. ergo services and hosts maps
    // 
    // if you are special purpose and require support for one and 
    // only one service and endpoint\uri, then you may reduce this 
    // to a simple tuple of Uri, WorkflowService, WorkflowServiceHost

    // services represents a definition of hosted services
    private readonly Dictionary<Uri, WorkflowService> _services = 
        new Dictionary<Uri, WorkflowService> ();

    // hosts represents actual running instances of services
    private readonly Dictionary<Uri, WorkflowServiceHost> _hosts = 
        new Dictionary<Uri, WorkflowServiceHost> ();

    // constructor accepts a map of Uris (ie service endpoints) to
    // workflow service definitions
    public WorkflowHost (IDictionary<Uri, WorkflowService> services)
    {
        foreach (KeyValuePair<Uri, WorkflowService> servicePair in services)
        {
            _services.Add (servicePair.Key, servicePair.Value);
        }
    }

    // have your windows service invoke this to start hosting
    public void Start ()
    {
        if (_hosts.Count > 0)
        {
            Stop ();
        }

        foreach (KeyValuePair<Uri, WorkflowService> servicePair in _services)
        {
            WorkflowService service = servicePair.Value;
            Uri uri = servicePair.Key;
            WorkflowServiceHost host = new WorkflowServiceHost (service, uri);

            host.Open ();

            _hosts.Add (uri, host);
        }
    }

    // have your windows service invoke this to stop hosting
    public void Stop ()
    {
        if (_hosts.Count > 0)
        {
            foreach (KeyValuePair<Uri, WorkflowService> servicePair in 
                _services)
            {
                WorkflowService service = servicePair.Value;
                Uri uri = servicePair.Key;

                IDisposable host = _hosts[uri];
                host.Dispose ();
            }

            _hosts.Clear ();
        }
    }
}

我认为可以通过App.config中的标准Wcf服务配置部分设置端点配置。在我的Workflow实验中,我没有亲自尝试更改默认传输层。

上面代表了一个通用的纯托管类[即它自托管WorkflowServices]。这允许我们在控制台,WinForm,WPF或是,甚至是WindowsService应用程序中重用此托管功能。下面是一个利用我们的主机类

的WindowsService
// windows service. personally i would abstract service behind
// an interface and inject it, but again, for brevity ;)
public partial class WorkflowWindowsService : ServiceBase
{
    WorkflowHost _host;

    public WorkflowWindowsService ()
    {
        InitializeComponent();

        Dictionary<Uri, WorkflowService> services = 
            new Dictionary<Uri, WorkflowService> ();

        // do your service loading ...

        // create host
        _host = new WorkflowHost (services);
    }

    protected override void OnStart(string[] args)
    {
        _host.Start ();
    }

    protected override void OnStop()
    {
        _host.Stop ();
    }
}

如果您在VS2010RC中摆弄了WorkflowServices,那么您可能已经知道WorkflowServices不是他们的Workflow表兄弟那样的第一类Xaml类。相反,它们被保存为带有.xamlx扩展名的松散Xaml文件。对于WorkflowServices没有设计时智能感知支持[据我所知]并且不被识别为声明类型,因此我们在运行时加载WorkflowService的唯一选择是

  • 直接从.xamlx文件中读取纯Xaml标记
  • 从其他来源[嵌入字符串,资源或其他来源]中读取纯Xaml标记

无论哪种方式,我们都必须解释标记并创建WorkflowService定义。以下内容将字符串[可能是文件名或标记]转换为WorkflowService。 Keeners还可能注意到此流程与将工作流标记转换为工作流定义的过程之间存在差异。

// converts a string value [either pure xaml or filename] to a
// WorkflowService definition
public WorkflowService ToWorkflowService (string value)
{
    WorkflowService service = null;

    // 1. assume value is Xaml
    string xaml = value;

    // 2. if value is file path,
    if (File.Exists (value))
    {
        // 2a. read contents to xaml
        xaml = File.ReadAllText (value);
    }

    // 3. build service
    using (StringReader xamlReader = new StringReader (xaml))
    {
        object untypedService = null;

        // NOTE: XamlServices, NOT ActivityXamlServices
        untypedService = XamlServices.Load (xamlReader);

        if (untypedService is WorkflowService)
        {
            service = (WorkflowService)(untypedService);
        }
        else
        {
            throw new ArgumentException (
                string.Format (
                "Unexpected error reading WorkflowService from " + 
                "value [{0}] and Xaml [{1}]. Xaml does not define a " + 
                "WorkflowService, but an instance of [{2}].", 
                value, 
                xaml, 
                untypedService.GetType ()));
        }
    }

    return service;
}

答案 1 :(得分:1)

是的,这是可能的。您必须创建自己的服务。请参阅MSDN上的Hosting and Consuming WCF Services,尤其是在Windows服务中托管部分。