471f9e4e10
This change adds an account freeze service with methods for checking if a user is frozen, freezing a user, and unfreezing a user. Furthermore, methods for altering the usage limits of a user or project have been implemented for use by the account freeze service. Change-Id: I77fecfac5c152f134bec90165acfe4f1dea957e7
192 lines
5.6 KiB
Go
192 lines
5.6 KiB
Go
// Copyright (C) 2022 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package console
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/zeebo/errs"
|
|
|
|
"storj.io/common/uuid"
|
|
)
|
|
|
|
// ErrAccountFreeze is the class for errors that occur during operation of the account freeze service.
|
|
var ErrAccountFreeze = errs.Class("account freeze service")
|
|
|
|
// AccountFreezeEvents exposes methods to manage the account freeze events table in database.
|
|
//
|
|
// architecture: Database
|
|
type AccountFreezeEvents interface {
|
|
// Upsert is a method for updating an account freeze event if it exists and inserting it otherwise.
|
|
Upsert(ctx context.Context, event *AccountFreezeEvent) (*AccountFreezeEvent, error)
|
|
// Get is a method for querying account freeze event from the database by user ID and event type.
|
|
Get(ctx context.Context, userID uuid.UUID, eventType AccountFreezeEventType) (*AccountFreezeEvent, error)
|
|
// DeleteAllByUserID is a method for deleting all account freeze events from the database by user ID.
|
|
DeleteAllByUserID(ctx context.Context, userID uuid.UUID) error
|
|
}
|
|
|
|
// AccountFreezeEvent represents an event related to account freezing.
|
|
type AccountFreezeEvent struct {
|
|
UserID uuid.UUID
|
|
Type AccountFreezeEventType
|
|
Limits *AccountFreezeEventLimits
|
|
CreatedAt time.Time
|
|
}
|
|
|
|
// AccountFreezeEventLimits represents the usage limits for a user's account and projects before they were frozen.
|
|
type AccountFreezeEventLimits struct {
|
|
User UsageLimits `json:"user"`
|
|
Projects map[uuid.UUID]UsageLimits `json:"projects"`
|
|
}
|
|
|
|
// AccountFreezeEventType is used to indicate the account freeze event's type.
|
|
type AccountFreezeEventType int
|
|
|
|
const (
|
|
// Freeze signifies that the user has been frozen.
|
|
Freeze AccountFreezeEventType = 0
|
|
// Warning signifies that the user has been warned that they may be frozen soon.
|
|
Warning AccountFreezeEventType = 1
|
|
)
|
|
|
|
// AccountFreezeService encapsulates operations concerning account freezes.
|
|
type AccountFreezeService struct {
|
|
freezeEventsDB AccountFreezeEvents
|
|
usersDB Users
|
|
projectsDB Projects
|
|
}
|
|
|
|
// NewAccountFreezeService creates a new account freeze service.
|
|
func NewAccountFreezeService(freezeEventsDB AccountFreezeEvents, usersDB Users, projectsDB Projects) *AccountFreezeService {
|
|
return &AccountFreezeService{
|
|
freezeEventsDB: freezeEventsDB,
|
|
usersDB: usersDB,
|
|
projectsDB: projectsDB,
|
|
}
|
|
}
|
|
|
|
// IsUserFrozen returns whether the user specified by the given ID is frozen.
|
|
func (s *AccountFreezeService) IsUserFrozen(ctx context.Context, userID uuid.UUID) (_ bool, err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
_, err = s.freezeEventsDB.Get(ctx, userID, Freeze)
|
|
switch {
|
|
case errors.Is(err, sql.ErrNoRows):
|
|
return false, nil
|
|
case err != nil:
|
|
return false, ErrAccountFreeze.Wrap(err)
|
|
default:
|
|
return true, nil
|
|
}
|
|
}
|
|
|
|
// FreezeUser freezes the user specified by the given ID.
|
|
func (s *AccountFreezeService) FreezeUser(ctx context.Context, userID uuid.UUID) (err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
user, err := s.usersDB.Get(ctx, userID)
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
|
|
event, err := s.freezeEventsDB.Get(ctx, userID, Freeze)
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
event = &AccountFreezeEvent{
|
|
UserID: userID,
|
|
Type: Freeze,
|
|
Limits: &AccountFreezeEventLimits{
|
|
User: UsageLimits{
|
|
Storage: user.ProjectStorageLimit,
|
|
Bandwidth: user.ProjectBandwidthLimit,
|
|
Segment: user.ProjectSegmentLimit,
|
|
},
|
|
Projects: make(map[uuid.UUID]UsageLimits),
|
|
},
|
|
}
|
|
} else if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
|
|
userLimits := UsageLimits{
|
|
Storage: user.ProjectStorageLimit,
|
|
Bandwidth: user.ProjectBandwidthLimit,
|
|
Segment: user.ProjectSegmentLimit,
|
|
}
|
|
// If user limits have been zeroed already, we should not override what is in the freeze table.
|
|
if userLimits != (UsageLimits{}) {
|
|
event.Limits.User = userLimits
|
|
}
|
|
|
|
projects, err := s.projectsDB.GetOwn(ctx, userID)
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
for _, p := range projects {
|
|
projLimits := UsageLimits{}
|
|
if p.StorageLimit != nil {
|
|
projLimits.Storage = p.StorageLimit.Int64()
|
|
}
|
|
if p.BandwidthLimit != nil {
|
|
projLimits.Bandwidth = p.BandwidthLimit.Int64()
|
|
}
|
|
if p.SegmentLimit != nil {
|
|
projLimits.Segment = *p.SegmentLimit
|
|
}
|
|
// If project limits have been zeroed already, we should not override what is in the freeze table.
|
|
if projLimits != (UsageLimits{}) {
|
|
event.Limits.Projects[p.ID] = projLimits
|
|
}
|
|
}
|
|
|
|
_, err = s.freezeEventsDB.Upsert(ctx, event)
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
|
|
err = s.usersDB.UpdateUserProjectLimits(ctx, userID, UsageLimits{})
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
|
|
for _, proj := range projects {
|
|
err := s.projectsDB.UpdateUsageLimits(ctx, proj.ID, UsageLimits{})
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// UnfreezeUser reverses the freeze placed on the user specified by the given ID.
|
|
func (s *AccountFreezeService) UnfreezeUser(ctx context.Context, userID uuid.UUID) (err error) {
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
event, err := s.freezeEventsDB.Get(ctx, userID, Freeze)
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
return ErrAccountFreeze.New("user is not frozen")
|
|
}
|
|
|
|
if event.Limits == nil {
|
|
return ErrAccountFreeze.New("freeze event limits are nil")
|
|
}
|
|
|
|
for id, limits := range event.Limits.Projects {
|
|
err := s.projectsDB.UpdateUsageLimits(ctx, id, limits)
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
}
|
|
|
|
err = s.usersDB.UpdateUserProjectLimits(ctx, userID, event.Limits.User)
|
|
if err != nil {
|
|
return ErrAccountFreeze.Wrap(err)
|
|
}
|
|
|
|
return ErrAccountFreeze.Wrap(s.freezeEventsDB.DeleteAllByUserID(ctx, userID))
|
|
}
|