GoLang http webserver 提供视频 (mp4)

Posted

技术标签:

【中文标题】GoLang http webserver 提供视频 (mp4)【英文标题】:GoLang http webserver provide video (mp4) 【发布时间】:2016-10-30 09:21:26 【问题描述】:

我使用 golang 开发了一个网络服务器。漂亮的平面东西,它只提供了 html/js/css 和图像,效果很好:

func main() 
    http.Handle("/", new(viewHandler))
    http.ListenAndServe(":8080", nil)


func (vh *viewHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) 
    path := r.URL.Path[1:]
    log.Println(path)
    data, err := ioutil.ReadFile(string(path))

    if err == nil 

        var contentType string

        if strings.HasSuffix(path, ".html") 
            contentType = "text/html"
         else if strings.HasSuffix(path, ".css") 
            contentType = "text/css"
         else if strings.HasSuffix(path, ".js") 
            contentType = "application/javascript"
         else if strings.HasSuffix(path, ".png") 
            contentType = "image/png"
         else if strings.HasSuffix(path, ".jpg") 
            contentType = "image/jpeg"
         

        w.Header().Add("Content-Type", contentType)
        w.Write(data)
     else 
        log.Println("ERROR!")
        w.WriteHeader(404)
        w.Write([]byte("404 - " + http.StatusText(404)))
    

我尝试以相同的方式将 mp4 格式的视频添加到我的网站:

<video   autoplay loop style="margin-top: 20px">
   <source src="../public/video/marketing.mp4" type="video/mp4">Your browser does not support the video tag.</video>

并通过以下方式增强了我的 go 应用程序的内容类型部分:

else if strings.HasSuffix(path, ".mp4") 
            contentType = "video/mp4"
        

当我直接在 Chrome 中打开 html 文件时,视频可以正常播放,但如果我通过在 http://localhost 调用 Web 服务器打开网站...则无法加载,因此无法播放。

没有加载视频,如果我尝试直接点击它,浏览器只会显示一些视频控制而不加载文件:

对此有什么想法吗?提前致谢。

更新:

按照建议,我还尝试了另一个运行良好的视频!但我不知道为什么,我比较了两个视频:

会不会是尺码??

干杯,谢谢!

更新2:

icza 是对的,我将他的帖子标记为对我的问题的正确答案。但让我解释一下我最终做了什么来让它工作:

我试图在不使用 http.FileServe 方法的情况下解决我的问题。因此我是这样实现的:

 else if strings.HasSuffix(path, ".mp4") 
            contentType = "video/mp4"
            size := binary.Size(data)
            if size > 0 
                requestedBytes := r.Header.Get("Range")
                w.Header().Add("Accept-Ranges", "bytes")
                w.Header().Add("Content-Length", strconv.Itoa(size))
                w.Header().Add("Content-Range", "bytes "+requestedBytes[6:len(requestedBytes)]+strconv.Itoa(size-1)+"/"+strconv.Itoa(size))
                w.WriteHeader(206)
            
        

        w.Header().Add("Content-Type", contentType)
        w.Write(data)

这在我第一次播放视频时奏效了。但由于我希望它循环播放,它应该直接从头开始,但它卡在视频的最后一帧。

所以我实现了 ServeFile() 方法,例如:

if contentType == "video/mp4" 
            http.ServeFile(w, r, path)
         else 
            w.Header().Add("Content-Type", contentType)
            w.Write(data)
        

视频现在也可以正确循环播放。但是我仍然不知道为什么 ServeFile() 正在工作,而另一种方法却没有,尽管两种方法的响应完全相同。不过现在的内容是这样的:

干杯并感谢所有帮助过我的人。

【问题讨论】:

您拥有的代码对我来说似乎工作正常。您可能需要详细说明您看到的具体问题。 (您说“无法加载,因此无法播放”,但我不确定您的意思。请具体说明您遇到的问题。) 您还没有描述您看到的实际问题。有错误吗?如果是这样,您在哪里看到该错误,该错误说明了什么? 您应该使用支持服务范围请求的http.ServeFile() 提供您的文件(视频需要能够搜索)。查看相关问题:How to serve http partial content with Go? @MichaelB 只需运行 wireshark 即可查看具体情况;) 如果 chrome 决定使用 Range 标头而不是 0- 发送另一个请求,那么这当然是问题所在。跨度> @MichaelB 也在新标签中打开 chrome://media-internals/ - 显示所有打开的媒体接收器的内部信息。您很可能能够看到到底出了什么问题。 【参考方案1】:

您的“问题”与视频长度/大小有关。我不知道 Chrome 使用的默认缓冲区(它可能取决于多种因素,例如可用内存、可用磁盘空间等),但您不应该依赖它来播放视频!

您的一个视频短了几秒钟,几乎是另一个视频的 3 倍。 Chrome 完全缓冲较小的视频(可能在内存中),而对另一个不做同样的事情。而且由于它比缓冲区大,它需要服务器支持范围请求(部分内容服务)。您提供视频内容的方式不支持部分内容服务(Chrome 期望在“大”视频的情况下提供),因此 Chrome 只是拒绝处理/播放视频。

您应该使用支持服务范围请求的http.ServeFile() 提供文件(对于大于特定大小的视频是必需的)。查看相关问题:How to serve http partial content with Go?

