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-Range
和 Accept-Range
值(否则搜索将是不可能的,我猜你不想要那个)。这应该让 Chrome 高兴。在处理请求中的Range
标头值时要小心——如果不是0-
,那么您必须实际寻找到请求的点并从那里返回数据。此外,我认为不应该将Transfer-Encoding
设置为chunked
的视频传输。
【讨论】:
以上是关于GoLang http webserver 提供视频 (mp4)的主要内容,如果未能解决你的问题,请参考以下文章
WEBserver应用程序serverHTTPserver差别