【问题标题】:Video streaming from database来自数据库的视频流
【发布时间】:2013-11-06 10:37:16
【问题描述】:

我正在尝试执行操作以显示来自数据库的媒体 (ASP.NET MVC4):

var memoryStream = new MemoryStream(mediaContent.File.FileData.Data);

return new FileStreamResult(memoryStream, MimeMapping.GetMimeMapping(mediaContent.File.Filename));

图片显示正常,但当我要在 Chrome 或 Firefox 中链接 mysite/media/4 时,视频 (.avi) 出现问题:

<embed width="100%" height="100%" name="plugin" src="http://mysite/media/4" type="video/x-msvideo">

但是视频没有播放(如果链接指向真实的视频文件会发生这种情况),但是如果我在 IE 中打开此链接,它会提示我下载文件,当我从播放器打开此文件时它工作正常。

响应标头:

Cache-Control:private, s-maxage=0
Content-Length:808680
Content-Type:video/x-msvideo
Date:Wed, 06 Nov 2013 10:03:09 GMT
Server:Microsoft-IIS/7.5
X-AspNet-Version:4.0.30319
X-AspNetMvc-Version:4.0
X-MiniProfiler-Ids:["e305dcdb-79be-4452-94d2-a9999ffaa13a","c0c81d12-8b31-425c-a57b-2ad186c958d5","1f7f3c09-a695-49f1-9203-6b5bf44b837a","fb0d637e-5926-4759-ad6f-f7322403e98c","f08c0392-10d6-4477-b2df-be52ab9a1d64","366d6122-15a5-41b4-840a-607fc6931996","11fd2eb7-efce-47a1-96f8-09fbdb0b1fa0","690e67b7-b1fb-46a3-9aa3-ef6207203f55","a51640ad-f31d-4f12-a807-6ea06ba0ee46","38adc052-9c41-4243-97d2-41dbf3b36093","9d255225-c122-44ef-8021-5b6f9d4dd549","2b249ff3-9e37-43c3-b6ab-b78b26c6d6ce","2bec0b1b-4898-4b14-bf12-cc331e27ecfc","49c72e01-c8d4-495f-af7e-8ffd687e94e9","1c87e454-f90d-49f4-9618-8dfe0d9c0329","2152a9a8-54ae-47d8-b98a-83ac32dbdb0c","9cf93254-9552-4834-826e-df7e8a7d8e73","a2d782e2-96ca-4e9c-b612-9782a37a06ca","e10ecc8a-5811-4cca-b566-3f09e1de3f2c","3769bb15-60f9-43c3-ad6c-285f3fb47112","1996c4aa-9f76-4f33-95fa-3f7f5b3e72f4"]
X-Powered-By:ASP.NET

我做错了什么?我想要一个可以在&lt;object&gt; 中使用的链接,以便在页面上显示此视频。

更新 1: 如果我输入物理视频文件的 url,我得到的响应没有任何意义:

HTTP/1.1 304 Not Modified
Last-Modified: Tue, 05 Nov 2013 17:07:56 GMT
Accept-Ranges: bytes
ETag: "5d50369249dace1:0"
Server: Microsoft-IIS/7.5
X-Powered-By: ASP.NET
Date: Wed, 06 Nov 2013 11:20:32 GMT

但它可以使用 intaled VLC 播放器插件在浏览器中开始播放视频。

更新 2: 我尝试了不同的实现来返回视频并尝试将链接放到srcobject

<object id="video-player" class="preview-container" type="video/x-msvideo" src="{link to video}" loop="true" controls="false" autoplay="true"></object>

因此,如果链接到真实视频,例如“localhost/media/some_video.avi” 那么它在object 内工作正常,如果直接访问 url。

我检查了不同实现的行为

1) return File(memoryStream, MimeMapping.GetMimeMapping(mediaContent.File.Filename), mediaContent.File.Filename);

to url:提示下载视频文件; 在对象中:显示空插件;

2) return new FileContentResult(mediaContent.File.FileData.Data, "application/x-vlc-plugin")

to url:显示空插件; 在对象中:显示空插件;

3)return new FileContentResult(mediaContent.File.FileData.Data, "video/avi")

to url:提示下载视频文件; 在对象中:显示空插件;

更新 3: 我做了一个HttpHandler:

public void ProcessRequest(HttpContext context)
{
    MediaContent mediaContent;
    //getting mediaContent

    context.Response.Clear();
    context.Response.Buffer = true;
    context.Response.AppendHeader("Content-Disposition", "inline; filename=" + mediaContent.File.Filename);
    context.Response.ContentType = MimeMapping.GetMimeMapping(mediaContent.File.Filename);
    context.Response.BinaryWrite(mediaContent.File.FileData.Data);
    context.Response.End();
}

而且它很有效。行。然后我用与 Handler 相同的逻辑进行操作:

[HttpGet]
public void Media(int id)
{
    MediaContent mediaContent;
    //getting mediaContent

    Response.Clear();
    Response.Buffer = true;
    Response.AppendHeader("Content-Disposition", "inline; filename=" + mediaContent.File.Filename);
    Response.ContentType = MimeMapping.GetMimeMapping(mediaContent.File.Filename);
    Response.BinaryWrite(mediaContent.File.FileData.Data);
    Response.End();
}

