mirror of
https://github.com/idanoo/autobrr
synced 2025-07-23 00:39:13 +00:00
refactor(http): api key cache handling (#1632)
This commit is contained in:
parent
0d53f7e5fc
commit
d13b421c42
6 changed files with 117 additions and 50 deletions
|
@ -10,6 +10,7 @@ import (
|
||||||
|
|
||||||
"github.com/autobrr/autobrr/internal/domain"
|
"github.com/autobrr/autobrr/internal/domain"
|
||||||
"github.com/autobrr/autobrr/internal/logger"
|
"github.com/autobrr/autobrr/internal/logger"
|
||||||
|
"github.com/autobrr/autobrr/pkg/errors"
|
||||||
|
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
)
|
)
|
||||||
|
@ -17,7 +18,6 @@ import (
|
||||||
type Service interface {
|
type Service interface {
|
||||||
List(ctx context.Context) ([]domain.APIKey, error)
|
List(ctx context.Context) ([]domain.APIKey, error)
|
||||||
Store(ctx context.Context, key *domain.APIKey) error
|
Store(ctx context.Context, key *domain.APIKey) error
|
||||||
Update(ctx context.Context, key *domain.APIKey) error
|
|
||||||
Delete(ctx context.Context, key string) error
|
Delete(ctx context.Context, key string) error
|
||||||
ValidateAPIKey(ctx context.Context, token string) bool
|
ValidateAPIKey(ctx context.Context, token string) bool
|
||||||
}
|
}
|
||||||
|
@ -26,63 +26,75 @@ type service struct {
|
||||||
log zerolog.Logger
|
log zerolog.Logger
|
||||||
repo domain.APIRepo
|
repo domain.APIRepo
|
||||||
|
|
||||||
keyCache []domain.APIKey
|
keyCache map[string]domain.APIKey
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewService(log logger.Logger, repo domain.APIRepo) Service {
|
func NewService(log logger.Logger, repo domain.APIRepo) Service {
|
||||||
return &service{
|
return &service{
|
||||||
log: log.With().Str("module", "api").Logger(),
|
log: log.With().Str("module", "api").Logger(),
|
||||||
repo: repo,
|
repo: repo,
|
||||||
keyCache: []domain.APIKey{},
|
keyCache: map[string]domain.APIKey{},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) List(ctx context.Context) ([]domain.APIKey, error) {
|
func (s *service) List(ctx context.Context) ([]domain.APIKey, error) {
|
||||||
if len(s.keyCache) > 0 {
|
if len(s.keyCache) > 0 {
|
||||||
return s.keyCache, nil
|
keys := make([]domain.APIKey, 0, len(s.keyCache))
|
||||||
|
|
||||||
|
for _, key := range s.keyCache {
|
||||||
|
keys = append(keys, key)
|
||||||
|
}
|
||||||
|
|
||||||
|
return keys, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return s.repo.GetKeys(ctx)
|
return s.repo.GetAllAPIKeys(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) Store(ctx context.Context, key *domain.APIKey) error {
|
func (s *service) Store(ctx context.Context, apiKey *domain.APIKey) error {
|
||||||
key.Key = GenerateSecureToken(16)
|
apiKey.Key = GenerateSecureToken(16)
|
||||||
|
|
||||||
if err := s.repo.Store(ctx, key); err != nil {
|
if err := s.repo.Store(ctx, apiKey); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(s.keyCache) > 0 {
|
if len(s.keyCache) > 0 {
|
||||||
// set new key
|
// set new apiKey
|
||||||
s.keyCache = append(s.keyCache, *key)
|
s.keyCache[apiKey.Key] = *apiKey
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) Update(ctx context.Context, key *domain.APIKey) error {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *service) Delete(ctx context.Context, key string) error {
|
func (s *service) Delete(ctx context.Context, key string) error {
|
||||||
// reset
|
err := s.repo.Delete(ctx, key)
|
||||||
s.keyCache = []domain.APIKey{}
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "could not delete api key: %s", key)
|
||||||
|
}
|
||||||
|
|
||||||
return s.repo.Delete(ctx, key)
|
// remove key from cache
|
||||||
|
delete(s.keyCache, key)
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) ValidateAPIKey(ctx context.Context, key string) bool {
|
func (s *service) ValidateAPIKey(ctx context.Context, key string) bool {
|
||||||
keys, err := s.repo.GetKeys(ctx)
|
if _, ok := s.keyCache[key]; ok {
|
||||||
|
s.log.Trace().Msgf("api service key cache hit: %s", key)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
apiKey, err := s.repo.GetKey(ctx, key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
s.log.Trace().Msgf("api service key cache invalid key: %s", key)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, k := range keys {
|
s.log.Trace().Msgf("api service key cache miss: %s", key)
|
||||||
if k.Key == key {
|
|
||||||
return true
|
s.keyCache[key] = *apiKey
|
||||||
}
|
|
||||||
}
|
return true
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateSecureToken(length int) string {
|
func GenerateSecureToken(length int) string {
|
||||||
|
|
|
@ -25,9 +25,8 @@ func NewAPIRepo(log logger.Logger, db *DB) domain.APIRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
type APIRepo struct {
|
type APIRepo struct {
|
||||||
log zerolog.Logger
|
log zerolog.Logger
|
||||||
db *DB
|
db *DB
|
||||||
cache map[string]domain.APIKey
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *APIRepo) Store(ctx context.Context, key *domain.APIKey) error {
|
func (r *APIRepo) Store(ctx context.Context, key *domain.APIKey) error {
|
||||||
|
@ -57,9 +56,7 @@ func (r *APIRepo) Store(ctx context.Context, key *domain.APIKey) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *APIRepo) Delete(ctx context.Context, key string) error {
|
func (r *APIRepo) Delete(ctx context.Context, key string) error {
|
||||||
queryBuilder := r.db.squirrel.
|
queryBuilder := r.db.squirrel.Delete("api_key").Where(sq.Eq{"key": key})
|
||||||
Delete("api_key").
|
|
||||||
Where(sq.Eq{"key": key})
|
|
||||||
|
|
||||||
query, args, err := queryBuilder.ToSql()
|
query, args, err := queryBuilder.ToSql()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -76,14 +73,9 @@ func (r *APIRepo) Delete(ctx context.Context, key string) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *APIRepo) GetKeys(ctx context.Context) ([]domain.APIKey, error) {
|
func (r *APIRepo) GetAllAPIKeys(ctx context.Context) ([]domain.APIKey, error) {
|
||||||
queryBuilder := r.db.squirrel.
|
queryBuilder := r.db.squirrel.
|
||||||
Select(
|
Select("name", "key", "scopes", "created_at").
|
||||||
"name",
|
|
||||||
"key",
|
|
||||||
"scopes",
|
|
||||||
"created_at",
|
|
||||||
).
|
|
||||||
From("api_key")
|
From("api_key")
|
||||||
|
|
||||||
query, args, err := queryBuilder.ToSql()
|
query, args, err := queryBuilder.ToSql()
|
||||||
|
@ -116,3 +108,35 @@ func (r *APIRepo) GetKeys(ctx context.Context) ([]domain.APIKey, error) {
|
||||||
|
|
||||||
return keys, nil
|
return keys, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *APIRepo) GetKey(ctx context.Context, key string) (*domain.APIKey, error) {
|
||||||
|
queryBuilder := r.db.squirrel.
|
||||||
|
Select("name", "key", "scopes", "created_at").
|
||||||
|
From("api_key").
|
||||||
|
Where(sq.Eq{"key": key})
|
||||||
|
|
||||||
|
query, args, err := queryBuilder.ToSql()
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "error building query")
|
||||||
|
}
|
||||||
|
|
||||||
|
row := r.db.handler.QueryRowContext(ctx, query, args...)
|
||||||
|
if err := row.Err(); err != nil {
|
||||||
|
if errors.Is(err, sql.ErrNoRows) {
|
||||||
|
return nil, domain.ErrRecordNotFound
|
||||||
|
}
|
||||||
|
return nil, errors.Wrap(err, "error executing query")
|
||||||
|
}
|
||||||
|
|
||||||
|
var apiKey domain.APIKey
|
||||||
|
|
||||||
|
var name sql.NullString
|
||||||
|
|
||||||
|
if err := row.Scan(&name, &apiKey.Key, pq.Array(&apiKey.Scopes), &apiKey.CreatedAt); err != nil {
|
||||||
|
return nil, errors.Wrap(err, "error scanning row")
|
||||||
|
}
|
||||||
|
|
||||||
|
apiKey.Name = name.String
|
||||||
|
|
||||||
|
return &apiKey, nil
|
||||||
|
}
|
||||||
|
|
|
@ -69,7 +69,7 @@ func TestAPIRepo_Delete(t *testing.T) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAPIRepo_GetKeys(t *testing.T) {
|
func TestAPIRepo_GetAllAPIKeys(t *testing.T) {
|
||||||
for dbType, db := range testDBs {
|
for dbType, db := range testDBs {
|
||||||
log := setupLoggerForTest()
|
log := setupLoggerForTest()
|
||||||
repo := NewAPIRepo(log, db)
|
repo := NewAPIRepo(log, db)
|
||||||
|
@ -77,7 +77,7 @@ func TestAPIRepo_GetKeys(t *testing.T) {
|
||||||
t.Run(fmt.Sprintf("GetKeys_Returns_Keys_If_Exists [%s]", dbType), func(t *testing.T) {
|
t.Run(fmt.Sprintf("GetKeys_Returns_Keys_If_Exists [%s]", dbType), func(t *testing.T) {
|
||||||
key := &domain.APIKey{Name: "TestKey", Key: "123", Scopes: []string{"read", "write"}}
|
key := &domain.APIKey{Name: "TestKey", Key: "123", Scopes: []string{"read", "write"}}
|
||||||
_ = repo.Store(context.Background(), key)
|
_ = repo.Store(context.Background(), key)
|
||||||
keys, err := repo.GetKeys(context.Background())
|
keys, err := repo.GetAllAPIKeys(context.Background())
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Greater(t, len(keys), 0)
|
assert.Greater(t, len(keys), 0)
|
||||||
// Cleanup
|
// Cleanup
|
||||||
|
@ -85,9 +85,32 @@ func TestAPIRepo_GetKeys(t *testing.T) {
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run(fmt.Sprintf("GetKeys_Returns_Empty_If_No_Keys [%s]", dbType), func(t *testing.T) {
|
t.Run(fmt.Sprintf("GetKeys_Returns_Empty_If_No_Keys [%s]", dbType), func(t *testing.T) {
|
||||||
keys, err := repo.GetKeys(context.Background())
|
keys, err := repo.GetAllAPIKeys(context.Background())
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, 0, len(keys))
|
assert.Equal(t, 0, len(keys))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestAPIRepo_GetKey(t *testing.T) {
|
||||||
|
for dbType, db := range testDBs {
|
||||||
|
log := setupLoggerForTest()
|
||||||
|
repo := NewAPIRepo(log, db)
|
||||||
|
|
||||||
|
t.Run(fmt.Sprintf("GetKey_Returns_Key_If_Exists [%s]", dbType), func(t *testing.T) {
|
||||||
|
key := &domain.APIKey{Name: "TestKey", Key: "123", Scopes: []string{"read", "write"}}
|
||||||
|
_ = repo.Store(context.Background(), key)
|
||||||
|
apiKey, err := repo.GetKey(context.Background(), key.Key)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.NotNil(t, apiKey)
|
||||||
|
// Cleanup
|
||||||
|
_ = repo.Delete(context.Background(), key.Key)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run(fmt.Sprintf("GetKeys_Returns_Empty_If_No_Keys [%s]", dbType), func(t *testing.T) {
|
||||||
|
key, err := repo.GetKey(context.Background(), "nonexistent")
|
||||||
|
assert.ErrorIs(t, err, domain.ErrRecordNotFound)
|
||||||
|
assert.Nil(t, key)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -11,7 +11,8 @@ import (
|
||||||
type APIRepo interface {
|
type APIRepo interface {
|
||||||
Store(ctx context.Context, key *APIKey) error
|
Store(ctx context.Context, key *APIKey) error
|
||||||
Delete(ctx context.Context, key string) error
|
Delete(ctx context.Context, key string) error
|
||||||
GetKeys(ctx context.Context) ([]APIKey, error)
|
GetAllAPIKeys(ctx context.Context) ([]APIKey, error)
|
||||||
|
GetKey(ctx context.Context, key string) (*APIKey, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type APIKey struct {
|
type APIKey struct {
|
||||||
|
|
|
@ -17,7 +17,6 @@ import (
|
||||||
type apikeyService interface {
|
type apikeyService interface {
|
||||||
List(ctx context.Context) ([]domain.APIKey, error)
|
List(ctx context.Context) ([]domain.APIKey, error)
|
||||||
Store(ctx context.Context, key *domain.APIKey) error
|
Store(ctx context.Context, key *domain.APIKey) error
|
||||||
Update(ctx context.Context, key *domain.APIKey) error
|
|
||||||
Delete(ctx context.Context, key string) error
|
Delete(ctx context.Context, key string) error
|
||||||
ValidateAPIKey(ctx context.Context, token string) bool
|
ValidateAPIKey(ctx context.Context, token string) bool
|
||||||
}
|
}
|
||||||
|
@ -51,18 +50,14 @@ func (h apikeyHandler) list(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h apikeyHandler) store(w http.ResponseWriter, r *http.Request) {
|
func (h apikeyHandler) store(w http.ResponseWriter, r *http.Request) {
|
||||||
|
var data domain.APIKey
|
||||||
var (
|
|
||||||
ctx = r.Context()
|
|
||||||
data domain.APIKey
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
||||||
h.encoder.Error(w, err)
|
h.encoder.Error(w, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := h.service.Store(ctx, &data); err != nil {
|
if err := h.service.Store(r.Context(), &data); err != nil {
|
||||||
h.encoder.Error(w, err)
|
h.encoder.Error(w, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,7 @@ func (e encoder) StatusResponse(w http.ResponseWriter, status int, response inte
|
||||||
if response != nil {
|
if response != nil {
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(status)
|
w.WriteHeader(status)
|
||||||
|
|
||||||
if err := json.NewEncoder(w).Encode(response); err != nil {
|
if err := json.NewEncoder(w).Encode(response); err != nil {
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
@ -37,6 +38,7 @@ func (e encoder) StatusResponseMessage(w http.ResponseWriter, status int, messag
|
||||||
if message != "" {
|
if message != "" {
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(status)
|
w.WriteHeader(status)
|
||||||
|
|
||||||
if err := json.NewEncoder(w).Encode(statusResponse{Message: message}); err != nil {
|
if err := json.NewEncoder(w).Encode(statusResponse{Message: message}); err != nil {
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
@ -53,6 +55,7 @@ func (e encoder) StatusCreated(w http.ResponseWriter) {
|
||||||
func (e encoder) StatusCreatedData(w http.ResponseWriter, data interface{}) {
|
func (e encoder) StatusCreatedData(w http.ResponseWriter, data interface{}) {
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(http.StatusCreated)
|
w.WriteHeader(http.StatusCreated)
|
||||||
|
|
||||||
if err := json.NewEncoder(w).Encode(data); err != nil {
|
if err := json.NewEncoder(w).Encode(data); err != nil {
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
@ -74,7 +77,11 @@ func (e encoder) NotFoundErr(w http.ResponseWriter, err error) {
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(http.StatusNotFound)
|
w.WriteHeader(http.StatusNotFound)
|
||||||
json.NewEncoder(w).Encode(res)
|
|
||||||
|
if err := json.NewEncoder(w).Encode(res); err != nil {
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e encoder) StatusInternalError(w http.ResponseWriter) {
|
func (e encoder) StatusInternalError(w http.ResponseWriter) {
|
||||||
|
@ -88,7 +95,11 @@ func (e encoder) Error(w http.ResponseWriter, err error) {
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
json.NewEncoder(w).Encode(res)
|
|
||||||
|
if err := json.NewEncoder(w).Encode(res); err != nil {
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e encoder) StatusError(w http.ResponseWriter, status int, err error) {
|
func (e encoder) StatusError(w http.ResponseWriter, status int, err error) {
|
||||||
|
@ -98,6 +109,7 @@ func (e encoder) StatusError(w http.ResponseWriter, status int, err error) {
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
w.WriteHeader(status)
|
w.WriteHeader(status)
|
||||||
|
|
||||||
if err := json.NewEncoder(w).Encode(res); err != nil {
|
if err := json.NewEncoder(w).Encode(res); err != nil {
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue