2024-02-01 12:11:39 +01:00
|
|
|
package security
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http/httptest"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
const jwtString = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCIsInYiOjV9.eyJpYXQiOjE3MDY3ODY3NzUsImlzcyI6ImNocmx5Iiwic2NvcGVzIjpbInByb2ZpbGVzIl19.LrXrKo5iRFFHCDlMsVDhmJJheZqxbxuEVXB4XswHFKY"
|
|
|
|
|
|
|
|
func TestJwtAuth_NewToken(t *testing.T) {
|
|
|
|
jwt := NewJwt([]byte("secret"))
|
|
|
|
now = func() time.Time {
|
|
|
|
return time.Date(2024, 2, 1, 11, 26, 15, 0, time.UTC)
|
|
|
|
}
|
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
t.Run("with known scope", func(t *testing.T) {
|
|
|
|
token, err := jwt.NewToken(ProfilesScope, SignScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.NoError(t, err)
|
2024-03-05 13:07:54 +01:00
|
|
|
require.Equal(t, "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCIsInYiOjV9.eyJpc3MiOiJjaHJseSIsImlhdCI6MTcwNjc4Njc3NSwic2NvcGVzIjpbInByb2ZpbGVzIiwic2lnbiJdfQ.HkNGiDba3I_bLGN6sF0eTE5n6rMLgYfAZZEqI4xb2X4", token)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with unknown scope", func(t *testing.T) {
|
|
|
|
token, err := jwt.NewToken("scope-123")
|
|
|
|
require.ErrorContains(t, err, "unknown")
|
|
|
|
require.Empty(t, token)
|
2024-02-01 12:11:39 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("no scopes", func(t *testing.T) {
|
|
|
|
token, err := jwt.NewToken()
|
|
|
|
require.Error(t, err)
|
|
|
|
require.Empty(t, token)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestJwtAuth_Authenticate(t *testing.T) {
|
|
|
|
jwt := NewJwt([]byte("secret"))
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "Bearer "+jwtString)
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
})
|
|
|
|
|
2024-03-05 13:07:54 +01:00
|
|
|
t.Run("has no required scope", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "Bearer "+jwtString)
|
|
|
|
err := jwt.Authenticate(req, SignScope)
|
|
|
|
require.ErrorContains(t, err, "scope")
|
|
|
|
})
|
|
|
|
|
2024-02-01 12:11:39 +01:00
|
|
|
t.Run("request without auth header", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.ErrorIs(t, err, MissingAuthenticationError)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("no bearer token prefix", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "trash")
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.ErrorIs(t, err, InvalidTokenError)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("bearer token but not jwt", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "Bearer seems.like.jwt")
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.ErrorIs(t, err, InvalidTokenError)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("invalid signature", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "Bearer "+jwtString+"123")
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.ErrorIs(t, err, InvalidTokenError)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("missing v header", func(t *testing.T) {
|
|
|
|
req := httptest.NewRequest("POST", "http://localhost", nil)
|
|
|
|
req.Header.Add("Authorization", "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpYXQiOjE3MDY3ODY3NzUsImlzcyI6ImNocmx5Iiwic2NvcGVzIjpbInByb2ZpbGVzIl19.zOX2ZKyU37kjwt1p9uCHxALxWQD2UC0wWcAcNvBXGq0")
|
2024-03-05 13:07:54 +01:00
|
|
|
err := jwt.Authenticate(req, ProfilesScope)
|
2024-02-01 12:11:39 +01:00
|
|
|
require.ErrorIs(t, err, InvalidTokenError)
|
|
|
|
require.ErrorContains(t, err, "missing v header")
|
|
|
|
})
|
|
|
|
}
|