这篇文章很长,我为此事先道歉。
我正在尝试让一个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,如下所示:
<?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>
<?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,它看起来像:
<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以及会暴露这个的方法,但无论我尝试什么,我都会遇到同样的异常。我完全不知道下一步该做什么,我希望你能帮助我更进一步。
亲切的问候
答案 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 WebGetAttribute和REST/POX端点的帖子),因此它也无法正常工作。
REST/JSON上的帖子是AFAIK,这是使SL能够在自托管WCF服务上调用服务的“规范”方式。您将需要两个端点(可能在两个接口中),一个用于服务本身,一个用于策略文件。该帖子有关于这个问题的更多细节。