Implement Allow-Methods and Allow-Headers, refactor isOriginAllowed

This commit is contained in:
Rafael Magana 2014-09-18 11:48:37 -05:00
parent 3f537796eb
commit f6023442e9
3 changed files with 84 additions and 16 deletions

View File

@ -106,8 +106,10 @@ func (a *api) CORSHandler(w http.ResponseWriter, req *http.Request) {
origin := req.Header.Get("Origin")
if a.cors.isOriginAllowed(origin) {
w.Header().Set("Access-Control-Allow-Origin", origin)
w.Header().Set("Access-Control-Allow-Headers", a.cors.AllowedHeaders())
w.Header().Set("Access-Control-Allow-Methods", a.cors.AllowedMethods())
w.Header().Set("Content-Type", a.cors.ContentType)
}
}
func (a *api) SetDebug() {

View File

@ -1,24 +1,56 @@
package api
import (
"regexp"
"strings"
)
type CORS struct {
AllowOrigins []string
AllowHeaders []string // Not yet implemented
AllowMethods []string // ditto
AllowOrigins []string
AllowHeaders []string
AllowMethods []string
ContentType string
allowOriginPatterns []string
}
func NewCORS(allowedOrigins []string) *CORS {
return &CORS{
cors := &CORS{
AllowOrigins: allowedOrigins,
AllowMethods: []string{"GET", "POST"},
AllowHeaders: []string{"Origin", "Content-Type"},
ContentType: "application/json; charset=utf-8",
}
cors.generatePatterns()
return cors
}
func (c *CORS) isOriginAllowed(origin string) (allowed bool) {
for _, allowedOriginPattern := range c.allowOriginPatterns {
allowed, _ = regexp.MatchString(allowedOriginPattern, origin)
if allowed {
return
}
}
return
}
func (c *CORS) generatePatterns() {
if c.AllowOrigins != nil {
for _, origin := range c.AllowOrigins {
pattern := regexp.QuoteMeta(origin)
pattern = strings.Replace(pattern, "\\*", ".*", -1)
pattern = strings.Replace(pattern, "\\?", ".", -1)
c.allowOriginPatterns = append(c.allowOriginPatterns, "^"+pattern+"$")
}
}
}
func (c *CORS) isOriginAllowed(currentOrigin string) bool {
for _, allowedOrigin := range c.AllowOrigins {
if "*" == allowedOrigin || currentOrigin == allowedOrigin {
return true
}
}
return false
func (c *CORS) AllowedHeaders() string {
return strings.Join(c.AllowHeaders, ",")
}
func (c *CORS) AllowedMethods() string {
return strings.Join(c.AllowMethods, ",")
}

View File

@ -16,11 +16,19 @@ func TestNewCORS(t *testing.T) {
}
func TestNewCorsSetsProperties(t *testing.T) {
allowedOrigins := []string{"http://server:port"}
allowedOrigins := []string{"http://*server:*", "http://localhost:*"}
allowedMethods := []string{"GET", "POST"}
allowedHeaders := []string{"Origin", "Content-Type"}
contentType := "application/json; charset=utf-8"
allowOriginPatterns := []string{"^http://.*server:.*$", "^http://localhost:.*$"}
cors := NewCORS(allowedOrigins)
gobot.Assert(t, cors.AllowOrigins, allowedOrigins)
gobot.Assert(t, cors.AllowMethods, allowedMethods)
gobot.Assert(t, cors.AllowHeaders, allowedHeaders)
gobot.Assert(t, cors.ContentType, contentType)
gobot.Assert(t, cors.allowOriginPatterns, allowOriginPatterns)
}
func TestCORSIsOriginAllowed(t *testing.T) {
@ -32,16 +40,42 @@ func TestCORSIsOriginAllowed(t *testing.T) {
gobot.Assert(t, cors.isOriginAllowed("http://server.com"), true)
// When one origin is accepted
cors.AllowOrigins = []string{"http://localhost:8000"}
cors = NewCORS([]string{"http://localhost:8000"})
gobot.Assert(t, cors.isOriginAllowed("http://localhost:8000"), true)
gobot.Assert(t, cors.isOriginAllowed("http://localhost:3001"), false)
gobot.Assert(t, cors.isOriginAllowed("http://server.com"), false)
// When several origins are accepted
cors.AllowOrigins = []string{"http://localhost:8000", "http://server.com"}
cors = NewCORS([]string{"http://localhost:*", "http://server.com"})
gobot.Assert(t, cors.isOriginAllowed("http://localhost:8000"), true)
gobot.Assert(t, cors.isOriginAllowed("http://localhost:3001"), false)
gobot.Assert(t, cors.isOriginAllowed("http://localhost:3001"), true)
gobot.Assert(t, cors.isOriginAllowed("http://server.com"), true)
// When several origins are accepted within the same domain
cors = NewCORS([]string{"http://*.server.com"})
gobot.Assert(t, cors.isOriginAllowed("http://localhost:8000"), false)
gobot.Assert(t, cors.isOriginAllowed("http://localhost:3001"), false)
gobot.Assert(t, cors.isOriginAllowed("http://foo.server.com"), true)
gobot.Assert(t, cors.isOriginAllowed("http://api.server.com"), true)
}
func TestCORSAllowedHeaders(t *testing.T) {
cors := NewCORS([]string{"*"})
cors.AllowHeaders = []string{"Header1", "Header2"}
gobot.Assert(t, cors.AllowedHeaders(), "Header1,Header2")
}
func TestCORSAllowedMethods(t *testing.T) {
cors := NewCORS([]string{"*"})
gobot.Assert(t, cors.AllowedMethods(), "GET,POST")
cors.AllowMethods = []string{"GET", "POST", "PUT"}
gobot.Assert(t, cors.AllowedMethods(), "GET,POST,PUT")
}