WCF Silverlight CommunicationException

时间:2011-08-12 12:36:14

标签: silverlight wcf communicationexception

这篇文章很长,我为此事先道歉。

我正在尝试让一个Silverlight应用程序与我从visual studio运行的服务进行通信,目前它不在IIS中托管。我试图为这个原型尽可能少的代码和配置,只是最低限度。互联网上充满了关于此问题的答案,但它们似乎都没有适用于我的特殊困境。

我的服务解决方案有一个SimpleServiceAsyncResult,它实现了IAsyncResult接口,服务接口,ISimpleService以及SimpleService中服务的实际实现。该服务实现如下:

public class SimpleService : ISimpleService
{
    public IAsyncResult BeginSimpleMethod(string msg, AsyncCallback callback, object state)
    {
        return new SimpleServiceAsyncResult<string>(msg);
    }

    public string EndSimpleMethod(IAsyncResult result)
    {
        SimpleServiceAsyncResult<string> res = result as SimpleServiceAsyncResult<string>;
        return res.Result;
    }

    public string SimpleMethod(string msg)
    {
        return msg;
    }


    public Message ProvideCrossDomainFile()
    {
        FileStream fileStream = File.Open(@"CrossDomain.xml", FileMode.Open, FileAccess.Read, FileShare.Read);

        XmlReaderSettings settings = new XmlReaderSettings();
        settings.DtdProcessing = DtdProcessing.Parse;

        XmlReader reader = XmlReader.Create(fileStream, settings);

        Message result =  Message.CreateMessage(MessageVersion.None, "", reader);
        return result;
    }

    public Message ProvidePolicyFile()
    {
        FileStream fileStream = File.Open(@"ClientAccessPolicy.xml", FileMode.Open, FileAccess.Read, FileShare.Read);
        XmlReader reader = XmlReader.Create(fileStream);
        Message result = Message.CreateMessage(MessageVersion.None, "", reader);
        return result;
    }
}

项目中有2个xml文件; ClientAccessPolicy.xml和CrossDomain.xml,如下所示:

ClientAccessPolicy.xml

<?xml version="1.0" encoding="utf-8"?>
<access-policy>
  <cross-domain-access>
    <policy>
      <allow-from http-request-headers="*">
        <domain uri="*"/>
      </allow-from>
      <grant-to>
        <resource path="/" include-subpaths="true"/>
      </grant-to>
    </policy>
  </cross-domain-access>
</access-policy>

的crossdomain.xml

<?xml version="1.0"?>
<!DOCTYPE cross-domain-policy SYSTEM "http://www.macromedia.com/xml/dtds/cross-domain-policy.dtd">
<cross-domain-policy>
  <allow-access-from domain="*" />
  <allow-http-request-headers-from domain="*" headers="SOAPAction" />
</cross-domain-policy> 

这是一个App.config,它看起来像:

的App.config

                                                                                                                                           

<behaviors>
  <serviceBehaviors>
    <behavior name="SimpleServiceBehavior">
      <serviceMetadata httpGetEnabled="True"/>
    </behavior>
  </serviceBehaviors>
</behaviors>

现在,我想看看这一切是否真的有效。所以我创建了一个带有控制台项目的解决方案,只有Program.cs和App.config。

                                                                 该计划当然没什么特别的,只是最低限度:

       var factory = new ChannelFactory<ISimpleService>("SimpleService");
        ISimpleService service = factory.CreateChannel();

        var asResult = service.BeginSimpleMethod("tesmsg", null, null);

        Console.WriteLine("Data: " + service.EndSimpleMethod(asResult);

这一切都很好用,我得到了我正在寻找的数据,所以我很高兴。 我想要实现的下一件事是使用Silverlight应用程序从同一服务获取一些数据。

为此,我在我的客户端解决方案中创建了一个Silverlight项目,创建了服务引用并执行了soms管道。我试图通过在MainPage.xaml.cs中添加一些代码来执行SimpleMethod:

    void SimpleMethodCompletedProxyCallback(object sender, SimpleMethodCompletedEventArgs e)
    {
        if (e.Error == null)
        {
            data = e.Result;
        }
    }

    private void button1_Click(object sender, RoutedEventArgs e)
    {
        SimpleServiceClient proxy = new SimpleServiceClient();

        proxy.SimpleMethodCompleted += new EventHandler<SimpleMethodCompletedEventArgs>(SimpleMethodCompletedProxyCallback);
        try
        {

            proxy.SimpleMethodAsync("teststring");
        }
        catch (CommunicationException cex)
        {
            label1.Content = cex.Message;
        }
    }

完成这个管道后,我以为我能够获得一些数据,但是当它试图调用SimpleMethod时,它会继续从生成的ServiceReference代码中抛出一个CommunicationException:

        public string EndSimpleMethod(System.IAsyncResult result) {
            object[] _args = new object[0];
            string _result = ((string)(base.EndInvoke("SimpleMethod", _args, result)));
            return _result;
        }

我已经在网上浏览了几个小时,结束了CrossDomain.xml和ClientAccessPolicy.xml以及会暴露这个的方法,但无论我尝试什么,我都会遇到同样的异常。我完全不知道下一步该做什么,我希望你能帮助我更进一步。

亲切的问候

1 个答案:

答案 0 :(得分:2)

在您的服务代码中,SimpleService类实现了一个接口(ISimpleService),我假设它使用[ServiceContract]进行修饰并定义了五个操作契约(BeginSimpleMethod,EndSimpleMethod,SimpleMethod) ,ProvideCrossDomainFile,ProvidePolicyFile)。也许ProvidePolicyFile也用[WebGet(UriTemplate =“/ clientaccesspolicy.xml”)]进行修饰。

现在,由于您只有一个接口,因此您有以下两种情况之一:服务器中的端点是SOAP端点(例如,基于basicHttpBinding),或者端点是REST端点(即, webHttpBinding / webHttp行为)。

如果是前者,则返回clientaccesspolicy.xml的操作的位置不正确。根据{{​​3}}上的备注部分,它是被动行为,只有在端点具有WebHttpBehavior时才会受到尊重。 Silverlight将在一个地方(http://your_service/clientaccesspolicy.xml)查找该策略文件,并且可以在其他地方访问它(http:// your_service / endpoint,其中SOAPAction标头指示该方法)。

如果是后者,那么Silverlight默认情况下不能调用这些端点(请参阅从SL调用documentation for WebGetAttributeREST/POX端点的帖子),因此它也无法正常工作。

REST/JSON上的帖子是AFAIK,这是使SL能够在自托管WCF服务上调用服务的“规范”方式。您将需要两个端点(可能在两个接口中),一个用于服务本身,一个用于策略文件。该帖子有关于这个问题的更多细节。