78 lines
3.1 KiB
Go
78 lines
3.1 KiB
Go
// Filename: internal/repository/key_writer.go
|
|
|
|
package repository
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"gemini-balancer/internal/errors"
|
|
"gemini-balancer/internal/models"
|
|
"strconv"
|
|
"time"
|
|
)
|
|
|
|
func (r *gormKeyRepository) UpdateKeyUsageTimestamp(ctx context.Context, groupID, keyID uint) {
|
|
lruKey := fmt.Sprintf(KeyGroupLRU, groupID)
|
|
timestamp := float64(time.Now().UnixMilli())
|
|
|
|
members := map[string]float64{
|
|
strconv.FormatUint(uint64(keyID), 10): timestamp,
|
|
}
|
|
|
|
if err := r.store.ZAdd(ctx, lruKey, members); err != nil {
|
|
r.logger.WithError(err).Warnf("Failed to update usage timestamp for key %d in group %d", keyID, groupID)
|
|
}
|
|
}
|
|
|
|
func (r *gormKeyRepository) SyncKeyStatusInPollingCaches(ctx context.Context, groupID, keyID uint, newStatus models.APIKeyStatus) {
|
|
r.logger.Infof("SYNC: Directly updating polling caches for G:%d K:%d -> %s", groupID, keyID, newStatus)
|
|
r.updatePollingCachesLogic(ctx, groupID, keyID, newStatus)
|
|
}
|
|
|
|
func (r *gormKeyRepository) HandleCacheUpdateEvent(ctx context.Context, groupID, keyID uint, newStatus models.APIKeyStatus) {
|
|
r.logger.Infof("EVENT: Updating polling caches for G:%d K:%d -> %s from an event", groupID, keyID, newStatus)
|
|
r.updatePollingCachesLogic(ctx, groupID, keyID, newStatus)
|
|
}
|
|
|
|
func (r *gormKeyRepository) updatePollingCachesLogic(ctx context.Context, groupID, keyID uint, newStatus models.APIKeyStatus) {
|
|
keyIDStr := strconv.FormatUint(uint64(keyID), 10)
|
|
sequentialKey := fmt.Sprintf(KeyGroupSequential, groupID)
|
|
lruKey := fmt.Sprintf(KeyGroupLRU, groupID)
|
|
mainPoolKey := fmt.Sprintf(KeyGroupRandomMain, groupID)
|
|
cooldownPoolKey := fmt.Sprintf(KeyGroupRandomCooldown, groupID)
|
|
|
|
_ = r.store.LRem(ctx, sequentialKey, 0, keyIDStr)
|
|
_ = r.store.ZRem(ctx, lruKey, keyIDStr)
|
|
_ = r.store.SRem(ctx, mainPoolKey, keyIDStr)
|
|
_ = r.store.SRem(ctx, cooldownPoolKey, keyIDStr)
|
|
|
|
if newStatus == models.StatusActive {
|
|
if err := r.store.LPush(ctx, sequentialKey, keyIDStr); err != nil {
|
|
r.logger.WithError(err).Warnf("Failed to add key %d to sequential list for group %d", keyID, groupID)
|
|
}
|
|
members := map[string]float64{keyIDStr: 0}
|
|
if err := r.store.ZAdd(ctx, lruKey, members); err != nil {
|
|
r.logger.WithError(err).Warnf("Failed to add key %d to LRU zset for group %d", keyID, groupID)
|
|
}
|
|
if err := r.store.SAdd(ctx, mainPoolKey, keyIDStr); err != nil {
|
|
r.logger.WithError(err).Warnf("Failed to add key %d to random main pool for group %d", keyID, groupID)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r *gormKeyRepository) UpdateKeyStatusAfterRequest(ctx context.Context, group *models.KeyGroup, key *models.APIKey, success bool, apiErr *errors.APIError) {
|
|
if success {
|
|
if group.PollingStrategy == models.StrategyWeighted {
|
|
go r.UpdateKeyUsageTimestamp(context.Background(), group.ID, key.ID)
|
|
}
|
|
return
|
|
}
|
|
if apiErr == nil {
|
|
r.logger.Warnf("Request failed for KeyID %d in GroupID %d but no specific API error was provided.", key.ID, group.ID)
|
|
return
|
|
}
|
|
r.logger.Warnf("Request failed for KeyID %d in GroupID %d with error: %s. Temporarily removing from active polling caches.", key.ID, group.ID, apiErr.Message)
|
|
|
|
r.SyncKeyStatusInPollingCaches(ctx, group.ID, key.ID, models.StatusCooldown)
|
|
}
|