Skip to content

Commit

Permalink
Implement Allow-Methods and Allow-Headers, refactor isOriginAllowed
Browse files Browse the repository at this point in the history
  • Loading branch information
rafmagana committed Sep 18, 2014
1 parent 3f53779 commit f602344
Show file tree
Hide file tree
Showing 3 changed files with 82 additions and 14 deletions.
4 changes: 3 additions & 1 deletion api/api.go
Original file line number Diff line number Diff line change
Expand Up @@ -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() {
Expand Down
50 changes: 41 additions & 9 deletions api/cors.go
Original file line number Diff line number Diff line change
@@ -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(currentOrigin string) bool {
for _, allowedOrigin := range c.AllowOrigins {
if "*" == allowedOrigin || currentOrigin == allowedOrigin {
return true
func (c *CORS) isOriginAllowed(origin string) (allowed bool) {
for _, allowedOriginPattern := range c.allowOriginPatterns {
allowed, _ = regexp.MatchString(allowedOriginPattern, origin)
if allowed {
return
}
}
return false
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) AllowedHeaders() string {
return strings.Join(c.AllowHeaders, ",")
}

func (c *CORS) AllowedMethods() string {
return strings.Join(c.AllowMethods, ",")
}
42 changes: 38 additions & 4 deletions api/cors_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -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) {
Expand All @@ -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")
}

0 comments on commit f602344

Please sign in to comment.