相关方法: https://pkg.go.dev/github.com/gogf/gf/v2/net/ghttp#Response

CORSW3互联网标准组织对HTTP跨域请求的标准,在ghttp模块中,我们可以通过CORSOptions对象来管理对应的跨域请求选项。定义如下:

  1. // See https://www.w3.org/TR/cors/ .
  2. // 服务端允许跨域请求选项
  3. type CORSOptions struct {
  4. AllowDomain []string // Used for allowing requests from custom domains
  5. AllowOrigin string // Access-Control-Allow-Origin
  6. AllowCredentials string // Access-Control-Allow-Credentials
  7. ExposeHeaders string // Access-Control-Expose-Headers
  8. MaxAge int // Access-Control-Max-Age
  9. AllowMethods string // Access-Control-Allow-Methods
  10. AllowHeaders string // Access-Control-Allow-Headers
  11. }

具体参数的介绍请查看W3组织 官网手册

CORS配置

当然,为方便跨域设置,在ghttp模块中也提供了默认的跨域请求选项,通过DefaultCORSOptions方法获取。大多数情况下,我们在需要允许跨域请求的接口中(一般情况下使用中间件)可以直接使用CORSDefault()允许该接口跨域访问。

限制Origin来源

大多数时候,我们需要限制请求来源为我们受信任的域名列表,我们可以使用AllowDomain配置,使用方式:

示例1,基本使用

我们来看一个简单的接口示例:

  1. package main
  2. import (
  3. "github.com/gogf/gf/v2/frame/g"
  4. "github.com/gogf/gf/v2/net/ghttp"
  5. func Order(r *ghttp.Request) {
  6. r.Response.Write("GET")
  7. }
  8. func main() {
  9. s.Group("/api.v1", func(group *ghttp.RouterGroup) {
  10. group.GET("/order", Order)
  11. })
  12. s.SetPort(8199)
  13. s.Run()
  14. }

接口地址是 http://localhost/api.v1/order ,当然这个接口是不允许跨域的。我们打开一个不同的域名,例如:百度首页(正好用了jQuery,方便调试),然后按F12打开开发者面板,在console下执行以下AJAX请求:

结果如下:

返回了不允许跨域请求的提示错误,接着我们修改一下服务端接口的测试代码,如下:

  1. package main
  2. import (
  3. "github.com/gogf/gf/v2/frame/g"
  4. "github.com/gogf/gf/v2/net/ghttp"
  5. )
  6. func MiddlewareCORS(r *ghttp.Request) {
  7. r.Response.CORSDefault()
  8. r.Middleware.Next()
  9. }
  10. func Order(r *ghttp.Request) {
  11. r.Response.Write("GET")
  12. }
  13. func main() {
  14. s := g.Server()
  15. s.Group("/api.v1", func(group *ghttp.RouterGroup) {
  16. group.Middleware(MiddlewareCORS)
  17. group.GET("/order", Order)
  18. })
  19. s.SetPort(8199)
  20. s.Run()

返回刚才的百度首页,再次执行请求AJAX请求,这次便成功了:

CORS跨域处理 - 图2

当然我们也可以通过CORSOptions对象以及CORS方法对跨域请求做更多的设置。

在大多数场景中,我们是需要自定义授权跨域的Origin,那么我们可以将以上的例子改进如下,在该示例中,我们仅允许goframe.orgbaidu.com跨域请求访问。

示例3,自定义检测授权

不知大家是否有注意,在以上的示例中有个细节,即使当前接口不允许跨域访问,但是只要接口被调用,接口完整逻辑仍会被执行,在服务端其实也已经走完了一次请求流程。针对于这个问题,我们可以通过自定义授权Origin并在中间件中通过CORSAllowedOrigin方法来做判断,如果当前请求的Origin在服务端是被允许执行的,那么才会执行后续流程,否则便会终止执行。

  1. package main
  2. import (
  3. "github.com/gogf/gf/v2/frame/g"
  4. "github.com/gogf/gf/v2/net/ghttp"
  5. )
  6. func MiddlewareCORS(r *ghttp.Request) {
  7. corsOptions := r.Response.DefaultCORSOptions()
  8. corsOptions.AllowDomain = []string{"goframe.org"}
  9. if !r.Response.CORSAllowedOrigin(corsOptions) {
  10. r.Response.WriteStatus(http.StatusForbidden)
  11. return
  12. }
  13. r.Response.CORS(corsOptions)
  14. r.Middleware.Next()
  15. }
  16. func Order(r *ghttp.Request) {
  17. r.Response.Write("GET")
  18. }
  19. func main() {
  20. s := g.Server()
  21. s.Group("/api.v1", func(group *ghttp.RouterGroup) {
  22. group.Middleware(MiddlewareCORS)
  23. group.GET("/order", Order)
  24. })
  25. s.SetPort(8199)