但此操作仍然不适用于视频,我开始比较响应标头。

处理程序:

HTTP/1.1 200 OK
Cache-Control: private
Content-Type: video/x-msvideo
Server: Microsoft-IIS/7.5
Content-Disposition: inline; filename=Reebok_App_attract640L.avi
X-AspNet-Version: 4.0.30319
X-Powered-By: ASP.NET
Date: Wed, 06 Nov 2013 17:02:45 GMT
Content-Length: 808680

行动:

HTTP/1.1 200 OK
Cache-Control: private, s-maxage=0
Content-Type: video/x-msvideo
Transfer-Encoding: chunked
Server: Microsoft-IIS/7.5
X-AspNetMvc-Version: 4.0
Content-Disposition: inline; filename=Reebok_App_attract640L.avi
X-AspNet-Version: 4.0.30319
X-Powered-By: ASP.NET
Date: Wed, 06 Nov 2013 17:02:47 GMT

为了摆脱 Transfer-Encoding,我在我的操作中添加了 Content-Length:

Response.AddHeader("Content-Length", mediaContent.File.FileData.Data.Length.ToString());

我无法摆脱 s-maxage=0,但现在标题相似(s-maxage=0X-AspNetMvc-Version: 4.0 和标题顺序除外)

【问题讨论】:

  • 您是否将 http 响应与同类型视频的有效实现进行了比较? (注意:这种方法只适用于简单的静态视频;不适用于自适应流)。我不知道它是否会有所帮助,但是您是否尝试过添加内联的内容处置?
  • 补充思想:avi (video/x-msvideo) 是一种可以在内部使用一系列编解码器的伞式格式;编解码器适合浏览器吗?
  • 感谢 cmets。我尝试添加 Response.AppendHeader("Content-Disposition", "inline; filename=" + mediaContent.File.Filename);但它不起作用
  • 您可能需要一个自定义控制器操作来仅播放视频而不是下载。检查这个devcurry.com/2013/04/play-videos-in-aspnet-mvc-html5-using.html

标签: c# asp.net sql-server-2008 asp.net-mvc-4 video


【解决方案1】:

好吧,我已尽力镜像您的初始代码,减去数据库,下面的内容对我来说很好。 VS2012,MVC4,IIS Express 本地,IIS7.5 远程,适用于 IE10 本地,IE10 远程,IE9 远程。我使用的 AVI 文件是我在 windows\winsxs 文件夹中找到的一些随机文件。我将建议您遇到客户端问题(特别是 IE)。可能是 cookie 处理问题 (http://mvolo.com/iis-70-forms-authentication-and-embedded-media-players/)、IE 安全区域设置或其他问题?

顺便说一句,嵌入时 height=100% 对我不起作用,需要像素。

控制器

namespace MvcApplication4.Controllers
{
    public class HomeController : Controller
    {

        public ActionResult Index()
        {
            return new ViewResult();
        }

        public ActionResult Media(int id)
        {
            string fn = Server.MapPath("~/App_Data/boxed-delete.avi");
            var memoryStream = new MemoryStream(System.IO.File.ReadAllBytes(fn));
            return new FileStreamResult(memoryStream, MimeMapping.GetMimeMapping(System.IO.Path.GetFileName(fn)));
        }

    }
}

查看

@{
    Layout = null;
}

<!DOCTYPE html>

<html>
<head>
    <meta name="viewport" content="width=device-width" />
    <title>Index</title>
</head>
<body>
    <div>
        <!-- http://localhost:54941/Home/Media/3  -->
    <embed width="100%" height="500"  name="plugin" src="~/Home/Media/3" type="video/x-msvideo">     
    </div>
</body>
</html>

【讨论】:

  • 我没有提供,但是我尝试实现我的操作的控制器被标记为 AuthorizeAttribute,这是一个问题,用 AllowAnnonymousAttribute 标记这个操作可以解决这个问题。非常感谢 - 您提供的链接为我指明了正确的方向
  • File.ReadAllBytes 将整个文件加载到内存中。如果这对您有用,您根本不需要使用流,只需返回包含字节数组的 FileContentResult。如果您想流式传输文件而不将其加载到内存中,则需要改用 File.OpenRead。
【解决方案2】:

您可以将存储在数据库中的文件位置放入字节缓冲区数组中,并以块的形式从数组中读取,而不是一次读取整个数组

WebRequest wreq = (HttpWebRequest)WebRequest.Create(url);
using (HttpWebResponse wresp = (HttpWebResponse)wreq.GetResponse())
using (Stream mystream = wresp.GetResponseStream())
{
  using (BinaryReader reader = new BinaryReader(mystream))
  {
    int length = Convert.ToInt32(wresp.ContentLength);
    byte[] buffer = new byte[length];
    buffer = reader.ReadBytes(length);

    Response.Clear();
    Response.Buffer = false;
    Response.ContentType = "video/mp4";
    while(true) {
    int bytesRead = myStream.Read(buffer, 0, buffer.Length);
    if (bytesRead == 0) break;
    Response.OutputStream.Write(buffer, 0, bytesRead);
}
    Response.End();
  }
}

您还可以查看它有帮助的给定网址天气, http://www.devcurry.com/2013/04/play-videos-in-aspnet-mvc-html5-using.html

谢谢

【讨论】:

    猜你喜欢
    • 2013-10-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多