2016-06-10 31 views
4

我需要我的WCF服务将事件提交给客户端。我读过一个碰巧经过回调通道,我已经实现了它以下列方式: 服务接口:如何在通过ChannelFactory创建的WCF服务上设置回调通道?

public interface IServiceCallback 
{ 
    [OperationContract(IsOneWay = true)] 
    void OnNewAlert(Alert a); 
    [OperationContract(IsOneWay = true)] 
    void OnProductEdited(Product p); 
    [OperationContract(IsOneWay = true)] 
    void OnHighlightChanged(Dictionary<User, List<Product>> highlighted); 
    [OperationContract(IsOneWay = true)] 
    void OnCatalogUpdated(); 


    event EventHandler NewAlert; 
    event EventHandler ProductEdited; 
    event EventHandler HighlightChanged; 
    event EventHandler CatalogUpdated; 
} 
[ServiceContract(CallbackContract = typeof(IServiceCallback))] 
public interface IService : IDisposable 
{ 
    [OperationContract] 
    List<Product> GetProducts(Predicate<Product> match = null, int limit = 0, string username = null); 
    [OperationContract] 
    Product GetProduct(Predicate<Product> match, string username = null); 
    [OperationContract] 
    Product GetRandomProduct(Predicate<Product> match = null, string username = null); 
    [OperationContract] 
    int GetFlagIndex(string flagName); 
    [OperationContract] 
    void SetFlag(string pid, string flagName, bool value); 
    [OperationContract] 
    List<Alert> GetAlerts(string username); 
    [OperationContract] 
    void DismissAlert(Alert alert, String username); 
    [OperationContract] 
    void HighlightProduct(List<string> pids, string user); 
    [OperationContract] 
    void EditProduct(string pid, Dictionary<string, object> fieldValues, string username = null); 
    [OperationContract] 
    void AttachModule(IModule m); 
    [OperationContract] 
    void Ping(); 

    event EventHandler NewAlert; 
    event EventHandler ProductEdited; 
    event EventHandler HighlightChanged; 
    event EventHandler CatalogUpdated; 
} 

服务实现:

namespace Service 
{ 
[ServiceBehavior(InstanceContextMode = InstanceContextMode.PerCall, ConcurrencyMode = ConcurrencyMode.Reentrant)] 
public class ServiceInstance : IService 
{ 
    List<IServiceCallback> callbackChannels = new List<IServiceCallback>(); 
    //other vars 

    public ServiceInstance() 
    { 
      //lots of stuff here 
    } 

    private User SignalUser(string username) 
    { 
     if (username == null) 
      return null; 

     IServiceCallback channel = OperationContext.Current.GetCallbackChannel<IServiceCallback>(); 
     if (!callbackChannels.Contains(channel)) //if CallbackChannels not contain current one. 
     { 
      callbackChannels.Add(channel); 
     } 

     User user = knownUsers.Find(p => p.username == username); 
     if (user == null) 
     { 
      user = new User(); 
      user.username = username; 
      user.highlighColor = Color.FromArgb(r.Next(0, 128), r.Next(0, 128), r.Next(0, 128)); 
      knownUsers.Add(user); 
      foreach (KeyValuePair<Alert, List<User>> kvp in alerts) 
      { 
       kvp.Value.Add(user); 
      } 
     } 
     user.lastOnline = DateTime.Now; 
     if(!onlineUsers.Contains(user)) 
      onlineUsers.Add(user); 

     return user; 
    } 

    //lots of other things here 
} 
} 

上的客户端回调实现:

class ServiceEventHandler : IServiceCallback 
{ 
    public event EventHandler NewAlert; 
    public event EventHandler ProductEdited; 
    public event EventHandler HighlightChanged; 
    public event EventHandler CatalogUpdated; 

    public void OnCatalogUpdated() 
    { 
     CatalogUpdated?.BeginInvoke(null, null, null, null); 
    } 

    public void OnHighlightChanged(Dictionary<User, List<Product>> highlighted) 
    { 
     HighlightChanged?.BeginInvoke(highlighted, EventArgs.Empty, null, null); 
    } 

    public void OnNewAlert(Alert a) 
    { 
     NewAlert?.BeginInvoke(a, EventArgs.Empty, null, null); 
    } 

    public void OnProductEdited(Product p) 
    { 
     ProductEdited?.BeginInvoke(p, EventArgs.Empty, null, null); 
    } 
} 

但这是我的问题: 在客户端,我是应该将它传递给这样的服务:根据这个StackOverflow的答案

EventHandler eventHandler = new EventHandler(); 
MyServiceClient client = new MyServiceClient(new InstanceContext(eventHandler)); 

https://stackoverflow.com/a/1143777/2018696

但我不连接到我的服务就是这样,因为我的客户不知道的实施的服务,它只知道两个接口! 所以我连这样的:

public static IService GetService(string serviceAddress) 
    { 
     Uri service_uri = new Uri(serviceAddress); 
     var endpoint = new EndpointAddress(service_uri, new[] { AddressHeader.CreateAddressHeader(settings["username"], "", "") }); 
     IService service = ChannelFactory<IService>.CreateChannel(new BasicHttpBinding(), endpoint); 
     return service; 
    } 

那么,如何让回调的工作?

UPDATE:

好,从而提出一个意见,我更换的ChannelFactory与DuplexChannelFactory和basicHttpBinding的与WsDualHTTPBinding,我没有得到来自服务器的响应。如果我抓取回调处理程序,我会得到BasicHTTPBinding的响应。所以基本上:

[ServiceContract] 
BasicHttpBinding(); 
ChannelFactory<IService>.CreateChannel(binding, endpoint); 

^这个作品

[ServiceContract(CallbackContract = typeof(IServiceCallback))] 
WSDualHttpBinding(WSDualHttpSecurityMode.None); 
DuplexChannelFactory<IService>.CreateChannel(new InstanceContext(handler), binding, endpoint); 

^这个没有。

它适用于localhost,但不适用于LAN或Internet。防火墙在服务器和客户端都处于关闭状态。当我尝试联系服务器时,我收到了60秒的超时时间。

+3

请查看[DuplexChannelFactory类](https://msdn.microsoft.com/en-us/library/ms576164(v = vs.110).aspx),它应该指向您继续的方向。 – carlosfigueira

+0

@carlosfigueira谢谢,我做了一些修改,但是现在我没有得到服务的回应,请在操作中查看“更新”。谢谢! – Daniel

回答

0

我发现我的连接问题是由于在客户端的路径上没有设置端口转发。由于我无法确保正确的端口访问客户端,我已恢复为非回调模式,并将使用来自客户端的常规请求来接收来自服务的累积事件数据。可能效率不高,但似乎迄今为止工作的一种万无一失的方法。谢谢大家的关注。

相关问题