通过.NET远程处理将流传递给方法

时间:2013-11-14 14:00:57

标签: c# .net-4.0 stream remoting channel

让我简要描述一下情况。

首先,我有一个使用此方法的WCF RestFul Web服务:

[WebInvoke(UriTemplate = "/Dynamic/{*sParameter}", Method = "POST", ResponseFormat = WebMessageFormat.Json)]
    public string ExecuteWebAPIRequest(string sParameter, Stream streamPost)
    {
        ...

        var oClientFormatSinkProvider = new BinaryClientFormatterSinkProvider();
        IDictionary aoProps = new Hashtable();
        aoProps["port"] = 4626;
        aoProps["timeout"] = "-1";
        aoProps["name"] = "clientChan";
        TcpClientChannel channel = new TcpClientChannel(aoProps, oClientFormatSinkProvider);
        ChannelServices.RegisterChannel(channel, false);

        //-- Call Bridge
        string result = GetBridgeObject().ExecuteWebAPIRequest(sIpAddress, streamPost, sParameter);

        //-- Return result
        return result ?? "";
    }

这里是GetBridgeObject()方法的内容(即远程客户端):

private IBridge GetBridgeObject()
    {
        return (IBridge)Activator.GetObject(typeof(IBridge), "tcp://localhost:4626/RemoteBridge");
    }

接下来,有一个包含此方法的桥接进程(即远程服务器):

public void StartService()
    {
        //-- Initialize .NET remoting
        var oServerFormatSinkProvider = new BinaryServerFormatterSinkProvider();
        oServerFormatSinkProvider.TypeFilterLevel = TypeFilterLevel.Full;
        IDictionary aoProps = new Hashtable();
        aoProps["port"] = 4626;
        aoProps["timeout"] = "-1";
        aoProps["name"] = "serverChan";
        TcpServerChannel channel = new TcpServerChannel(aoProps, oServerFormatSinkProvider);
        ChannelServices.RegisterChannel(channel, false);
        RemotingConfiguration.RegisterWellKnownServiceType(typeof (Bridge), "RemoteBridge", WellKnownObjectMode.Singleton);
    }

最后,在远程桥对象中,这种方法:

public string WUPP_ExecuteWebAPIRequestI(string sPPInstanceName, Stream oInputStream, string sParameter)
    {
        ...

        int read = oInputStream.Read(buffer, 0, buffer.Length); //That's where the problem is
        ...    
    }

如代码段中所述,当我尝试读取流时出现问题,我收到此错误:

  

异常:System.Runtime.Remoting.RemotingException:此远程处理代理没有通道接收器,这意味着服务器没有正在侦听的已注册服务器通道,或者此应用程序没有合适的客户端通道与服务器通信。      在System.Runtime.Remoting.Proxies.RemotingProxy.InternalInvoke(IMethodCallMessage reqMcmMsg,Boolean useDispatchMessage,Int32 callType)      在System.Runtime.Remoting.Proxies.RemotingProxy.Invoke(IMessage reqMsg)      在System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData,Int32 type)      在System.IO.Stream.Read(Byte [] buffer,Int32 offset,Int32 count)

我确信我可以通过.NET远程传递流,因为在桥中,还有其他方法可以返回流并且运行良好。

我想当我注册频道时问题出现在远程服务器或客户端的某处,但经过两天的研究和测试后,我仍然没有找到答案。

提前感谢您的帮助。

1 个答案:

答案 0 :(得分:0)

由于我遇到了类似的问题并且需要解决方案,我终于找到了一个解决方案:客户端通道需要以不同方式注册才能使其正常工作。

        // Creating a custom formatter for a TcpChannel sink chain.
        BinaryServerFormatterSinkProvider provider = new BinaryServerFormatterSinkProvider();
        BinaryClientFormatterSinkProvider clientProvider = new BinaryClientFormatterSinkProvider();
        provider.TypeFilterLevel = TypeFilterLevel.Full;

        Dictionary<string, object> dict = new Dictionary<string, object>();
        dict.Add("typeFilterLevel", "Full");
        dict.Add("port", "0");

        ChannelServices.RegisterChannel(new TcpChannel(dict, clientProvider, provider), false);

这里有趣的事情是

  1. 频道是双向的
  2. 端口0告诉远程处理框架决定它应该使用哪个端口。
  3. 用于上传和下载typeFilterLevel Full所需的流 集。
  4. 有兴趣知道为什么我的答案得到了一个downvote,因为解决方案适用于生产代码。