我正在尝试使用ServeContent来提供文件(可能是大型电影文件,所以它会使用字节范围),但我不知道如何处理修改时间。如果我使用以下程序来播放电影,如果我给出文件的实际修改时间,则会失败,如图所示。我认为发生的事情是第一个请求有效,但后续的(文件的不同字节范围)认为它已经有文件,因此它们失败并且电影不能播放。有什么我做错了吗?
请注意,如果我使用time.Now()而不是文件的实际修改时间,代码可以正常工作(并且电影可以正常播放),但当然这是不正确的。
package main
import (
"fmt"
"net/http"
"os"
"path"
"time"
)
func main() {
http.HandleFunc("/", handler)
http.ListenAndServe(":3000", nil)
}
func handler(w http.ResponseWriter, r *http.Request) {
filePath := "." + r.URL.Path
file, err := os.Open(filePath)
if err != nil {
fmt.Printf("%s not found\n", filePath)
w.WriteHeader(http.StatusNotFound)
fmt.Fprint(w, "<html><body style='font-size:100px'>four-oh-four</body></html>")
return
}
defer file.Close()
fileStat, err := os.Stat(filePath)
if err != nil {
fmt.Println(err)
}
fmt.Printf("serve %s\n", filePath)
_, filename := path.Split(filePath)
t := fileStat.ModTime()
fmt.Printf("time %+v\n", t)
http.ServeContent(w, r, filename, t, file)
}
答案 0 :(得分:2)
根据文件,
如果modtime不是零时间,则ServeContent将其包含在响应中的Last-Modified标头中。如果请求包含If-Modified-Since标头,则ServeContent使用modtime来决定是否需要发送内容。
因此,根据客户端是否发送If-Modified-Since
标头,此功能将正常运行。这似乎是预期的行为,在正常情况下确实有用,可以优化服务器的带宽。
但是,在您的情况下,由于您必须处理部分内容请求,除非第一个请求返回30X HTTP代码,否则您没有理由为后续请求处理此机制。
禁用此行为的正确方法是将“零”日期传递给ServeContent
:
http.ServeContent(w, r, filename, time.Time{}, file)
您可以尝试解析请求范围标头,以便仅在必要时传递零日期。