执行此操作后,您的两个视频文件都可以正常播放。

展望未来,没有实际理由不使用http.ServeFile()。它不会像您那样将完整的文件加载到内存中,这对于大(视频)文件来说是一个非常糟糕的主意。 http.ServeFile() 处理 Range 请求(如果有要求),并正确设置响应标头,包括 Content-Type(并且知道 .mp4 文件需要 video/mp4 MIME 类型)。

分析大视频文件的请求

测试这个简单的应用程序:

func fileh(w http.ResponseWriter, r *http.Request) 
    http.ServeFile(w, r, "/path/to/a/big/video.mp4")    


func main() 
    http.HandleFunc("/", fileh)
    panic(http.ListenAndServe("localhost:8080", nil))

从 Chrome 中打开 http://localhost:8080,浏览器发出 2 个请求(我故意过滤掉了不重要/不相关的请求-响应标头):

请求 #1

GET / HTTP/1.1

获取请求的(根)路径。

响应 #1

HTTP/1.1 200 OK
Accept-Ranges: bytes
Content-Length: 104715956
Content-Type: video/mp4

如您所见,Go 应用报告视频大小约为 100 MB。 http.ServeFile() 还包含响应标头Accept-Ranges: bytes,它让客户知道我们支持 Range 请求。

Chrome 对此有何反应?在收到 32 KB(包括标头在内的 32.2 KB;这足以查看流并告诉它可以用它做什么)后关闭连接,然后触发另一个请求(即使未发送 Accept-Ranges 响应标头也会发生这种情况- 你的示例代码没有发送它):

请求 #2

GET / HTTP/1.1
Referer: http://localhost:8080/
Range: bytes=0-

Chrome 会触发 Range 请求(从第一个字节开始询问所有内容),因此如果您的 Go 应用程序不支持此功能,那么您就有麻烦了。请参阅响应 #2。

响应 #2

HTTP/1.1 206 Partial Content
Accept-Ranges: bytes
Content-Length: 104715956
Content-Range: bytes 0-104715955/104715956
Content-Type: video/mp4

http.ServeFile() 表现良好,可以正确响应 Range 请求(HTTP 206 Partial Content 状态代码和 Content-Range 标头),并发送请求的范围(本例中的所有内容)。

您的示例代码没有以HTTP 206 Partial Content 响应,而是以简单的HTTP 200 OK 响应。当 Chrome 收到意外的HTTP 200 OK 时,这就是它放弃播放视频的关键。如果视频文件很小,Chrome 也不会做太多,因为它也会接受HTTP 200 OK 响应(并且可能只是将小视频存储在内存或缓存文件中)。

【讨论】:

谢谢!这很有帮助。我将您的陈述标记为正确答案。【参考方案2】:

您的代码对我来说似乎工作正常。这是我在 OS X 上使用 Chrome 测试的完整工作示例。

main.go:

package main

import "io/ioutil"
import "log"
import "net/http"
import "strings"

type viewHandler struct

func (vh *viewHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) 
    path := r.URL.Path[1:]

    data, err := ioutil.ReadFile(string(path))

    if err != nil 
        log.Printf("Error with path %s: %v", path, err)
        w.WriteHeader(404)
        w.Write([]byte("404"))
    

    if strings.HasSuffix(path, ".html") 
        w.Header().Add("Content-Type", "text/html")
     else if strings.HasSuffix(path, ".mp4") 
        w.Header().Add("Content-Type", "video/mp4")
    

    w.Write(data)


func main() 
    http.Handle("/", new(viewHandler))
    http.ListenAndServe(":8080", nil)

index.html:

<!doctype html>
<html>
<body>
    <video autoplay loop>
        <source src="big_buck_bunny.mp4" type="video/mp4">
    </video>
</body>
</html>

从http://www.quirksmode.org/html5/videos/big_buck_bunny.mp4下载的视频。

【讨论】:

你完全正确!我从未尝试过其他视频,但它确实有效。奇怪的是,我不知道为什么……我更新了我的帖子并添加了两张新图片。感谢您的帮助【参考方案3】:

这是一个支持所有文件类型的示例。 contentType := http.DetectContentType(buffer) https://play.golang.org/p/XuVgmsCHUZB

【讨论】:

【参考方案4】:

好吧,因为它不是实时流(因为大小和持续时间是已知的,因为源是 mp4 文件)我认为您的服务器应该尊重请求中的 Range 标头并以正确的 Content-Length 响应, Content-RangeAccept-Range 值(否则搜索将是不可能的,我猜你不想要那个)。这应该让 Chrome 高兴。在处理请求中的Range 标头值时要小心——如果不是0-,那么您必须实际寻找到请求的点并从那里返回数据。此外,我认为不应该将Transfer-Encoding 设置为chunked 的视频传输。

【讨论】:

以上是关于GoLang http webserver 提供视频 (mp4)的主要内容,如果未能解决你的问题,请参考以下文章

C++ 手写一个WebServer

WEBserver应用程序serverHTTPserver差别

如何使用 http 而不是 https 在 Heroku 上提供 Golang 应用程序?

理解Tomcat工作原理

请教一下webserver怎么用

Java类WebServer及中间件拿webshell方法总结