如何异步通知客户端wf4-wcf-server?

时间:2012-08-09 07:09:32

标签: c# wcf asynchronous workflow-foundation-4 wcf-client

在过去的几天里,我花了很多时间找到一个解决方案来解决如何从工作流(WF4) - 服务器向客户端发送简单消息(在我的案例中是WPF)。

有一些例子如何回复请求(通过回调),但我找不到任何例子如何异步联系客户端。

我希望能够进行这样的沟通:

  • [上午8:00]客户:“Hello Server,这是我的要求。请使用此配方开始您的工作流程'MakeCoffee'。”
  • [8:01 am] WF-Server:“你好客户,我就在它上面。”
  • [8:05 am] WF-Server:“我找不到任何咖啡。我会买一些。”
  • [8:10 am] WF-Server:“我正在煮水。”
  • [8:15 am] WF-Server:“你想要糖吗?”
  • [8:17 am]客户:“是的。”
  • [8:18 am] WF-Server:“我完成了。”

到目前为止,我找到了这些解决方案:

1。 WCF - 使用CallBack-Channel进行服务

This example似乎很酷,但我不知道如何在我的Workflow-Service中使用它 - 因为没有“实施服务”类。一切都是可视的,所以我不能应用“SubscribeClient”方法(或者至少我不知道在哪里放置服务实现)。

2。通知事件的工作流程

This example是一个很好的解决方案 - 但(据我所知)这不适用于单独的工作流服务器 - 它似乎要求工作流服务器和客户端是相同的。

3。访问OperationContext

This example在客户端创建了一个WorkflowServiceHost,但它不是很容易处理,我无法根据我的要求进行调整。

有没有人知道如何从(单独的)工作流服务器向客户端发送消息的方便解决方案/模式?客户端不知道,服务器何时向他发送消息以及如何它会很多。

我是WF4的新手,非常感谢每一个提示/建议/想法。

提前致谢,

蒂莫

更新

根据Maurice的评论,我尝试在我的客户端使用简单的服务,但不幸的是,我的客户端没有收到并显示任何消息。

这是我的客户代码:

public partial class MainWindow : Window  
{  
    private ServiceHost _serviceHost;  
    private void Window_Loaded(object sender, RoutedEventArgs e)  
    { 
        ApplicationInterface._app = this;  
        //prepare the serviceHost  
        string clientAddress = "http://localhost:8000/ClientService";  
        System.ServiceModel.Channels.Binding bBinding = new BasicHttpBinding();  
        _serviceHost.AddServiceEndpoint(typeof(IClientService), bBinding, clientAddress);  
        _serviceHost.Open();  
    }

    public ListBox GetEventListBox()
    {
        return this.lstEvents;
    }
}  

这是我的IClientService-Interface:

[ServiceContract]  
public interface IClientService  
{  
    [OperationContract(IsOneWay=true)]  
    void MeldeStatus(String statusText);  
}  

这是IClientService-Interface的实现:

public class ClientService : IClientService  
{  
    public void MeldeStatus(string statusText)  
    {  
        ApplicationInterface.AddEvent(statusText);  
    }  
}  

这是静态ApplicationInterface:

public static class ApplicationInterface  
{  
    public static MainWindow _app { get; set; }  

    public static void AddEvent(String eventText)  
    {  
        if (_app != null)  
        {  
            new ListBoxTextWriter(_app.GetEventListBox()).WriteLine(eventText);  
        }  
    }  
}

这就是ListBoxTextWriter-class,它应该将消息添加到给定的ListBox中:

public class ListBoxTextWriter : TextWriter
{
    const string textClosed = "This TextWriter must be opened before use";

    private Encoding _encoding;
    private bool _isOpen = false;
    private ListBox _listBox;

    public ListBoxTextWriter()
    {
        // Get the static list box
        _listBox = ApplicationInterface._app.GetEventListBox();
        if (_listBox != null)
            _isOpen = true;
    }

