test: add CORS middleware tests
Add tests for CORS functionality: - validateCORSConfig (valid and invalid configs) - SetCORSConfig (update and validation) - resolveAllowedOrigin (exact match, wildcard, case insensitive) - CORS middleware (allow/forbid origins, OPTIONS handling) Coverage: middleware 36.4% → 37.4%
This commit is contained in:
215
internal/api/middleware/cors_test.go
Normal file
215
internal/api/middleware/cors_test.go
Normal file
@@ -0,0 +1,215 @@
|
|||||||
|
package middleware
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/user-management-system/internal/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestValidateCORSConfig(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
cfg config.CORSConfig
|
||||||
|
wantErr bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "valid config with specific origins",
|
||||||
|
cfg: config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"https://example.com"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
},
|
||||||
|
wantErr: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "valid config with wildcard no credentials",
|
||||||
|
cfg: config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"*"},
|
||||||
|
AllowCredentials: false,
|
||||||
|
},
|
||||||
|
wantErr: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "invalid config with wildcard and credentials",
|
||||||
|
cfg: config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"*"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
},
|
||||||
|
wantErr: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "empty origins",
|
||||||
|
cfg: config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{},
|
||||||
|
AllowCredentials: false,
|
||||||
|
},
|
||||||
|
wantErr: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
err := validateCORSConfig(tt.cfg)
|
||||||
|
if tt.wantErr {
|
||||||
|
assert.Error(t, err)
|
||||||
|
} else {
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSetCORSConfig(t *testing.T) {
|
||||||
|
// Save original config
|
||||||
|
originalConfig := corsConfig
|
||||||
|
defer func() { corsConfig = originalConfig }()
|
||||||
|
|
||||||
|
t.Run("valid config", func(t *testing.T) {
|
||||||
|
cfg := config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"https://example.com"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
}
|
||||||
|
err := SetCORSConfig(cfg)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, cfg, corsConfig)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("invalid config", func(t *testing.T) {
|
||||||
|
cfg := config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"*"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
}
|
||||||
|
err := SetCORSConfig(cfg)
|
||||||
|
assert.Error(t, err)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestResolveAllowedOrigin(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
origin string
|
||||||
|
allowedOrigins []string
|
||||||
|
allowCredentials bool
|
||||||
|
wantOrigin string
|
||||||
|
wantAllowed bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "exact match",
|
||||||
|
origin: "https://example.com",
|
||||||
|
allowedOrigins: []string{"https://example.com"},
|
||||||
|
allowCredentials: true,
|
||||||
|
wantOrigin: "https://example.com",
|
||||||
|
wantAllowed: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "wildcard without credentials",
|
||||||
|
origin: "https://any.com",
|
||||||
|
allowedOrigins: []string{"*"},
|
||||||
|
allowCredentials: false,
|
||||||
|
wantOrigin: "*",
|
||||||
|
wantAllowed: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "wildcard with credentials returns origin",
|
||||||
|
origin: "https://any.com",
|
||||||
|
allowedOrigins: []string{"*"},
|
||||||
|
allowCredentials: true,
|
||||||
|
wantOrigin: "https://any.com",
|
||||||
|
wantAllowed: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "no match",
|
||||||
|
origin: "https://evil.com",
|
||||||
|
allowedOrigins: []string{"https://example.com"},
|
||||||
|
allowCredentials: false,
|
||||||
|
wantOrigin: "",
|
||||||
|
wantAllowed: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "case insensitive match",
|
||||||
|
origin: "HTTPS://EXAMPLE.COM",
|
||||||
|
allowedOrigins: []string{"https://example.com"},
|
||||||
|
allowCredentials: false,
|
||||||
|
wantOrigin: "HTTPS://EXAMPLE.COM",
|
||||||
|
wantAllowed: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "empty origins list",
|
||||||
|
origin: "https://example.com",
|
||||||
|
allowedOrigins: []string{},
|
||||||
|
allowCredentials: false,
|
||||||
|
wantOrigin: "",
|
||||||
|
wantAllowed: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
gotOrigin, gotAllowed := resolveAllowedOrigin(tt.origin, tt.allowedOrigins, tt.allowCredentials)
|
||||||
|
assert.Equal(t, tt.wantOrigin, gotOrigin)
|
||||||
|
assert.Equal(t, tt.wantAllowed, gotAllowed)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCORS(t *testing.T) {
|
||||||
|
gin.SetMode(gin.TestMode)
|
||||||
|
|
||||||
|
// Save and restore original config
|
||||||
|
originalConfig := corsConfig
|
||||||
|
defer func() { corsConfig = originalConfig }()
|
||||||
|
|
||||||
|
// Set test config
|
||||||
|
corsConfig = config.CORSConfig{
|
||||||
|
AllowedOrigins: []string{"https://example.com"},
|
||||||
|
AllowCredentials: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
router := gin.New()
|
||||||
|
router.Use(CORS())
|
||||||
|
router.GET("/test", func(c *gin.Context) {
|
||||||
|
c.String(200, "OK")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("allow valid origin", func(t *testing.T) {
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
req, _ := http.NewRequest("GET", "/test", nil)
|
||||||
|
req.Header.Set("Origin", "https://example.com")
|
||||||
|
router.ServeHTTP(w, req)
|
||||||
|
|
||||||
|
assert.Equal(t, 200, w.Code)
|
||||||
|
assert.Equal(t, "https://example.com", w.Header().Get("Access-Control-Allow-Origin"))
|
||||||
|
assert.Equal(t, "true", w.Header().Get("Access-Control-Allow-Credentials"))
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("forbid invalid origin", func(t *testing.T) {
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
req, _ := http.NewRequest("GET", "/test", nil)
|
||||||
|
req.Header.Set("Origin", "https://evil.com")
|
||||||
|
router.ServeHTTP(w, req)
|
||||||
|
|
||||||
|
assert.Equal(t, 403, w.Code)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("handle OPTIONS request", func(t *testing.T) {
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
req, _ := http.NewRequest("OPTIONS", "/test", nil)
|
||||||
|
req.Header.Set("Origin", "https://example.com")
|
||||||
|
router.ServeHTTP(w, req)
|
||||||
|
|
||||||
|
assert.Equal(t, 204, w.Code)
|
||||||
|
assert.Equal(t, "GET, POST, PUT, DELETE, OPTIONS", w.Header().Get("Access-Control-Allow-Methods"))
|
||||||
|
assert.NotEmpty(t, w.Header().Get("Access-Control-Allow-Headers"))
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("no origin header", func(t *testing.T) {
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
req, _ := http.NewRequest("GET", "/test", nil)
|
||||||
|
router.ServeHTTP(w, req)
|
||||||
|
|
||||||
|
assert.Equal(t, 200, w.Code)
|
||||||
|
})
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user