没有参数的 WebGet 或 UriTemplate 失败
WebGet with No Parameters or UriTemplate Fails
我有一个 RESTful WCF 网络服务,其中 API:
[WebGet(ResponseFormat = WebMessageFormat.Json)]
MyResponseContract GetFileInfo();
尝试访问端点(使用 SOAPUI)时,我看到以下错误消息:
The server encountered an error processing the request. Please see the
service help page for constructing valid requests to the service.
我将 SOAPUI 设置为使用 GET 方法调用来命中它。当我将它切换到没有正文的 POST 时,它失败并显示以下消息:
Method not allowed.
这很有道理:不能用 POST 命中 GET。所以我更新了我的代码如下:
[WebInvoke(ResponseFormat = WebMessageFormat.Json)]
MyResponseContract GetFileInfo();
现在我使用 POST 方法从 SOAPUI 调用它并且它有效。好奇的。所以我现在更改我的代码如下:
[WebInvoke(ResponseFormat = WebMessageFormat.Json, Method = "GET")]
MyResponseContract GetFileInfo();
我在一些帖子中看到这基本上等同于 WebGet
属性。这也不行。
所以我的问题是:即使我不接受参数或不使用自定义 UriTemplate,为什么它不能作为 WebGet
工作?
我尝试使用的 Url(它本地托管在 IIS 中)是:
http://localhost/Utilities/API/GetFileInfo
更新
鉴于下面的评论和给出的答案,我仍然面临这个问题。一些额外的细节。
我的网络层web.config
<?xml version="1.0"?>
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.0" />
<customErrors mode="Off" />
</system.web>
<system.serviceModel>
<serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" />
<standardEndpoints>
<webHttpEndpoint>
<standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true" maxReceivedMessageSize="10000000" />
</webHttpEndpoint>
</standardEndpoints>
<behaviors>
<endpointBehaviors>
<behavior name="exampleBehavior">
<callbackDebug includeExceptionDetailInFaults="true" />
<enableWebScript />
<webHttp helpEnabled="true" />
</behavior>
</endpointBehaviors>
</behaviors>
<bindings>
<webHttpBinding>
<binding name="WebHttpBinding" maxReceivedMessageSize="10000000" />
</webHttpBinding>
</bindings>
<client>
<endpoint address="http://LOCALHOST/Utilities.AppService/API"
binding="webHttpBinding" bindingConfiguration="WebHttpBinding"
contract="Utilities.Common.API.IMyApi"
behaviorConfiguration="exampleBehavior" />
</client>
</system.serviceModel>
</configuration>
我的应用程序层 web.config:
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.0" />
<customErrors mode="Off" />
</system.web>
<system.serviceModel>
<serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" />
<standardEndpoints>
<webHttpEndpoint>
<standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true" maxReceivedMessageSize="10000000" />
</webHttpEndpoint>
</standardEndpoints>
</system.serviceModel>
<system.webServer>
<modules runAllManagedModulesForAllRequests="true" />
<handlers>
<remove name="ExtensionlessUrlHandler-Integrated-4.0" />
<add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="GET,HEAD,POST,DEBUG,PUT,DELETE" type="System.Web.Handlers.TransferRequestHandler" resourceType="Unspecified" requireAccess="Script" preCondition="integratedMode,runtimeVersionv4.0" />
</handlers>
</system.webServer>
</configuration>
我的服务界面
[ServiceContract(Namespace = "API")]
public interface IMyApi
{
[WebGet]
MyResponseContract GetFileInfo();
}
我的网络层实现
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiWebService : ClientBase<IMyApi>, IMyApi
{
public MyResponseContract GetFileInfo()
{
return Channel.GetFileInfo();
}
}
我的应用层实现
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiAppService : IMyApi
{
public MyResponseContract GetFileInfo()
{
return new MyResponseContract();
}
}
我的网络层Global.asax:
protected void Application_Start(object sender, EventArgs e)
{
RouteTable.Routes.Add(new ServiceRoute("API", new WebServiceHostFactory(), typeof(MyApiWebService)));
}
我的应用程序层 Global.asax:
protected void Application_Start(object sender, EventArgs e)
{
RouteTable.Routes.Add(new ServiceRoute("API", new WebServiceHostFactory(), typeof(MyApiAppService)));
}
我不确定我可以提供多少详细信息。如您所见,鉴于提供的解决方案,我已经实施了所有建议但无济于事。无论我是试图通过在浏览器中放置 Web 层服务 url 或使用 SOAPUI 来尝试此 WebGet
方法,还是尝试使用服务客户端通过 C# 单元测试来尝试它,我都无法使用 WebGet
。再次感谢您的帮助。
有趣的是,应用层 URL 有效。但是web层没有。所以:
localhost/Utilities.AppService/API/GetFileInfo
有效,而
localhost/Utilities.WebService/API/GetFileInfo
不会。
.NET WCF Web 服务默认设置为发送文本编码的 SOAP 消息。这意味着HTTP方法是POST并且需要headers来告诉服务调用什么方法。
我使用您的服务端点创建了一个快速示例,这里是 fiddler 生成的与该端点对话的请求。
POST http://localhost/Utilities/API/GetFileInfo/Service1.svc HTTP/1.1
Content-Type: text/xml; charset=utf-8
SOAPAction: "http://tempuri.org/IService1/GetFileInfo"
Host: localhost:8888
Content-Length: 136
Expect: 100-continue
Connection: Keep-Alive
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"><s:Body><GetFileInfo xmlns="http://tempuri.org/"/></s:Body></s:Envelope>
正在返回
的响应
HTTP/1.1 200 OK
Cache-Control: private
Content-Length: 398
Content-Type: text/xml; charset=utf-8
Server: Microsoft-IIS/8.0
X-AspNet-Version: 4.0.30319
X-SourceFiles: =?UTF-8?B?QzpcV29ya1xFSVAgV29ya1xRdWVzdGlvbnNcV0NGR2V0VGVzdFxTZXJ2aWNlMS5zdmM=?=
X-Powered-By: ASP.NET
Date: Thu, 21 May 2015 19:47:49 GMT
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"><s:Body><GetFileInfoResponse xmlns="http://tempuri.org/"><GetFileInfoResult xmlns:a="http://schemas.datacontract.org/2004/07/WCFGetTest" xmlns:i="http://www.w3.org/2001/XMLSchema-instance"><a:BoolValue>true</a:BoolValue><a:StringValue>MyResponseContract </a:StringValue></GetFileInfoResult></GetFileInfoResponse></s:Body></s:Envelope>
对于您,我认为您的 SoapUI 中的某些设置不正确。 post 数据或 headers.
所以在我最近更新之前这可能并不明显,但我有两个 RESTful 服务可以相互通信但位于不同的域中。 Web 层服务是第一个接触点,应用层服务是实际的工作执行者。在这种情况下,我能够进一步调试并发现实际的异常是从 Web 到 App 层的调用中的 405(方法不允许)。经过大量挖掘后,我发现 this link 解决了我的问题。
当使用 ClientBase<>
作为服务之间的通信方法时,您本质上需要在调用之间重新建立操作上下文。否则一切都变成 POST,因此,只有 POST 有效。
我希望这对其他人有所帮助,非常感谢大家帮助调试。
为了演示它的样子,这是我更新后的工作 Web 服务实现的样子:
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiWebService : ClientBase<IMyApi>, IMyApi
{
public MyResponseContract GetFileInfo()
{
MyResponseContract output = null;
using(var context = new OperationContext(Channel as IContextChannel))
{
output = Channel.GetFileInfo();
}
return output;
}
}
我有一个 RESTful WCF 网络服务,其中 API:
[WebGet(ResponseFormat = WebMessageFormat.Json)]
MyResponseContract GetFileInfo();
尝试访问端点(使用 SOAPUI)时,我看到以下错误消息:
The server encountered an error processing the request. Please see the service help page for constructing valid requests to the service.
我将 SOAPUI 设置为使用 GET 方法调用来命中它。当我将它切换到没有正文的 POST 时,它失败并显示以下消息:
Method not allowed.
这很有道理:不能用 POST 命中 GET。所以我更新了我的代码如下:
[WebInvoke(ResponseFormat = WebMessageFormat.Json)]
MyResponseContract GetFileInfo();
现在我使用 POST 方法从 SOAPUI 调用它并且它有效。好奇的。所以我现在更改我的代码如下:
[WebInvoke(ResponseFormat = WebMessageFormat.Json, Method = "GET")]
MyResponseContract GetFileInfo();
我在一些帖子中看到这基本上等同于 WebGet
属性。这也不行。
所以我的问题是:即使我不接受参数或不使用自定义 UriTemplate,为什么它不能作为 WebGet
工作?
我尝试使用的 Url(它本地托管在 IIS 中)是:
http://localhost/Utilities/API/GetFileInfo
更新 鉴于下面的评论和给出的答案,我仍然面临这个问题。一些额外的细节。
我的网络层web.config
<?xml version="1.0"?>
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.0" />
<customErrors mode="Off" />
</system.web>
<system.serviceModel>
<serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" />
<standardEndpoints>
<webHttpEndpoint>
<standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true" maxReceivedMessageSize="10000000" />
</webHttpEndpoint>
</standardEndpoints>
<behaviors>
<endpointBehaviors>
<behavior name="exampleBehavior">
<callbackDebug includeExceptionDetailInFaults="true" />
<enableWebScript />
<webHttp helpEnabled="true" />
</behavior>
</endpointBehaviors>
</behaviors>
<bindings>
<webHttpBinding>
<binding name="WebHttpBinding" maxReceivedMessageSize="10000000" />
</webHttpBinding>
</bindings>
<client>
<endpoint address="http://LOCALHOST/Utilities.AppService/API"
binding="webHttpBinding" bindingConfiguration="WebHttpBinding"
contract="Utilities.Common.API.IMyApi"
behaviorConfiguration="exampleBehavior" />
</client>
</system.serviceModel>
</configuration>
我的应用程序层 web.config:
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.0" />
<customErrors mode="Off" />
</system.web>
<system.serviceModel>
<serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" />
<standardEndpoints>
<webHttpEndpoint>
<standardEndpoint name="" helpEnabled="true" automaticFormatSelectionEnabled="true" maxReceivedMessageSize="10000000" />
</webHttpEndpoint>
</standardEndpoints>
</system.serviceModel>
<system.webServer>
<modules runAllManagedModulesForAllRequests="true" />
<handlers>
<remove name="ExtensionlessUrlHandler-Integrated-4.0" />
<add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="GET,HEAD,POST,DEBUG,PUT,DELETE" type="System.Web.Handlers.TransferRequestHandler" resourceType="Unspecified" requireAccess="Script" preCondition="integratedMode,runtimeVersionv4.0" />
</handlers>
</system.webServer>
</configuration>
我的服务界面
[ServiceContract(Namespace = "API")]
public interface IMyApi
{
[WebGet]
MyResponseContract GetFileInfo();
}
我的网络层实现
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiWebService : ClientBase<IMyApi>, IMyApi
{
public MyResponseContract GetFileInfo()
{
return Channel.GetFileInfo();
}
}
我的应用层实现
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiAppService : IMyApi
{
public MyResponseContract GetFileInfo()
{
return new MyResponseContract();
}
}
我的网络层Global.asax:
protected void Application_Start(object sender, EventArgs e)
{
RouteTable.Routes.Add(new ServiceRoute("API", new WebServiceHostFactory(), typeof(MyApiWebService)));
}
我的应用程序层 Global.asax:
protected void Application_Start(object sender, EventArgs e)
{
RouteTable.Routes.Add(new ServiceRoute("API", new WebServiceHostFactory(), typeof(MyApiAppService)));
}
我不确定我可以提供多少详细信息。如您所见,鉴于提供的解决方案,我已经实施了所有建议但无济于事。无论我是试图通过在浏览器中放置 Web 层服务 url 或使用 SOAPUI 来尝试此 WebGet
方法,还是尝试使用服务客户端通过 C# 单元测试来尝试它,我都无法使用 WebGet
。再次感谢您的帮助。
有趣的是,应用层 URL 有效。但是web层没有。所以:
localhost/Utilities.AppService/API/GetFileInfo
有效,而
localhost/Utilities.WebService/API/GetFileInfo
不会。
.NET WCF Web 服务默认设置为发送文本编码的 SOAP 消息。这意味着HTTP方法是POST并且需要headers来告诉服务调用什么方法。
我使用您的服务端点创建了一个快速示例,这里是 fiddler 生成的与该端点对话的请求。
POST http://localhost/Utilities/API/GetFileInfo/Service1.svc HTTP/1.1
Content-Type: text/xml; charset=utf-8
SOAPAction: "http://tempuri.org/IService1/GetFileInfo"
Host: localhost:8888
Content-Length: 136
Expect: 100-continue
Connection: Keep-Alive
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"><s:Body><GetFileInfo xmlns="http://tempuri.org/"/></s:Body></s:Envelope>
正在返回
的响应HTTP/1.1 200 OK
Cache-Control: private
Content-Length: 398
Content-Type: text/xml; charset=utf-8
Server: Microsoft-IIS/8.0
X-AspNet-Version: 4.0.30319
X-SourceFiles: =?UTF-8?B?QzpcV29ya1xFSVAgV29ya1xRdWVzdGlvbnNcV0NGR2V0VGVzdFxTZXJ2aWNlMS5zdmM=?=
X-Powered-By: ASP.NET
Date: Thu, 21 May 2015 19:47:49 GMT
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"><s:Body><GetFileInfoResponse xmlns="http://tempuri.org/"><GetFileInfoResult xmlns:a="http://schemas.datacontract.org/2004/07/WCFGetTest" xmlns:i="http://www.w3.org/2001/XMLSchema-instance"><a:BoolValue>true</a:BoolValue><a:StringValue>MyResponseContract </a:StringValue></GetFileInfoResult></GetFileInfoResponse></s:Body></s:Envelope>
对于您,我认为您的 SoapUI 中的某些设置不正确。 post 数据或 headers.
所以在我最近更新之前这可能并不明显,但我有两个 RESTful 服务可以相互通信但位于不同的域中。 Web 层服务是第一个接触点,应用层服务是实际的工作执行者。在这种情况下,我能够进一步调试并发现实际的异常是从 Web 到 App 层的调用中的 405(方法不允许)。经过大量挖掘后,我发现 this link 解决了我的问题。
当使用 ClientBase<>
作为服务之间的通信方法时,您本质上需要在调用之间重新建立操作上下文。否则一切都变成 POST,因此,只有 POST 有效。
我希望这对其他人有所帮助,非常感谢大家帮助调试。
为了演示它的样子,这是我更新后的工作 Web 服务实现的样子:
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Required)]
[ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)]
public class MyApiWebService : ClientBase<IMyApi>, IMyApi
{
public MyResponseContract GetFileInfo()
{
MyResponseContract output = null;
using(var context = new OperationContext(Channel as IContextChannel))
{
output = Channel.GetFileInfo();
}
return output;
}
}