我有一个Web Api控制器。它表现得很奇怪。当我使用PostMan时,我可以在Web Api上访问POST方法,但是当我从.net使用HttpWebRequest时,它返回(405)Method Not Allowed。 我把Web Api代码放在这里:
using Newtonsoft.Json;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Web;
using System.Web.Http;
namespace MyProject.Controllers
{
public class TestController : ApiController
{
[HttpPost]
public int buy(OrderResult value)
{
try
{
if (value.InvoiceDate != null && value.Result == 1)
{
return 0;
}
}
catch{}
return -1;
}
}
public class OrderResult
{
public int Result { get; set; }
public long InvoiceNumber { get; set; }
public string InvoiceDate { get; set; }
public long TimeStamp { get; set; }
}
}
这是我的WebApiConfig.cs:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web.Http;
namespace MyProject
{
public static class WebApiConfig
{
public static void Register(HttpConfiguration config)
{
config.Routes.MapHttpRoute(
name: "DefaultApi",
routeTemplate: "api/{controller}/{action}/{id}",
defaults: new { action = RouteParameter.Optional, id = RouteParameter.Optional }
);
}
}
}
这是我从另一个.NET项目发送POST请求的方式:
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Text;
using System.Web;
namespace MyProject.Controllers
{
public static class WebReq
{
public static string PostRequest(string Url, string postParameters)
{
try
{
HttpWebRequest myReq = (HttpWebRequest)WebRequest.Create(Url);
myReq.Method = "POST";
myReq.Timeout = 30000;
myReq.Proxy = null;
byte[] postData = Encoding.UTF8.GetBytes(postParameters);
myReq.ContentLength = postData.Length;
myReq.ContentType = "application/x-www-form-urlencoded";
using (Stream requestWrite = myReq.GetRequestStream())
{
requestWrite.Write(postData, 0, postData.Length);
requestWrite.Close();
using (HttpWebResponse webResponse = (HttpWebResponse)myReq.GetResponse())
{
if (webResponse.StatusCode == HttpStatusCode.OK)
{
using (Stream str = webResponse.GetResponseStream())
{
using (StreamReader sr = new StreamReader(str))
{
return sr.ReadToEnd();
}
}
}
return null;
}
}
}
catch (Exception e)
{
var message = e.Message;
return null;
}
}
}
}
我已经在web.config中添加了以下代码:
<modules runAllManagedModulesForAllRequests="true">
<remove name="WebDAVModule" />
</modules>
很奇怪,因为我可以从PostMan成功发送POST请求。 PostMan将此代码发送给API。
POST /api/Test/buy HTTP/1.1
Host: domain.com
Cache-Control: no-cache
Postman-Token: 9220a4e6-e707-5c5f-ea61-55171a5dd95f
Content-Type: application/x-www-form-urlencoded
InvoiceDate=28012016&Result=1
我将不胜感激任何解决问题的建议。
答案 0 :(得分:3)
我知道这是一个老帖子,但也许这可能对其他人有帮助。我刚刚遇到类似的问题,当我明显在邮递员中发帖时,得到405错误“不被允许”。原来,我使用http而不是https提交到URL。更改为https修复了它。
答案 1 :(得分:2)
byte[] postData = Encoding.UTF8.GetBytes(postParameters);
myReq.ContentLength = postData.Length;
myReq.ContentType = "application/x-www-form-urlencoded";
using (Stream requestWrite = myReq.GetRequestStream())
{
requestWrite.Write(postData, 0, postData.Length);
您很可能不会发送正确的x-www-form-urlencoded请求。您可能无法正确编码从postParameters传入的数据中的数据。见Post form data using HttpWebRequest
由于您未根据x-www-form-urlencoded生成有效的OrderResult对象,因此路由选择器不会选择您的Buy
操作。这就是你不允许POST的原因。
如果你改变了OrderResult value = null
,你可能会进入控制器,因为它现在是一个可选参数。然而,这并不是你想要的,除非你有一个奇怪的控制器,其行为如下:
Buy(OrderResult value = null)
{
if(value== null)
value = MyCustomDeserializeFromRequestBody(RequestContext)
...
}
最终你真的不应该使用这个课程,现代发展有更好的结构https://stackoverflow.com/a/31147762/37055 https://github.com/hhariri/EasyHttp在我的头顶
答案 2 :(得分:2)
我找到了解决方案。
我检查了Fiddler的请求。当我向API发送POST请求时,它会自动重定向到使用此新参数的同一地址AspxAutoDetectCookieSupport=1
How to remove AspxAutoDetectCookieSupport=1
最后,我将web.config中的cookieless="AutoDetect"
更改为cookieless="UseCookies"
,问题解决了。
答案 3 :(得分:0)
在我的情况下,我在项目中有一个与路径同名的物理文件夹(例如,沙箱),并且IIS中的静态文件处理程序(显然)而不是WebAPI运行时截获了任何POST请求。
得到一个误导性的405错误而不是更令人期待的404错误,这是我花了一些时间进行故障排除的原因。
不容易陷入这种情况,但是有可能。希望对别人有帮助。
答案 4 :(得分:-1)
用小提琴手或类似的东西抓住你的请求,你可能正在发送&#34; OPTIONS&#34;方法。
这与CORS有关,所以我认为你的解决方案是在WebAPI上启用Cors
http://www.asp.net/web-api/overview/security/enabling-cross-origin-requests-in-web-api