为了账号安全,请及时绑定邮箱和手机立即绑定

处理对ASP.NET MVC操作的CORS预检请求

处理对ASP.NET MVC操作的CORS预检请求

明月笑刀无情 2019-12-26 09:36:00
我正在尝试对ASP.NET MVC控制器操作执行跨域POST请求。该控制器动作接受并使用各种参数。问题在于,当发生预检请求时,控制器操作实际上会尝试执行&,因为OPTIONS请求没有传递任何数据,所以控制器操作会抛出500 HTTP错误。如果删除使用该参数或参数本身的代码,则整个请求链将成功完成。如何实现此示例:控制器动作public ActionResult GetData(string data){    return new JsonResult    {        Data = data.ToUpper(),        JsonRequestBehavior = JsonRequestBehavior.AllowGet    };}客户端代码<script type="text/javascript">        $(function () {            $("#button-request").click(function () {                var ajaxConfig = {                    dataType: "json",                    url: "http://localhost:8100/host/getdata",                    contentType: 'application/json',                    data: JSON.stringify({ data: "A string of data" }),                    type: "POST",                    success: function (result) {                        alert(result);                    },                    error: function (jqXHR, textStatus, errorThrown) {                        alert('Error: Status: ' + textStatus + ', Message: ' + errorThrown);                    }                };                $.ajax(ajaxConfig);            });        });    </script>现在,每当发生预检请求时,它都会返回500个HTTP代码,因为“ data”参数为null,因为OPTIONS请求未传递任何值。服务器应用程序已在端口8100的本地IIS中设置,并且在8200端口上设置了运行客户端代码的页面,以模拟跨域调用。我还使用以下标头配置了主机(在8100上):Access-Control-Allow-Headers: Content-TypeAccess-Control-Allow-Methods: POST, GETAccess-Control-Allow-Origin: http://localhost:8200我发现的一种解决方法是检查执行该操作的HTTP方法,如果这是一个仅返回空白内容的OPTIONS请求,则执行该操作代码。像这样:public ActionResult GetData(string data){    if (Request.HttpMethod == "OPTIONS") {        return new ContentResult();    } else {        return new JsonResult        {            Data = data.ToUpper(),            JsonRequestBehavior = JsonRequestBehavior.AllowGet        };    }}但是这种方法对我来说很笨拙。我考虑过将这种逻辑添加到中Attribute,但这甚至意味着装饰将使用CORS调用的每个动作。是否有更优雅的解决方案来使此功能正常工作?
查看完整描述

3 回答

?
MYYA

TA贡献1868条经验 获得超4个赞

因此,我找到了可行的解决方案。对于每个请求,我都会检查它是否是CORS请求,以及该请求是否与OPTIONS动词一起出现,表明它是预检请求。如果是这样,我只发送回一个空响应(当然只包含在IIS中配置的标头),从而使控制器动作执行无效。


然后,如果客户端根据预检返回的标头确认允许执行请求,则执行实际的POST并执行控制器动作。我的代码示例:


protected void Application_BeginRequest()

{

    if (Request.Headers.AllKeys.Contains("Origin", StringComparer.OrdinalIgnoreCase) &&

        Request.HttpMethod == "OPTIONS") {

        Response.Flush();

    }

}

如前所述,这对我有用,但是如果有人知道更好的方法或我当前实现中的任何缺陷,我将不胜感激。


查看完整回答
反对 回复 2019-12-26
?
慕标琳琳

TA贡献1830条经验 获得超9个赞

接受的答案就像一个超级按钮一样工作,但是我发现请求实际上已经传递给了控制器。我正在接收200状态代码,但是响应主体包含许多HTML,但控制器中有异常。因此Response.Flush(),我发现最好使用而不是使用Response.End(),它确实会停止执行请求。此替代解决方案如下所示:


编辑:修复了原始答案中的错字。


protected void Application_BeginRequest()

{

    if (Request.Headers.AllKeys.Contains("Origin", StringComparer.OrdinalIgnoreCase) &&

        Request.HttpMethod == "OPTIONS") {

        Response.End();

    }

}


查看完整回答
反对 回复 2019-12-26
  • 3 回答
  • 0 关注
  • 618 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信