我的测试应用程序是使用Web Api使用Vs 2013 Web应用程序空模板开发的。
这是我的路由初始化的样子:
public static class WebApiConfig
{
public static void Register(HttpConfiguration config)
{
// Web API configuration and services
config.EnableCors();
// Web API routes
config.MapHttpAttributeRoutes();
}
}
这就是我的控制器的样子:
[EnableCors(origins: "*", headers: "*", methods: "*")]
public class TestController : ApiController
{
[Route ("Test")]
[HttpGet]
public string Get()
{
return "Response to get Request";
}
[Route("Test")]
[HttpPost]
public string Post(string text)
{
return string.Format("Received: '{0}'", text);
}
}
这是我的web.config文件在包含所有建议修复后的样子:
<system.web>
<compilation debug="true" targetFramework="4.5.1" />
<httpRuntime targetFramework="4.5.1" />
<webServices>
<protocols>
<add name="HttpGet" />
<add name="HttpPost" />
<add name="HttpPut" />
<add name="HttpDelete" />
</protocols>
</webServices>
</system.web>
<system.webServer>
<handlers>
<remove name="WebDAV" />
<remove name="OPTIONSVerbHandler" />
<remove name="TRACEVerbHandler" />
<remove name="ExtensionlessUrlHandler-ISAPI-4.0_32bit" />
<remove name="ExtensionlessUrlHandler-ISAPI-4.0_64bit" />
<remove name="ExtensionlessUrlHandler-Integrated-4.0" />
<add name="ExtensionlessUrlHandler-ISAPI-4.0_32bit" path="*." verb="GET,HEAD,POST,DEBUG,PUT,DELETE,PATCH,OPTIONS" modules="IsapiModule" scriptProcessor="%windir%\Microsoft.NET\Framework\v4.0.30319\aspnet_isapi.dll" preCondition="classicMode,runtimeVersionv4.0,bitness32" responseBufferLimit="0" />
<add name="ExtensionlessUrlHandler-ISAPI-4.0_64bit" path="*." verb="GET,HEAD,POST,DEBUG,PUT,DELETE,PATCH,OPTIONS" modules="IsapiModule" scriptProcessor="%windir%\Microsoft.NET\Framework64\v4.0.30319\aspnet_isapi.dll" preCondition="classicMode,runtimeVersionv4.0,bitness64" responseBufferLimit="0" />
<add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="GET,HEAD,POST,DEBUG,PUT,DELETE,PATCH,OPTIONS" type="System.Web.Handlers.TransferRequestHandler" preCondition="integratedMode,runtimeVersionv4.0" />
</handlers>
<security>
<requestFiltering>
<verbs allowUnlisted="false">
<add verb="GET" allowed="true" />
<add verb="POST" allowed="true" />
<add verb="DELETE" allowed="true" />
<add verb="PUT" allowed="true" />
</verbs>
</requestFiltering>
</security>
<modules runAllManagedModulesForAllRequests="true">
<remove name="WebDAVModule" />
</modules>
</system.webServer>
我已经使用IIS管理器验证了处理程序映射和请求过滤我的应用程序,并且POST动词显示为已启用。
我已经跟踪了ISS拒绝的请求,我发现ManagedPipeLineHandler设置了405错误,如下所示:
ModuleName ManagedPipelineHandler
通知128
HttpStatus 405
不允许使用HttpReason方法
HttpSubStatus 0
ErrorCode 0
ConfigExceptionInfo
通知EXECUTE_REQUEST_HANDLER
ErrorCode操作成功完成。 (为0x0)
我的假设是请求没有转到我的应用程序。我对么?
我一直在使用Fiddler测试,我发现响应标题有一个名为&#39; Security&#39;用一行阅读&#39;允许:GET&#39;,这可以解释为什么Get工作但POST没有。
这些信息来自哪里?这个问题的答案可能会让我找到解决方案。
我花了很多时间试图解决这个问题而没有运气。任何反馈将不胜感激。
答案 0 :(得分:0)
我一直在假设'不允许'来自IIS,似乎我错了。
My Post方法有一个string类型的参数。我假设,如果我的路由属性不包含如下所示的参数,则参数将从请求体中获取。
[Route("Test")]
[HttpPost]
public string Post(string text)
为了看看发生了什么,我在此参数中添加了[FromBody]属性。令我惊讶的是,我得到了415错误而不是405错误。
对415错误的研究把我带到了这个链接:
该链接告诉我字符串参数应该是uri的一部分,并且正文中的字符串与内容类型的应用程序/文本一起发送时没有默认的格式化程序。
我将字符串参数封装在json对象中,如我的新控制器所示,现在POST工作了。
[EnableCors(origins: "*", headers: "*", methods: "*")]
public class TestController : ApiController
{
[Route ("Test")]
[HttpGet]
public string Get()
{
return "Response to get Request";
}
[Route("Test")]
[HttpPost]
public string Post(StringAsJson stringAsJason)
{
return string.Format("Received: '{0}'", stringAsJason.Value);
}
}
public class StringAsJson
{
public string Value { get; set; }
}
我回到原来的程序并尝试做同样的事情,但我仍然得到405错误;不同的是,现在我知道它不是IIS问题。
如果我不能让我的程序工作,我会打开另一个案例。