Skip to content

Commit

Permalink
Merge pull request #201 from bboozzoo/bboozzoo/cors-access-control-re…
Browse files Browse the repository at this point in the history
…quest-headers

Empty Access-Control-Request-Headers in CORS preflight triggers 403 Forbidden
  • Loading branch information
ant0ine authored Dec 1, 2016
2 parents 709bbe3 + 7535cd0 commit ce4b71c
Show file tree
Hide file tree
Showing 3 changed files with 84 additions and 0 deletions.
43 changes: 43 additions & 0 deletions rest/cors_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package rest

import (
"net/http"
"testing"

"github.com/ant0ine/go-json-rest/rest/test"
)

func TestCorsMiddlewareEmptyAccessControlRequestHeaders(t *testing.T) {
api := NewApi()

// the middleware to test
api.Use(&CorsMiddleware{
OriginValidator: func(_ string, _ *Request) bool {
return true
},
AllowedMethods: []string{
"GET",
"POST",
"PUT",
},
AllowedHeaders: []string{
"Origin",
"Referer",
},
})

// wrap all
handler := api.MakeHandler()

req, _ := http.NewRequest("OPTIONS", "http://localhost", nil)
req.Header.Set("Origin", "http://another.host")
req.Header.Set("Access-Control-Request-Method", "PUT")
req.Header.Set("Access-Control-Request-Headers", "")

recorded := test.RunRequest(t, handler, req)
t.Logf("recorded: %+v\n", recorded.Recorder)
recorded.CodeIs(200)
recorded.HeaderIs("Access-Control-Allow-Methods", "GET,POST,PUT")
recorded.HeaderIs("Access-Control-Allow-Headers", "Origin,Referer")
recorded.HeaderIs("Access-Control-Allow-Origin", "http://another.host")
}
3 changes: 3 additions & 0 deletions rest/request.go
Original file line number Diff line number Diff line change
Expand Up @@ -120,6 +120,9 @@ func (r *Request) GetCorsInfo() *CorsInfo {
reqHeaders := []string{}
rawReqHeaders := r.Header[http.CanonicalHeaderKey("Access-Control-Request-Headers")]
for _, rawReqHeader := range rawReqHeaders {
if len(rawReqHeader) == 0 {
continue
}
// net/http does not handle comma delimited headers for us
for _, reqHeader := range strings.Split(rawReqHeader, ",") {
reqHeaders = append(reqHeaders, http.CanonicalHeaderKey(strings.TrimSpace(reqHeader)))
Expand Down
38 changes: 38 additions & 0 deletions rest/request_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -148,3 +148,41 @@ func TestCorsInfoPreflightCors(t *testing.T) {
t.Error("OriginUrl must be set")
}
}

func TestCorsInfoEmptyAccessControlRequestHeaders(t *testing.T) {
req := defaultRequest("OPTIONS", "http://localhost", nil, t)
req.Request.Header.Set("Origin", "http://another.host")

// make it a preflight request
req.Request.Header.Set("Access-Control-Request-Method", "PUT")

// WebKit based browsers may send `Access-Control-Request-Headers:` with
// no value, in which case, the header will be present in requests
// Header map, but its value is an empty string.
req.Request.Header.Set("Access-Control-Request-Headers", "")
corsInfo := req.GetCorsInfo()
if corsInfo == nil {
t.Error("Expected non nil CorsInfo")
}
if corsInfo.IsCors == false {
t.Error("This is a CORS request")
}
if len(corsInfo.AccessControlRequestHeaders) > 0 {
t.Error("Access-Control-Request-Headers should have been removed")
}

req.Request.Header.Set("Access-Control-Request-Headers", "")
corsInfo = req.GetCorsInfo()
if corsInfo == nil {
t.Error("Expected non nil CorsInfo")
}
if corsInfo.IsCors == false {
t.Error("This is a CORS request")
}
if corsInfo.IsPreflight == false {
t.Error("This is a Preflight request")
}
if len(corsInfo.AccessControlRequestHeaders) > 0 {
t.Error("Empty Access-Control-Request-Headers header should have been removed")
}
}

0 comments on commit ce4b71c

Please sign in to comment.