Path: blob/main/components/public-api-server/pkg/auth/personal_access_token_test.go
2500 views
// Copyright (c) 2022 Gitpod GmbH. All rights reserved.1// Licensed under the GNU Affero General Public License (AGPL).2// See License.AGPL.txt in the project root for license information.34package auth56import (7"crypto/sha256"8"encoding/base64"9"encoding/hex"10"fmt"11"regexp"12"testing"1314"github.com/stretchr/testify/require"15)1617func TestGeneratePersonalAccessToken(t *testing.T) {18signer := NewHS256Signer([]byte("my-secret"))1920pat, err := GeneratePersonalAccessToken(signer)21require.NoError(t, err)2223signature, err := signer.Sign([]byte(pat.value))24require.NoError(t, err)2526require.NotEmpty(t, pat.value)27require.Len(t, pat.value, 40)28require.Equal(t, PersonalAccessToken{29prefix: PersonalAccessTokenPrefix,30value: pat.value,31signature: base64.RawURLEncoding.EncodeToString(signature),32}, pat)33require.Equal(t, fmt.Sprintf("%s%s.%s", pat.prefix, pat.signature, pat.value), pat.String())3435// must also be able to parse the token back36parsed, err := ParsePersonalAccessToken(pat.String(), signer)37require.NoError(t, err)38require.Equal(t, pat, parsed)39}4041func TestPersonalAccessToken_HashValue(t *testing.T) {42signer := NewHS256Signer([]byte("my-secret"))43pat, err := GeneratePersonalAccessToken(signer)44require.NoError(t, err)4546h := sha256.Sum256([]byte(pat.value))4748require.Equal(t, hex.EncodeToString(h[:]), pat.ValueHash(), "hash value must be hex sha-256 hash of value")49}5051func TestParsePersonalAccessToken_Errors(t *testing.T) {52signer := NewHS256Signer([]byte("my-secret"))5354scenarios := []struct {55Name string56Token string57}{58{59Name: "empty token is rejected",60Token: "",61},62{63Name: "invalid prefix",64Token: "gitpod_yolo_fooo",65},66{67Name: "invalid token with correct prefix",68Token: "gitpod_pat_foo",69},70{71Name: "invalid token with correct prefix and empty value and signature",72Token: "gitpod_pat_.",73},74{75Name: "invalid token with correct prefix but missing signature",76Token: "gitpod_pat_.value",77},78{79Name: "invalid token with correct prefix but missing value",80Token: "gitpod_pat_signature.",81},82{83Name: "invalid signature",84Token: "gitpod_pat_signature.value",85},86}8788for _, s := range scenarios {89t.Run(s.Name, func(t *testing.T) {90_, err := ParsePersonalAccessToken(s.Token, signer)91require.Error(t, err)92})9394}95}9697func TestGenerateTokenValue_OnlyAlphaNumberic(t *testing.T) {98sizes := []int{10, 20, 30, 40, 50, 60, 70, 80}99100var tokens []string101for _, size := range sizes {102for i := 0; i < 10; i++ {103token, err := generateTokenValue(size)104require.NoError(t, err)105106tokens = append(tokens, token)107}108}109110for _, token := range tokens {111rxp := regexp.MustCompile(`([a-zA-Z]|\d)+`)112require.Regexp(t, rxp, token, "must match alphanumeric")113}114}115116func TestGenerateTokenValue_FailsWithSizeZeroOrSmaller(t *testing.T) {117_, err := generateTokenValue(0)118require.Error(t, err)119120_, err = generateTokenValue(-1)121require.Error(t, err)122}123124125