问题: 想尝试一下golang的http2功能,但是遇到一个问题, js文件无法正常解析...
go version go1.10.2 windows/amd64
详情: 后端代码
func main() {
var server http.Server
http2.VerboseLogs = true
server.Addr = ":8080"
http2.ConfigureServer(&server, &http2.Server{})
// 先把css和脚本服务上去
http.Handle("/static/",http.StripPrefix("/static/", http.FileServer(http.Dir("static/"))))
http.Handle("/views/",http.StripPrefix("/views/", http.FileServer(http.Dir("static/views/"))))
http.HandleFunc("/sayHello", sayHello)
logger.Info("正在启动服务器...")
// 启用http2
err := server.ListenAndServeTLS("./keystore/cert.crt", "./keystore/rsa_private.key") //前端不能正常解析js文件
//err := server.ListenAndServe() // 前端可以正常解析js文件
if err != nil {
logger.Errorf("服务器启动错误:%v", err)
}
}
func sayHello(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello world"))
}
前端代码:
<!DOCTYPE html>
<html lang="zh">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<div>
Hello world!
</div>
<script src="/static/js/vue.js" type="text/javascript"></script>
<script>
</script>
</body>
</html>
项目结构:
不能正常解析的情况
前端错误信息:
Refused to execute script from 'https://localhost:8080/static/js/vue.js' because its MIME type ('text/plain') is not executable, and strict MIME type checking is enabled.
注:
err := server.ListenAndServeTLS("./keystore/cert.crt", "./keystore/rsa_private.key") //用这一句,前端不能正常解析js文件
err := server.ListenAndServe() // 用这一句, 前端可以正常解析js文件
有疑问加站长微信联系(非本文作者)

各位大佬请帮忙指点一下, 十分感激
怎么都有没有人回复,是我的问题没有问清楚么
试试别用localhost,配置一个本地域名尝试
尝试了直接用ip访问, 或者改hosts用域名访问都不行
会不是是因为我用的是自签证书的问题
Refused to execute script from 'https://localhost:8080/static/js/vue.js' because its MIME type ('text/plain') is not executable, and strict MIME type checking is enabled. 翻译一下就是 拒绝执行脚本,因为类型text/plain不可执行,并且严格类型校验已开启。 很简单了,改一下content type就好了。
用别的命令比如wget或curl看看 https://localhost:8080/static/js/vue.js 返回什么,有可能这个文件本身就不存在
文件是存在的, 我直接请求这个文件是可以获取到这个文件内容的. 改Content-type的话, 前端我设置了<script> 的类型为text/javascirpt ,但是没有作用.
后端因为是用了 http.FileServer方法, 我不知道要怎么去改Content-type
多谢帮忙, 找到解决办法了
就可以了
http://daiguahub.com/2016/07/03/Chrome-%E6%8A%A5-Resource-interpreted-as-Script-but-transferred-with-MIME-type-text-plain-%E8%AD%A6%E5%91%8A%E7%9A%84%E8%A7%A3%E5%86%B3%E5%8A%9E%E6%B3%95/
可能是注册表里面的文件类型被修改成了'text/plain':
在“计算机/HKEY_CLASSES_ROOT/.js”目录下找到Content Type键,将其值改为“text/javascript”
http2相比http/1.x可能对文件类型做了更严格的检查
有可能是, 已经解决了
就可以了
这应该是覆盖了从注册表里面的读取值
https://github.com/golang/go/issues/24513
The proposal is to automatically include the X-Content-Type-Options: nosniff header when Content-Type is one of the values known to trigger content sniffing: text/plain, application/octet-stream, application/unknown, unknown/unknown, */* or an invalid value without a slash.
这个修改要到go1.11才发布,但x/net/http2已经改了,所以现在用go1.10的http会在content-type为text/plain时检查文件内容看实际是否可执行
说错了,是现在用go1.10的http server不会加X-Content-Type-Options: nosniff 这个header,导致客户端(浏览器)会检查文件内容看实际是否可执行
厉害! 多谢大佬解惑!
不过如果说是因为目前的 http server 不会加这个header, 会导致浏览器检查文件内容实际是否可执行. 那为什么
用这一句启动服务器, 也就是不启用http2,和https, 浏览器会正常解析js呢. 不是也应该不能正常解析么
// 启用http2 err := server.ListenAndServeTLS("./keystore/cert.crt", "./keystore/rsa_private.key") //前端不能正常解析js文件
这里用的 https://github.com/golang/go/issues/24795 修改过的HTTP2,所以加了X-Content-Type-Options: nosniff 这个header,导致不检测文件内容,content-type指定的类型不能执行就实际不执行
这里用的是1.10版本的标准库,1.11才更新上面修改,所以server没加X-Content-Type-Options: nosniff 这个header,导致浏览器在content-type为text/plain的情况下嗅探了文件的内容实际为可执行的js并最终执行该js
golang.org/x/net/http2
不属于标准库看了下代码,golang.org/x/net/http2应该很早就加了nosniff:https://github.com/golang/net/blob/master/http2/server.go#L2349
而不是https://github.com/golang/go/issues/24795 这里才加的
但标准库是在这里提出要加的:https://github.com/golang/go/issues/24513