    public ListBoxTextWriter(ListBox listBox)
    {
        this._listBox = listBox;
        this._isOpen = true;
    }

    public override Encoding Encoding
    {
        get
        {
            if (_encoding == null)
            {
                _encoding = new UnicodeEncoding(false, false);
            }
            return _encoding;
        }
    }

    public override void Close()
    {
        this.Dispose(true);
    }

    protected override void Dispose(bool disposing)
    {
        this._isOpen = false;
        base.Dispose(disposing);
    }

    public override string ToString()
    {
        return "";
    }

    public override void Write(char value)
    {
        if (!this._isOpen)
            throw new ApplicationException(textClosed); ;

        this._listBox.Dispatcher.BeginInvoke
            (new Action(() => this._listBox.Items.Add(value.ToString())));
    }

    public override void Write(string value)
    {
        if (!this._isOpen)
            throw new ApplicationException(textClosed); ;

        if (value != null)
            this._listBox.Dispatcher.BeginInvoke
                (new Action(() => this._listBox.Items.Add(value)));
    }

    public override void Write(char[] buffer, int index, int count)
    {
        String toAdd = "";

        if (!this._isOpen)
            throw new ApplicationException(textClosed); ;

        if (buffer == null || index < 0 || count < 0)
            throw new ArgumentOutOfRangeException("buffer");


        if ((buffer.Length - index) < count)
            throw new ArgumentException("The buffer is too small");

        for (int i = 0; i < count; i++)
            toAdd += buffer[i];

        this._listBox.Dispatcher.BeginInvoke
            (new Action(() => this._listBox.Items.Add(toAdd)));
    }
}

我的WF-Server使用此发送活动:

<Send Action="MeldeStatus" EndpointConfigurationName="BasicHttpBinding_Client" sap:VirtualizedContainerService.HintSize="255,90" OperationName="MeldeStatus" ProtectionLevel="None" ServiceContractName="p:IClientService">
  <SendParametersContent>
    <p1:InArgument x:TypeArguments="x:String" x:Key="statusText">Yes, it works.</p1:InArgument>
  </SendParametersContent>
</Send>

这是我在WF服务器端点的配置:

<system.serviceModel>
  <bindings>
    <basicHttpBinding>        
      <binding name="DefaultHTTPBinding" allowCookies="true" />
    </basicHttpBinding>
  </bindings>    
  <behaviors>
    <serviceBehaviors>
      <behavior>
        <serviceMetadata httpGetEnabled="true"/>          
        <serviceDebug includeExceptionDetailInFaults="true"/>
      </behavior>
    </serviceBehaviors>
  </behaviors>    
  <services>
    <service name="WFServer.ReklamationErfassen">
      <endpoint address="http://localhost:7812/ReklamationErfassen.xamlx"
                binding="basicHttpBinding"
                bindingConfiguration="DefaultHTTPBinding"
                contract="IReklamationService" />
    </service>
  </services>
  <client>      
      <endpoint address="http://localhost:8000/ClientService" binding="basicHttpBinding"
          bindingConfiguration="DefaultHTTPBinding" contract="IClientService"
          name="BasicHttpBinding_Client" />
  </client>
</system.serviceModel>
<system.webServer>
  <modules runAllManagedModulesForAllRequests="true"/>
</system.webServer>

我没有收到任何消息,错误或警告 - 有人可以快速检查一下给定的代码吗?

再次感谢,
蒂莫

1 个答案:

答案 0 :(得分:0)

有很多方法可以做到这一点,但最简单的两个方法是:

  • 使用“发送”活动将SOAP消息发送到客户端应用程序托管的WCF端点。如果客户端需要响应,只需添加一个Receive活动来等待答案。
  • 使用数据库表将消息写入。客户端应用程序轮询消息并将其显示给用户。如果需要回复,您可以使用回复恢复自定义活动的书签。

第一个依赖于WCF以及WorkflowServiceHost中托管的响应。第二种方法更为通用,不依赖于WCF,但需要更多代码和自定义活动。