forked from TrueCloudLab/distribution
feat: add WeakStringList type to support lists in aud claim
Signed-off-by: Mark Sagi-Kazar <mark.sagikazar@gmail.com>
This commit is contained in:
parent
78b9c98c5c
commit
97fa1183bf
2 changed files with 140 additions and 0 deletions
55
registry/auth/token/types.go
Normal file
55
registry/auth/token/types.go
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
package token
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
// WeakStringList is a slice of strings that can be deserialized from either a single string value or a list of strings.
|
||||||
|
type WeakStringList []string
|
||||||
|
|
||||||
|
func (s *WeakStringList) UnmarshalJSON(data []byte) (err error) {
|
||||||
|
var value interface{}
|
||||||
|
|
||||||
|
if err = json.Unmarshal(data, &value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch v := value.(type) {
|
||||||
|
case string:
|
||||||
|
*s = []string{v}
|
||||||
|
|
||||||
|
case []string:
|
||||||
|
*s = v
|
||||||
|
|
||||||
|
case []interface{}:
|
||||||
|
var ss []string
|
||||||
|
|
||||||
|
for _, vv := range v {
|
||||||
|
vs, ok := vv.(string)
|
||||||
|
if !ok {
|
||||||
|
return &json.UnsupportedTypeError{
|
||||||
|
Type: reflect.TypeOf(vv),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ss = append(ss, vs)
|
||||||
|
}
|
||||||
|
|
||||||
|
*s = ss
|
||||||
|
|
||||||
|
case nil:
|
||||||
|
return nil
|
||||||
|
|
||||||
|
default:
|
||||||
|
return &json.UnsupportedTypeError{
|
||||||
|
Type: reflect.TypeOf(v),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s WeakStringList) MarshalJSON() (b []byte, err error) {
|
||||||
|
return json.Marshal([]string(s))
|
||||||
|
}
|
85
registry/auth/token/types_test.go
Normal file
85
registry/auth/token/types_test.go
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
package token
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestWeakStringList_Unmarshal(t *testing.T) {
|
||||||
|
t.Run("OK", func(t *testing.T) {
|
||||||
|
testCases := []struct {
|
||||||
|
value string
|
||||||
|
expected WeakStringList
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
value: `"audience"`,
|
||||||
|
expected: WeakStringList{"audience"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: `["audience1", "audience2"]`,
|
||||||
|
expected: WeakStringList{"audience1", "audience2"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: `null`,
|
||||||
|
expected: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testCase := range testCases {
|
||||||
|
testCase := testCase
|
||||||
|
|
||||||
|
t.Run("", func(t *testing.T) {
|
||||||
|
var actual WeakStringList
|
||||||
|
|
||||||
|
err := json.Unmarshal([]byte(testCase.value), &actual)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assertStringListEqual(t, testCase.expected, actual)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Error", func(t *testing.T) {
|
||||||
|
var actual WeakStringList
|
||||||
|
|
||||||
|
err := json.Unmarshal([]byte("1234"), &actual)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("expected unmarshal to fail")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestWeakStringList_Marshal(t *testing.T) {
|
||||||
|
value := WeakStringList{"audience"}
|
||||||
|
|
||||||
|
expected := `["audience"]`
|
||||||
|
|
||||||
|
actual, err := json.Marshal(value)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if expected != string(actual) {
|
||||||
|
t.Errorf("expected marshaled list to be %v, got %v", expected, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func assertStringListEqual(t *testing.T, expected []string, actual []string) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
if len(expected) != len(actual) {
|
||||||
|
t.Errorf("length mismatch: expected %d long slice, got %d", len(expected), len(actual))
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range expected {
|
||||||
|
if v != actual[i] {
|
||||||
|
t.Errorf("expected %d. item to be %q, got %q", i, v, actual[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue