mirror of
https://github.com/labstack/echo.git
synced 2024-12-24 20:14:31 +02:00
Add middleware.CORSConfig.UnsafeWildcardOriginWithAllowCredentials to make UNSAFE usages of wildcard origin + allow cretentials less likely.
This commit is contained in:
parent
ef4aea97ef
commit
f909660bb9
@ -79,6 +79,15 @@ type (
|
|||||||
// See also: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Credentials
|
// See also: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Credentials
|
||||||
AllowCredentials bool `yaml:"allow_credentials"`
|
AllowCredentials bool `yaml:"allow_credentials"`
|
||||||
|
|
||||||
|
// UnsafeWildcardOriginWithAllowCredentials UNSAFE/INSECURE: allows wildcard '*' origin to be used with AllowCredentials
|
||||||
|
// flag. In that case we consider any origin allowed and send it back to the client with `Access-Control-Allow-Origin` header.
|
||||||
|
//
|
||||||
|
// This is INSECURE and potentially leads to [cross-origin](https://portswigger.net/research/exploiting-cors-misconfigurations-for-bitcoins-and-bounties)
|
||||||
|
// attacks. See: https://github.com/labstack/echo/issues/2400 for discussion on the subject.
|
||||||
|
//
|
||||||
|
// Optional. Default value is false.
|
||||||
|
UnsafeWildcardOriginWithAllowCredentials bool `yaml:"unsafe_wildcard_origin_with_allow_credentials"`
|
||||||
|
|
||||||
// ExposeHeaders determines the value of Access-Control-Expose-Headers, which
|
// ExposeHeaders determines the value of Access-Control-Expose-Headers, which
|
||||||
// defines a list of headers that clients are allowed to access.
|
// defines a list of headers that clients are allowed to access.
|
||||||
//
|
//
|
||||||
@ -203,7 +212,7 @@ func CORSWithConfig(config CORSConfig) echo.MiddlewareFunc {
|
|||||||
} else {
|
} else {
|
||||||
// Check allowed origins
|
// Check allowed origins
|
||||||
for _, o := range config.AllowOrigins {
|
for _, o := range config.AllowOrigins {
|
||||||
if o == "*" && config.AllowCredentials {
|
if o == "*" && config.AllowCredentials && config.UnsafeWildcardOriginWithAllowCredentials {
|
||||||
allowOrigin = origin
|
allowOrigin = origin
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
@ -11,106 +11,190 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestCORS(t *testing.T) {
|
func TestCORS(t *testing.T) {
|
||||||
|
var testCases = []struct {
|
||||||
|
name string
|
||||||
|
givenMW echo.MiddlewareFunc
|
||||||
|
whenMethod string
|
||||||
|
whenHeaders map[string]string
|
||||||
|
expectHeaders map[string]string
|
||||||
|
notExpectHeaders map[string]string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "ok, wildcard origin",
|
||||||
|
whenHeaders: map[string]string{echo.HeaderOrigin: "localhost"},
|
||||||
|
expectHeaders: map[string]string{echo.HeaderAccessControlAllowOrigin: "*"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, wildcard AllowedOrigin with no Origin header in request",
|
||||||
|
notExpectHeaders: map[string]string{echo.HeaderAccessControlAllowOrigin: ""},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, specific AllowOrigins and AllowCredentials",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"localhost"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
MaxAge: 3600,
|
||||||
|
}),
|
||||||
|
whenHeaders: map[string]string{echo.HeaderOrigin: "localhost"},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "localhost",
|
||||||
|
echo.HeaderAccessControlAllowCredentials: "true",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with matching origin for `AllowOrigins`",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"localhost"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
MaxAge: 3600,
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{
|
||||||
|
echo.HeaderOrigin: "localhost",
|
||||||
|
echo.HeaderContentType: echo.MIMEApplicationJSON,
|
||||||
|
},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "localhost",
|
||||||
|
echo.HeaderAccessControlAllowMethods: "GET,HEAD,PUT,PATCH,POST,DELETE",
|
||||||
|
echo.HeaderAccessControlAllowCredentials: "true",
|
||||||
|
echo.HeaderAccessControlMaxAge: "3600",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with wildcard `AllowOrigins` and `AllowCredentials` true",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"*"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
MaxAge: 3600,
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{
|
||||||
|
echo.HeaderOrigin: "localhost",
|
||||||
|
echo.HeaderContentType: echo.MIMEApplicationJSON,
|
||||||
|
},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "*", // Note: browsers will ignore and complain about responses having `*`
|
||||||
|
echo.HeaderAccessControlAllowMethods: "GET,HEAD,PUT,PATCH,POST,DELETE",
|
||||||
|
echo.HeaderAccessControlAllowCredentials: "true",
|
||||||
|
echo.HeaderAccessControlMaxAge: "3600",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with wildcard `AllowOrigins` and `AllowCredentials` false",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"*"},
|
||||||
|
AllowCredentials: false, // important for this testcase
|
||||||
|
MaxAge: 3600,
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{
|
||||||
|
echo.HeaderOrigin: "localhost",
|
||||||
|
echo.HeaderContentType: echo.MIMEApplicationJSON,
|
||||||
|
},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "*",
|
||||||
|
echo.HeaderAccessControlAllowMethods: "GET,HEAD,PUT,PATCH,POST,DELETE",
|
||||||
|
echo.HeaderAccessControlMaxAge: "3600",
|
||||||
|
},
|
||||||
|
notExpectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowCredentials: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, INSECURE preflight request with wildcard `AllowOrigins` and `AllowCredentials` true",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"*"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
UnsafeWildcardOriginWithAllowCredentials: true, // important for this testcase
|
||||||
|
MaxAge: 3600,
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{
|
||||||
|
echo.HeaderOrigin: "localhost",
|
||||||
|
echo.HeaderContentType: echo.MIMEApplicationJSON,
|
||||||
|
},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "localhost", // This could end up as cross-origin attack
|
||||||
|
echo.HeaderAccessControlAllowMethods: "GET,HEAD,PUT,PATCH,POST,DELETE",
|
||||||
|
echo.HeaderAccessControlAllowCredentials: "true",
|
||||||
|
echo.HeaderAccessControlMaxAge: "3600",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with Access-Control-Request-Headers",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"*"},
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{
|
||||||
|
echo.HeaderOrigin: "localhost",
|
||||||
|
echo.HeaderContentType: echo.MIMEApplicationJSON,
|
||||||
|
echo.HeaderAccessControlRequestHeaders: "Special-Request-Header",
|
||||||
|
},
|
||||||
|
expectHeaders: map[string]string{
|
||||||
|
echo.HeaderAccessControlAllowOrigin: "*",
|
||||||
|
echo.HeaderAccessControlAllowHeaders: "Special-Request-Header",
|
||||||
|
echo.HeaderAccessControlAllowMethods: "GET,HEAD,PUT,PATCH,POST,DELETE",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with `AllowOrigins` which allow all subdomains aaa with *",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"http://*.example.com"},
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{echo.HeaderOrigin: "http://aaa.example.com"},
|
||||||
|
expectHeaders: map[string]string{echo.HeaderAccessControlAllowOrigin: "http://aaa.example.com"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "ok, preflight request with `AllowOrigins` which allow all subdomains bbb with *",
|
||||||
|
givenMW: CORSWithConfig(CORSConfig{
|
||||||
|
AllowOrigins: []string{"http://*.example.com"},
|
||||||
|
}),
|
||||||
|
whenMethod: http.MethodOptions,
|
||||||
|
whenHeaders: map[string]string{echo.HeaderOrigin: "http://bbb.example.com"},
|
||||||
|
expectHeaders: map[string]string{echo.HeaderAccessControlAllowOrigin: "http://bbb.example.com"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
e := echo.New()
|
e := echo.New()
|
||||||
|
|
||||||
// Wildcard origin
|
mw := CORS()
|
||||||
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
if tc.givenMW != nil {
|
||||||
|
mw = tc.givenMW
|
||||||
|
}
|
||||||
|
h := mw(func(c echo.Context) error {
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
method := http.MethodGet
|
||||||
|
if tc.whenMethod != "" {
|
||||||
|
method = tc.whenMethod
|
||||||
|
}
|
||||||
|
req := httptest.NewRequest(method, "/", nil)
|
||||||
rec := httptest.NewRecorder()
|
rec := httptest.NewRecorder()
|
||||||
c := e.NewContext(req, rec)
|
c := e.NewContext(req, rec)
|
||||||
h := CORS()(echo.NotFoundHandler)
|
for k, v := range tc.whenHeaders {
|
||||||
req.Header.Set(echo.HeaderOrigin, "localhost")
|
req.Header.Set(k, v)
|
||||||
h(c)
|
}
|
||||||
assert.Equal(t, "*", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
|
|
||||||
// Wildcard AllowedOrigin with no Origin header in request
|
err := h(c)
|
||||||
req = httptest.NewRequest(http.MethodGet, "/", nil)
|
|
||||||
rec = httptest.NewRecorder()
|
|
||||||
c = e.NewContext(req, rec)
|
|
||||||
h = CORS()(echo.NotFoundHandler)
|
|
||||||
h(c)
|
|
||||||
assert.NotContains(t, rec.Header(), echo.HeaderAccessControlAllowOrigin)
|
|
||||||
|
|
||||||
// Allow origins
|
assert.NoError(t, err)
|
||||||
req = httptest.NewRequest(http.MethodGet, "/", nil)
|
header := rec.Header()
|
||||||
rec = httptest.NewRecorder()
|
for k, v := range tc.expectHeaders {
|
||||||
c = e.NewContext(req, rec)
|
assert.Equal(t, v, header.Get(k), "header: `%v` should be `%v`", k, v)
|
||||||
h = CORSWithConfig(CORSConfig{
|
}
|
||||||
AllowOrigins: []string{"localhost"},
|
for k, v := range tc.notExpectHeaders {
|
||||||
AllowCredentials: true,
|
if v == "" {
|
||||||
MaxAge: 3600,
|
assert.Len(t, header.Values(k), 0, "header: `%v` should not be set", k)
|
||||||
})(echo.NotFoundHandler)
|
} else {
|
||||||
req.Header.Set(echo.HeaderOrigin, "localhost")
|
assert.NotEqual(t, v, header.Get(k), "header: `%v` should not be `%v`", k, v)
|
||||||
h(c)
|
}
|
||||||
assert.Equal(t, "localhost", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
}
|
||||||
assert.Equal(t, "true", rec.Header().Get(echo.HeaderAccessControlAllowCredentials))
|
|
||||||
|
|
||||||
// Preflight request
|
|
||||||
req = httptest.NewRequest(http.MethodOptions, "/", nil)
|
|
||||||
rec = httptest.NewRecorder()
|
|
||||||
c = e.NewContext(req, rec)
|
|
||||||
req.Header.Set(echo.HeaderOrigin, "localhost")
|
|
||||||
req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON)
|
|
||||||
cors := CORSWithConfig(CORSConfig{
|
|
||||||
AllowOrigins: []string{"localhost"},
|
|
||||||
AllowCredentials: true,
|
|
||||||
MaxAge: 3600,
|
|
||||||
})
|
})
|
||||||
h = cors(echo.NotFoundHandler)
|
}
|
||||||
h(c)
|
|
||||||
assert.Equal(t, "localhost", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
assert.NotEmpty(t, rec.Header().Get(echo.HeaderAccessControlAllowMethods))
|
|
||||||
assert.Equal(t, "true", rec.Header().Get(echo.HeaderAccessControlAllowCredentials))
|
|
||||||
assert.Equal(t, "3600", rec.Header().Get(echo.HeaderAccessControlMaxAge))
|
|
||||||
|
|
||||||
// Preflight request with `AllowOrigins` *
|
|
||||||
req = httptest.NewRequest(http.MethodOptions, "/", nil)
|
|
||||||
rec = httptest.NewRecorder()
|
|
||||||
c = e.NewContext(req, rec)
|
|
||||||
req.Header.Set(echo.HeaderOrigin, "localhost")
|
|
||||||
req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON)
|
|
||||||
cors = CORSWithConfig(CORSConfig{
|
|
||||||
AllowOrigins: []string{"*"},
|
|
||||||
AllowCredentials: true,
|
|
||||||
MaxAge: 3600,
|
|
||||||
})
|
|
||||||
h = cors(echo.NotFoundHandler)
|
|
||||||
h(c)
|
|
||||||
assert.Equal(t, "localhost", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
assert.NotEmpty(t, rec.Header().Get(echo.HeaderAccessControlAllowMethods))
|
|
||||||
assert.Equal(t, "true", rec.Header().Get(echo.HeaderAccessControlAllowCredentials))
|
|
||||||
assert.Equal(t, "3600", rec.Header().Get(echo.HeaderAccessControlMaxAge))
|
|
||||||
|
|
||||||
// Preflight request with Access-Control-Request-Headers
|
|
||||||
req = httptest.NewRequest(http.MethodOptions, "/", nil)
|
|
||||||
rec = httptest.NewRecorder()
|
|
||||||
c = e.NewContext(req, rec)
|
|
||||||
req.Header.Set(echo.HeaderOrigin, "localhost")
|
|
||||||
req.Header.Set(echo.HeaderContentType, echo.MIMEApplicationJSON)
|
|
||||||
req.Header.Set(echo.HeaderAccessControlRequestHeaders, "Special-Request-Header")
|
|
||||||
cors = CORSWithConfig(CORSConfig{
|
|
||||||
AllowOrigins: []string{"*"},
|
|
||||||
})
|
|
||||||
h = cors(echo.NotFoundHandler)
|
|
||||||
h(c)
|
|
||||||
assert.Equal(t, "*", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
assert.Equal(t, "Special-Request-Header", rec.Header().Get(echo.HeaderAccessControlAllowHeaders))
|
|
||||||
assert.NotEmpty(t, rec.Header().Get(echo.HeaderAccessControlAllowMethods))
|
|
||||||
|
|
||||||
// Preflight request with `AllowOrigins` which allow all subdomains with *
|
|
||||||
req = httptest.NewRequest(http.MethodOptions, "/", nil)
|
|
||||||
rec = httptest.NewRecorder()
|
|
||||||
c = e.NewContext(req, rec)
|
|
||||||
req.Header.Set(echo.HeaderOrigin, "http://aaa.example.com")
|
|
||||||
cors = CORSWithConfig(CORSConfig{
|
|
||||||
AllowOrigins: []string{"http://*.example.com"},
|
|
||||||
})
|
|
||||||
h = cors(echo.NotFoundHandler)
|
|
||||||
h(c)
|
|
||||||
assert.Equal(t, "http://aaa.example.com", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
|
|
||||||
req.Header.Set(echo.HeaderOrigin, "http://bbb.example.com")
|
|
||||||
h(c)
|
|
||||||
assert.Equal(t, "http://bbb.example.com", rec.Header().Get(echo.HeaderAccessControlAllowOrigin))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func Test_allowOriginScheme(t *testing.T) {
|
func Test_allowOriginScheme(t *testing.T) {
|
||||||
|
Loading…
Reference in New Issue
Block a user