出于效率等原因,最近将web框架由martini切换为了beego,其他地方都很平顺,只是两个框架的handler签名不一致,需要修改,所以耗时较长,这是预计到的。但是有一个地方没有预计到,也耗费了较多时间,那就是静态文件的服务。
用过martini的tx都知道,在mairtini中如果我们设置一个目录为静态文件目录,只需添加martini的Static插件,如设置web子目录为应用的静态文件路径:
m.Use(martini.Static("web"))
此时,如果我们访问一个url,此url并没有在martini中注册,但是如果位于web目录中,就可以得到响应,例如:
http://127.0.0.1:8088/
//返回web目录下的index.html
http://127.0.0.1:8088/ js/jquery.js
//返回web/js/jquery.js
但是,切换为beego之后,却没有找到这样的功能。发现beego对于静态文件的支持设计的有点不够友好,比如我进行如下设置
beego.SetStaticPath("/web", "web")
这时候访问结果如下
http://127.0.0.1:8088/
//返回404页面
http://127.0.0.1:8088/web
//返回404页面
http://127.0.0.1:8088/web/index.html
//返回403 (Forbidden)
http://127.0.0.1:8088/web/chat.html
//返回正常
http://127.0.0.1:8088/web/images/test.png
//返回正常
据此结果,有两点不满意:
- 必须添加该staticdir 映射的path访问,设置为“/” 无效
- 竟然不能返回默认页面!看文档需要设置”
beego.DirectoryIndex=true
“ ,不是我需要的!
因此,我着手自己实现该需求。通过学习beego文档,发现可以设置Filter。于是,编写如下代码:
//main中如下设置filter
beego.InsertFilter("/*", beego.BeforeRouter, TransparentStatic)
.
.
.
func TransparentStatic(ctx *context.Context) {
defInd :=
maxInd := len(defHomes) -
orpath := ctx.Request.URL.Path
beego.Debug(" in trasparentstatic filter orpath", orpath)
if strings.Index(orpath, "api/") >= || strings.Index(orpath, "web/") >= {
return
}
DefaultStartPage:
p := orpath
if strings.EqualFold(p, "/") {
p += defHomes[defInd]
defInd++
}
ps := strings.Split(p, "/")
ps = append([]string{"web"}, ps...)
rp := strings.Join(ps, "/")
fp := fw.MapPath(rp)
beego.Debug("test fp", fp)
if !fileutils.Exists(fp) {
if defInd > && defInd < maxInd {
goto DefaultStartPage
}
return
} else {
beego.Debug("found static ", fp)
http.ServeFile(ctx.ResponseWriter, ctx.Request, fp)
//cannot use Redirect! will lead loop
//http.Redirect(ctx.ResponseWriter, ctx.Request, rp, http.StatusFound)
return
}
// }
运行之后,发现访问服务地址,带不带末尾的"/",都不能返回默认页面,如果明确访问/index.html可以实现访问。后经探索发现,虽然beego说明中说"/*"可以适配所有url,但是实际上不能适配"/",因此需要在注册一个filter到”/":
beego.InsertFilter("/", beego.BeforeRouter, TransparentStatic) //must has this for default page
beego.InsertFilter("/*", beego.BeforeRouter, TransparentStatic)
至此,一切正常了。