asp.net - 防止负载均衡器在长时间运行期间超时

标签 asp.net asp.net-mvc load-balancing

我使用 ASP.NET MVC 4 接受最大 30 mb 的视频上传。视频上传到网站后,视频就会使用 HttpWebRequest 发布到第三方 Web 服务。向第三方服务的上传必须完成并返回响应,然后我的网站才能向浏览器返回响应。在 Rackspace 云站点上,负载均衡器有 30 秒的超时时间。如果负载均衡器在 30 秒内没有从我的网站返回响应,则负载均衡器将向浏览器返回超时错误,并终止请求。

所以我被告知需要发生的是,当我的网站将视频发送到第三方系统时,我的网站需要通过将数据发送回浏览器来保持连接处于事件状态,以便负载均衡器不会' t 超时。你将如何完成这个任务?

我正在寻找两件事 - 上传时我应该将什么作为数据发送回浏览器?我的实际响应是 JSON,所以如果我能在最后保留常规的 JSON 响应,那将是理想的。

最后,如何在发送 keep-alive 数据的同时向第三方执行 HttpWebRequest 上传?

-

作为引用,这是来自rackspace云的负载均衡器连接超时文档:http://www.rackspace.com/knowledge_center/article/connection-timed-out-error-message-1

他们没有为这个问题提供任何具体的解决方案。

最佳答案

这是我解决问题的方法。此解决方案应该适用于任何使用具有不合理超时的负载均衡器的 .net 托管。 Rackspace 云站点的负载均衡器有 30 秒的超时时间。 Amazon 和 Azure 的超时时间较长,但如果您碰巧有一个长时间运行的操作,这仍然可能是一个有用的解决方案。

在 asp.net mvc 4 中,您可以将 Controller 更改为从 AsyncController 继承。这允许您分拆异步任务,然后等待它们完成。此解决方案创建一个“保持事件”任务,该任务每 10 秒通过 asp.net 响应发送回数据,以便负载均衡器检测事件并且不会超时。此任务将一直运行,直到发现 uploadFinished 变量设置为 true。

另一个任务执行长时间运行的上传,或者在 ASP.NET MVC Controller 结束其正在处理的 Web 请求之前需要完成的任何任务。长时间运行的操作完成后,它将 uploadFinished 变量设置为 true。

最后,我围绕实际的 json 响应手动构建一个容器 json 对象,以便我可以将“保持事件”数据作为发送回浏览器的有效 json 的一部分。一个有点丑陋的黑客,但它有效!

using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using System.Web;
using System.Web.Mvc;
using System.Web.Script.Serialization;

namespace MvcApplication4.Controllers
{
    public class ExampleController : AsyncController
    {

        [HttpPost]
        public string Upload(UploadModel model, HttpPostedFileBase videoFile)
        {
            // when using jquery ajax form for upload
            // IE requires text/plain content type
            // otherwise it will try to open the response as a file
            // instead of pass the response back to your javascript
            if (Request.AcceptTypes.Contains("application/json"))
            {
                Response.ContentType = "application/json";
            }
            else
            {
                Response.ContentType = "text/plain";
            }

            // start json object
            Response.Write("{\"keepalive\":\"");
            Response.Flush();

            Task[] tasks = new Task[2];
            tasks[0] = Task.Factory.StartNew(() => DoKeepAlive());
            tasks[1] = Task.Factory.StartNew(() => DoUpload(model, videoFile));
            Task.WaitAll(tasks);

            // end keepalive json property
            Response.Write("\",\"data\":");

            // insert actual response data
            JavaScriptSerializer json = new JavaScriptSerializer();
            Response.Write(json.Serialize(uploadResponse));

            // end json object
            Response.Write("}");

            return "";
        }

        public bool uploadFinished = false;
        public UploadResponseModel uploadResponse = null;

        public void DoUpload(UploadModel model, HttpPostedFileBase videoFile)
        {
            // do upload to 3rd party
            MyServiceClient c = new MyServiceClient();
            uploadResponse = c.UploadVideo(model, videoFile);
            uploadFinished = true;
        }

        public void DoKeepAlive()
        {
            // send . every 10 seconds
            while (!uploadFinished)
            {
                Response.Write(".");
                Response.Flush();
                Thread.Sleep(10000);
            }
        }
    }
}

此解决方案依赖于 .Net 4 来完成异步工作。如果您有 .Net 4.5,则可以在不依赖于 AsyncController 类的 mvc Controller 中执行异步的更新方法。

这个站点很有帮助,解释了如何在 .Net 框架的各个版本中的 asp.net mvc 中执行异步操作:

http://dotnet.dzone.com/news/net-zone-evolution

关于asp.net - 防止负载均衡器在长时间运行期间超时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12736906/

相关文章:

具有持久 TCP 连接的 HAProxy 中的 TCP 负载平衡

c# - 线程被中止 - 同时循环了很多行

asp.net-mvc - 自动生成 ASP.NET MVC CRUD UI 的工具?

asp.net-mvc - asp.net MVC复选框令人头疼!

c# - 如何在 Asp.net MVC 中编写 OAuth2 Web API 客户端

load-balancing - 服务发现、负载平衡和连接池方法

http - Web 应用程序如何在 HTTP 代理或负载平衡器之后恢复原始 URL?

c# - 在 WCF 中利用 ASP.NET session

c# - 如何使用 Linq Lambda 检索 XML?

c# - 如何实现异步缓存?