2023-03-16 20:05:39 +01:00
|
|
|
package limits
|
|
|
|
|
|
|
|
import (
|
2023-03-21 19:06:23 +01:00
|
|
|
"fmt"
|
|
|
|
"github.com/jmoiron/sqlx"
|
2023-03-27 21:29:25 +02:00
|
|
|
"github.com/openziti/zrok/controller/emailUi"
|
2023-03-16 20:05:39 +01:00
|
|
|
"github.com/openziti/zrok/controller/metrics"
|
|
|
|
"github.com/openziti/zrok/controller/store"
|
|
|
|
"github.com/openziti/zrok/controller/zrokEdgeSdk"
|
2023-03-17 18:51:12 +01:00
|
|
|
"github.com/openziti/zrok/util"
|
2023-03-21 19:06:23 +01:00
|
|
|
"github.com/pkg/errors"
|
2023-03-16 20:05:39 +01:00
|
|
|
"github.com/sirupsen/logrus"
|
2023-03-28 20:39:42 +02:00
|
|
|
"reflect"
|
2023-03-17 18:13:33 +01:00
|
|
|
"time"
|
2023-03-16 20:05:39 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Agent struct {
|
2023-03-23 22:16:35 +01:00
|
|
|
cfg *Config
|
|
|
|
ifx *influxReader
|
|
|
|
zCfg *zrokEdgeSdk.Config
|
|
|
|
str *store.Store
|
|
|
|
queue chan *metrics.Usage
|
2023-03-27 17:43:58 +02:00
|
|
|
acctWarningActions []AccountAction
|
|
|
|
acctLimitActions []AccountAction
|
|
|
|
acctRelaxActions []AccountAction
|
|
|
|
envWarningActions []EnvironmentAction
|
|
|
|
envLimitActions []EnvironmentAction
|
|
|
|
envRelaxActions []EnvironmentAction
|
|
|
|
shrWarningActions []ShareAction
|
|
|
|
shrLimitActions []ShareAction
|
|
|
|
shrRelaxActions []ShareAction
|
2023-03-23 22:16:35 +01:00
|
|
|
close chan struct{}
|
|
|
|
join chan struct{}
|
2023-03-16 20:05:39 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 21:29:25 +02:00
|
|
|
func NewAgent(cfg *Config, ifxCfg *metrics.InfluxConfig, zCfg *zrokEdgeSdk.Config, emailCfg *emailUi.Config, str *store.Store) (*Agent, error) {
|
2023-03-27 17:34:29 +02:00
|
|
|
edge, err := zrokEdgeSdk.Client(zCfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
a := &Agent{
|
2023-03-27 17:53:18 +02:00
|
|
|
cfg: cfg,
|
|
|
|
ifx: newInfluxReader(ifxCfg),
|
|
|
|
zCfg: zCfg,
|
|
|
|
str: str,
|
|
|
|
queue: make(chan *metrics.Usage, 1024),
|
2023-03-27 21:29:25 +02:00
|
|
|
acctWarningActions: []AccountAction{newAccountWarningAction(emailCfg, str, edge)},
|
2023-03-27 17:53:18 +02:00
|
|
|
acctLimitActions: []AccountAction{newAccountLimitAction(str, edge)},
|
|
|
|
acctRelaxActions: []AccountAction{newAccountRelaxAction(str, edge)},
|
2023-03-27 21:29:25 +02:00
|
|
|
envWarningActions: []EnvironmentAction{newEnvironmentWarningAction(emailCfg, str, edge)},
|
2023-03-27 17:53:18 +02:00
|
|
|
envLimitActions: []EnvironmentAction{newEnvironmentLimitAction(str, edge)},
|
|
|
|
envRelaxActions: []EnvironmentAction{newEnvironmentRelaxAction(str, edge)},
|
2023-03-27 21:29:25 +02:00
|
|
|
shrWarningActions: []ShareAction{newShareWarningAction(emailCfg, str, edge)},
|
2023-03-27 17:53:18 +02:00
|
|
|
shrLimitActions: []ShareAction{newShareLimitAction(str, edge)},
|
|
|
|
shrRelaxActions: []ShareAction{newShareRelaxAction(str, edge)},
|
|
|
|
close: make(chan struct{}),
|
|
|
|
join: make(chan struct{}),
|
2023-03-27 17:34:29 +02:00
|
|
|
}
|
|
|
|
return a, nil
|
2023-03-16 20:05:39 +01:00
|
|
|
}
|
|
|
|
|
2023-03-17 18:13:33 +01:00
|
|
|
func (a *Agent) Start() {
|
|
|
|
go a.run()
|
2023-03-16 20:05:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Agent) Stop() {
|
2023-03-17 18:13:33 +01:00
|
|
|
close(a.close)
|
|
|
|
<-a.join
|
2023-03-16 20:05:39 +01:00
|
|
|
}
|
|
|
|
|
2023-03-21 21:34:45 +01:00
|
|
|
func (a *Agent) CanCreateEnvironment(acctId int, trx *sqlx.Tx) (bool, error) {
|
2023-03-29 19:29:12 +02:00
|
|
|
if a.cfg.Enforcing {
|
2023-04-05 19:57:22 +02:00
|
|
|
if empty, err := a.str.IsAccountLimitJournalEmpty(acctId, trx); err == nil && !empty {
|
|
|
|
alj, err := a.str.FindLatestAccountLimitJournal(acctId, trx)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if alj.Action == store.LimitAction {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
} else if err != nil {
|
2023-03-21 21:34:45 +01:00
|
|
|
return false, err
|
|
|
|
}
|
2023-03-29 19:29:12 +02:00
|
|
|
|
|
|
|
if a.cfg.Environments > Unlimited {
|
|
|
|
envs, err := a.str.FindEnvironmentsForAccount(acctId, trx)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if len(envs)+1 > a.cfg.Environments {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
}
|
2023-03-21 21:34:45 +01:00
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|
2023-03-21 21:18:17 +01:00
|
|
|
|
2023-03-29 19:29:12 +02:00
|
|
|
func (a *Agent) CanCreateShare(acctId, envId int, trx *sqlx.Tx) (bool, error) {
|
|
|
|
if a.cfg.Enforcing {
|
|
|
|
if empty, err := a.str.IsAccountLimitJournalEmpty(acctId, trx); err == nil && !empty {
|
|
|
|
alj, err := a.str.FindLatestAccountLimitJournal(acctId, trx)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if alj.Action == store.LimitAction {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
} else if err != nil {
|
2023-03-21 21:18:17 +01:00
|
|
|
return false, err
|
|
|
|
}
|
2023-03-29 19:29:12 +02:00
|
|
|
|
|
|
|
if empty, err := a.str.IsEnvironmentLimitJournalEmpty(envId, trx); err == nil && !empty {
|
|
|
|
elj, err := a.str.FindLatestEnvironmentLimitJournal(envId, trx)
|
2023-03-21 21:34:45 +01:00
|
|
|
if err != nil {
|
2023-03-29 19:29:12 +02:00
|
|
|
return false, err
|
2023-03-21 21:34:45 +01:00
|
|
|
}
|
2023-03-29 19:29:12 +02:00
|
|
|
if elj.Action == store.LimitAction {
|
2023-03-21 21:34:45 +01:00
|
|
|
return false, nil
|
|
|
|
}
|
2023-03-29 19:29:12 +02:00
|
|
|
} else if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if a.cfg.Shares > Unlimited {
|
|
|
|
envs, err := a.str.FindEnvironmentsForAccount(acctId, trx)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
total := 0
|
|
|
|
for i := range envs {
|
|
|
|
shrs, err := a.str.FindSharesForEnvironment(envs[i].Id, trx)
|
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrapf(err, "unable to find shares for environment '%v'", envs[i].ZId)
|
|
|
|
}
|
|
|
|
total += len(shrs)
|
|
|
|
if total+1 > a.cfg.Shares {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
logrus.Infof("total = %d", total)
|
|
|
|
}
|
2023-03-21 21:18:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2023-03-16 20:05:39 +01:00
|
|
|
func (a *Agent) Handle(u *metrics.Usage) error {
|
2023-03-21 18:05:22 +01:00
|
|
|
logrus.Debugf("handling: %v", u)
|
|
|
|
a.queue <- u
|
2023-03-16 20:05:39 +01:00
|
|
|
return nil
|
|
|
|
}
|
2023-03-17 18:13:33 +01:00
|
|
|
|
|
|
|
func (a *Agent) run() {
|
|
|
|
logrus.Info("started")
|
|
|
|
defer logrus.Info("stopped")
|
|
|
|
|
2023-03-29 23:03:42 +02:00
|
|
|
lastCycle := time.Now()
|
2023-03-17 18:13:33 +01:00
|
|
|
mainLoop:
|
|
|
|
for {
|
|
|
|
select {
|
2023-03-21 18:05:22 +01:00
|
|
|
case usage := <-a.queue:
|
2023-03-23 20:13:59 +01:00
|
|
|
if err := a.enforce(usage); err != nil {
|
|
|
|
logrus.Errorf("error running enforcement: %v", err)
|
|
|
|
}
|
2023-03-29 23:03:42 +02:00
|
|
|
if time.Since(lastCycle) > a.cfg.Cycle {
|
|
|
|
if err := a.relax(); err != nil {
|
|
|
|
logrus.Errorf("error running relax cycle: %v", err)
|
|
|
|
}
|
|
|
|
lastCycle = time.Now()
|
|
|
|
}
|
2023-03-21 18:05:22 +01:00
|
|
|
|
2023-03-17 18:13:33 +01:00
|
|
|
case <-time.After(a.cfg.Cycle):
|
2023-03-23 20:13:59 +01:00
|
|
|
if err := a.relax(); err != nil {
|
|
|
|
logrus.Errorf("error running relax cycle: %v", err)
|
|
|
|
}
|
2023-03-29 23:03:42 +02:00
|
|
|
lastCycle = time.Now()
|
2023-03-17 18:13:33 +01:00
|
|
|
|
|
|
|
case <-a.close:
|
|
|
|
close(a.join)
|
|
|
|
break mainLoop
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-21 18:05:22 +01:00
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
func (a *Agent) enforce(u *metrics.Usage) error {
|
|
|
|
trx, err := a.str.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "error starting transaction")
|
|
|
|
}
|
|
|
|
defer func() { _ = trx.Rollback() }()
|
|
|
|
|
2023-03-27 19:00:05 +02:00
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkAccountLimit(u.AccountId); err == nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
if enforce {
|
2023-03-22 19:10:07 +01:00
|
|
|
enforced := false
|
|
|
|
var enforcedAt time.Time
|
|
|
|
if empty, err := a.str.IsAccountLimitJournalEmpty(int(u.AccountId), trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestAccountLimitJournal(int(u.AccountId), trx); err == nil {
|
|
|
|
enforced = latest.Action == store.LimitAction
|
|
|
|
enforcedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
2023-03-22 18:09:21 +01:00
|
|
|
|
2023-03-22 19:10:07 +01:00
|
|
|
if !enforced {
|
|
|
|
_, err := a.str.CreateAccountLimitJournal(&store.AccountLimitJournal{
|
|
|
|
AccountId: int(u.AccountId),
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.LimitAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
acct, err := a.str.GetAccount(int(u.AccountId), trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// run account limit actions
|
|
|
|
for _, action := range a.acctLimitActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleAccount(acct, rxBytes, txBytes, a.cfg.Bandwidth.PerAccount, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already enforced limit for account '#%d' at %v", u.AccountId, enforcedAt)
|
2023-03-22 18:09:21 +01:00
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
} else if warning {
|
2023-03-22 19:10:07 +01:00
|
|
|
warned := false
|
|
|
|
var warnedAt time.Time
|
|
|
|
if empty, err := a.str.IsAccountLimitJournalEmpty(int(u.AccountId), trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestAccountLimitJournal(int(u.AccountId), trx); err == nil {
|
|
|
|
warned = latest.Action == store.WarningAction || latest.Action == store.LimitAction
|
|
|
|
warnedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !warned {
|
|
|
|
_, err := a.str.CreateAccountLimitJournal(&store.AccountLimitJournal{
|
|
|
|
AccountId: int(u.AccountId),
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.WarningAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
acct, err := a.str.GetAccount(int(u.AccountId), trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// run account warning actions
|
|
|
|
for _, action := range a.acctWarningActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleAccount(acct, rxBytes, txBytes, a.cfg.Bandwidth.PerAccount, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already warned account '#%d' at %v", u.AccountId, warnedAt)
|
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
} else {
|
2023-03-27 19:00:05 +02:00
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkEnvironmentLimit(u.EnvironmentId); err == nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
if enforce {
|
2023-03-22 19:10:07 +01:00
|
|
|
enforced := false
|
|
|
|
var enforcedAt time.Time
|
|
|
|
if empty, err := a.str.IsEnvironmentLimitJournalEmpty(int(u.EnvironmentId), trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestEnvironmentLimitJournal(int(u.EnvironmentId), trx); err == nil {
|
|
|
|
enforced = latest.Action == store.LimitAction
|
|
|
|
enforcedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !enforced {
|
|
|
|
_, err := a.str.CreateEnvironmentLimitJournal(&store.EnvironmentLimitJournal{
|
|
|
|
EnvironmentId: int(u.EnvironmentId),
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.LimitAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
env, err := a.str.GetEnvironment(int(u.EnvironmentId), trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// run environment limit actions
|
|
|
|
for _, action := range a.envLimitActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleEnvironment(env, rxBytes, txBytes, a.cfg.Bandwidth.PerEnvironment, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already enforced limit for environment '#%d' at %v", u.EnvironmentId, enforcedAt)
|
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
} else if warning {
|
2023-03-22 19:10:07 +01:00
|
|
|
warned := false
|
|
|
|
var warnedAt time.Time
|
|
|
|
if empty, err := a.str.IsEnvironmentLimitJournalEmpty(int(u.EnvironmentId), trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestEnvironmentLimitJournal(int(u.EnvironmentId), trx); err == nil {
|
|
|
|
warned = latest.Action == store.WarningAction || latest.Action == store.LimitAction
|
|
|
|
warnedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !warned {
|
|
|
|
_, err := a.str.CreateEnvironmentLimitJournal(&store.EnvironmentLimitJournal{
|
|
|
|
EnvironmentId: int(u.EnvironmentId),
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.WarningAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
env, err := a.str.GetEnvironment(int(u.EnvironmentId), trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// run environment warning actions
|
|
|
|
for _, action := range a.envWarningActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleEnvironment(env, rxBytes, txBytes, a.cfg.Bandwidth.PerEnvironment, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already warned environment '#%d' at %v", u.EnvironmentId, warnedAt)
|
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
} else {
|
2023-03-27 19:00:05 +02:00
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkShareLimit(u.ShareToken); err == nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
if enforce {
|
2023-03-22 19:10:07 +01:00
|
|
|
shr, err := a.str.FindShareWithToken(u.ShareToken, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
enforced := false
|
|
|
|
var enforcedAt time.Time
|
|
|
|
if empty, err := a.str.IsShareLimitJournalEmpty(shr.Id, trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestShareLimitJournal(shr.Id, trx); err == nil {
|
|
|
|
enforced = latest.Action == store.LimitAction
|
|
|
|
enforcedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !enforced {
|
|
|
|
_, err := a.str.CreateShareLimitJournal(&store.ShareLimitJournal{
|
|
|
|
ShareId: shr.Id,
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.LimitAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
// run share limit actions
|
|
|
|
for _, action := range a.shrLimitActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleShare(shr, rxBytes, txBytes, a.cfg.Bandwidth.PerShare, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already enforced limit for share '%v' at %v", shr.Token, enforcedAt)
|
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
} else if warning {
|
2023-03-22 19:10:07 +01:00
|
|
|
shr, err := a.str.FindShareWithToken(u.ShareToken, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
warned := false
|
|
|
|
var warnedAt time.Time
|
|
|
|
if empty, err := a.str.IsShareLimitJournalEmpty(shr.Id, trx); err == nil && !empty {
|
|
|
|
if latest, err := a.str.FindLatestShareLimitJournal(shr.Id, trx); err == nil {
|
|
|
|
warned = latest.Action == store.WarningAction || latest.Action == store.LimitAction
|
|
|
|
warnedAt = latest.UpdatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !warned {
|
|
|
|
_, err := a.str.CreateShareLimitJournal(&store.ShareLimitJournal{
|
|
|
|
ShareId: shr.Id,
|
2023-03-29 23:03:42 +02:00
|
|
|
RxBytes: rxBytes,
|
|
|
|
TxBytes: txBytes,
|
2023-03-22 19:10:07 +01:00
|
|
|
Action: store.WarningAction,
|
|
|
|
}, trx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
// run share warning actions
|
|
|
|
for _, action := range a.shrWarningActions {
|
2023-03-27 19:51:48 +02:00
|
|
|
if err := action.HandleShare(shr, rxBytes, txBytes, a.cfg.Bandwidth.PerShare, trx); err != nil {
|
2023-03-28 20:39:42 +02:00
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-22 19:10:07 +01:00
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Debugf("already warned share '%v' at %v", shr.Token, warnedAt)
|
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-23 20:13:59 +01:00
|
|
|
func (a *Agent) relax() error {
|
2023-03-28 21:19:01 +02:00
|
|
|
logrus.Debug("relaxing")
|
2023-03-23 20:13:59 +01:00
|
|
|
|
|
|
|
trx, err := a.str.Begin()
|
2023-03-21 19:06:23 +01:00
|
|
|
if err != nil {
|
2023-03-23 20:13:59 +01:00
|
|
|
return errors.Wrap(err, "error starting transaction")
|
|
|
|
}
|
|
|
|
defer func() { _ = trx.Rollback() }()
|
|
|
|
|
|
|
|
commit := false
|
|
|
|
|
|
|
|
if sljs, err := a.str.FindAllLatestShareLimitJournal(trx); err == nil {
|
|
|
|
for _, slj := range sljs {
|
|
|
|
if shr, err := a.str.GetShare(slj.ShareId, trx); err == nil {
|
2023-03-27 19:00:05 +02:00
|
|
|
if slj.Action == store.WarningAction || slj.Action == store.LimitAction {
|
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkShareLimit(shr.Token); err == nil {
|
2023-03-23 20:13:59 +01:00
|
|
|
if !enforce && !warning {
|
2023-03-29 22:24:52 +02:00
|
|
|
if slj.Action == store.LimitAction {
|
|
|
|
// run relax actions for share
|
|
|
|
for _, action := range a.shrRelaxActions {
|
|
|
|
if err := action.HandleShare(shr, rxBytes, txBytes, a.cfg.Bandwidth.PerShare, trx); err != nil {
|
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
2023-03-29 23:03:42 +02:00
|
|
|
} else {
|
|
|
|
logrus.Infof("relaxing warning for '%v'", shr.Token)
|
2023-03-23 20:13:59 +01:00
|
|
|
}
|
|
|
|
if err := a.str.DeleteShareLimitJournalForShare(shr.Id, trx); err == nil {
|
|
|
|
commit = true
|
|
|
|
} else {
|
2023-03-27 19:00:05 +02:00
|
|
|
logrus.Errorf("error deleting share_limit_journal for '%v'", shr.Token)
|
2023-03-23 20:13:59 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Infof("share '%v' still over limit", shr.Token)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error checking share limit for '%v': %v", shr.Token, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error getting share for '#%d': %v", slj.ShareId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if eljs, err := a.str.FindAllLatestEnvironmentLimitJournal(trx); err == nil {
|
|
|
|
for _, elj := range eljs {
|
|
|
|
if env, err := a.str.GetEnvironment(elj.EnvironmentId, trx); err == nil {
|
2023-03-27 19:00:05 +02:00
|
|
|
if elj.Action == store.WarningAction || elj.Action == store.LimitAction {
|
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkEnvironmentLimit(int64(elj.EnvironmentId)); err == nil {
|
2023-03-23 20:13:59 +01:00
|
|
|
if !enforce && !warning {
|
2023-03-29 22:24:52 +02:00
|
|
|
if elj.Action == store.LimitAction {
|
|
|
|
// run relax actions for environment
|
|
|
|
for _, action := range a.envRelaxActions {
|
|
|
|
if err := action.HandleEnvironment(env, rxBytes, txBytes, a.cfg.Bandwidth.PerEnvironment, trx); err != nil {
|
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
2023-03-29 23:03:42 +02:00
|
|
|
} else {
|
|
|
|
logrus.Infof("relaxing warning for '%v'", env.ZId)
|
2023-03-23 20:13:59 +01:00
|
|
|
}
|
|
|
|
if err := a.str.DeleteEnvironmentLimitJournalForEnvironment(env.Id, trx); err == nil {
|
|
|
|
commit = true
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error deleteing environment_limit_journal for '%v': %v", env.ZId, err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Infof("environment '%v' still over limit", env.ZId)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error checking environment limit for '%v': %v", env.ZId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error getting environment for '#%d': %v", elj.EnvironmentId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if aljs, err := a.str.FindAllLatestAccountLimitJournal(trx); err == nil {
|
|
|
|
for _, alj := range aljs {
|
|
|
|
if acct, err := a.str.GetAccount(alj.AccountId, trx); err == nil {
|
2023-03-27 19:00:05 +02:00
|
|
|
if alj.Action == store.WarningAction || alj.Action == store.LimitAction {
|
|
|
|
if enforce, warning, rxBytes, txBytes, err := a.checkAccountLimit(int64(alj.AccountId)); err == nil {
|
2023-03-23 20:13:59 +01:00
|
|
|
if !enforce && !warning {
|
2023-03-29 22:24:52 +02:00
|
|
|
if alj.Action == store.LimitAction {
|
|
|
|
// run relax actions for account
|
|
|
|
for _, action := range a.acctRelaxActions {
|
|
|
|
if err := action.HandleAccount(acct, rxBytes, txBytes, a.cfg.Bandwidth.PerAccount, trx); err != nil {
|
|
|
|
return errors.Wrapf(err, "%v", reflect.TypeOf(action).String())
|
|
|
|
}
|
2023-03-27 19:00:05 +02:00
|
|
|
}
|
2023-03-29 23:03:42 +02:00
|
|
|
} else {
|
|
|
|
logrus.Infof("relaxing warning for '%v'", acct.Email)
|
2023-03-23 20:13:59 +01:00
|
|
|
}
|
|
|
|
if err := a.str.DeleteAccountLimitJournalForAccount(acct.Id, trx); err == nil {
|
|
|
|
commit = true
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error deleting account_limit_journal for '%v': %v", acct.Email, err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Infof("account '%v' still over limit", acct.Email)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error checking account limit for '%v': %v", acct.Email, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logrus.Errorf("error getting account for '#%d': %v", alj.AccountId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return err
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
|
2023-03-23 20:13:59 +01:00
|
|
|
if commit {
|
|
|
|
if err := trx.Commit(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-27 19:00:05 +02:00
|
|
|
func (a *Agent) checkAccountLimit(acctId int64) (enforce, warning bool, rxBytes, txBytes int64, err error) {
|
2023-03-21 19:06:23 +01:00
|
|
|
period := 24 * time.Hour
|
|
|
|
limit := DefaultBandwidthPerPeriod()
|
2023-03-21 18:05:22 +01:00
|
|
|
if a.cfg.Bandwidth != nil && a.cfg.Bandwidth.PerAccount != nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
limit = a.cfg.Bandwidth.PerAccount
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
if limit.Period > 0 {
|
|
|
|
period = limit.Period
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-23 20:13:59 +01:00
|
|
|
rx, tx, err := a.ifx.totalRxTxForAccount(acctId, period)
|
2023-03-21 18:05:22 +01:00
|
|
|
if err != nil {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
|
|
|
|
enforce, warning = a.checkLimit(limit, rx, tx)
|
2023-03-27 19:00:05 +02:00
|
|
|
return enforce, warning, rx, tx, nil
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 19:00:05 +02:00
|
|
|
func (a *Agent) checkEnvironmentLimit(envId int64) (enforce, warning bool, rxBytes, txBytes int64, err error) {
|
2023-03-21 19:06:23 +01:00
|
|
|
period := 24 * time.Hour
|
|
|
|
limit := DefaultBandwidthPerPeriod()
|
2023-03-21 18:05:22 +01:00
|
|
|
if a.cfg.Bandwidth != nil && a.cfg.Bandwidth.PerEnvironment != nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
limit = a.cfg.Bandwidth.PerEnvironment
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
if limit.Period > 0 {
|
|
|
|
period = limit.Period
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-23 20:13:59 +01:00
|
|
|
rx, tx, err := a.ifx.totalRxTxForEnvironment(envId, period)
|
2023-03-21 18:05:22 +01:00
|
|
|
if err != nil {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
|
|
|
|
2023-03-21 19:06:23 +01:00
|
|
|
enforce, warning = a.checkLimit(limit, rx, tx)
|
2023-03-27 19:00:05 +02:00
|
|
|
return enforce, warning, rx, tx, nil
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 19:00:05 +02:00
|
|
|
func (a *Agent) checkShareLimit(shrToken string) (enforce, warning bool, rxBytes, txBytes int64, err error) {
|
2023-03-21 19:06:23 +01:00
|
|
|
period := 24 * time.Hour
|
|
|
|
limit := DefaultBandwidthPerPeriod()
|
2023-03-21 18:05:22 +01:00
|
|
|
if a.cfg.Bandwidth != nil && a.cfg.Bandwidth.PerShare != nil {
|
2023-03-21 19:06:23 +01:00
|
|
|
limit = a.cfg.Bandwidth.PerShare
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
if limit.Period > 0 {
|
|
|
|
period = limit.Period
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|
2023-03-23 20:13:59 +01:00
|
|
|
rx, tx, err := a.ifx.totalRxTxForShare(shrToken, period)
|
2023-03-21 18:05:22 +01:00
|
|
|
if err != nil {
|
|
|
|
logrus.Error(err)
|
|
|
|
}
|
2023-03-21 19:06:23 +01:00
|
|
|
|
|
|
|
enforce, warning = a.checkLimit(limit, rx, tx)
|
|
|
|
if enforce || warning {
|
2023-03-27 21:29:25 +02:00
|
|
|
logrus.Debugf("'%v': %v", shrToken, describeLimit(limit, rx, tx))
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 19:00:05 +02:00
|
|
|
return enforce, warning, rx, tx, nil
|
2023-03-21 19:06:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Agent) checkLimit(cfg *BandwidthPerPeriod, rx, tx int64) (enforce, warning bool) {
|
|
|
|
if cfg.Limit.Rx != Unlimited && rx > cfg.Limit.Rx {
|
|
|
|
return true, false
|
|
|
|
}
|
|
|
|
if cfg.Limit.Tx != Unlimited && tx > cfg.Limit.Tx {
|
|
|
|
return true, false
|
|
|
|
}
|
|
|
|
if cfg.Limit.Total != Unlimited && rx+tx > cfg.Limit.Total {
|
|
|
|
return true, false
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Warning.Rx != Unlimited && rx > cfg.Warning.Rx {
|
|
|
|
return false, true
|
|
|
|
}
|
|
|
|
if cfg.Warning.Tx != Unlimited && tx > cfg.Warning.Tx {
|
|
|
|
return false, true
|
|
|
|
}
|
|
|
|
if cfg.Warning.Total != Unlimited && rx+tx > cfg.Warning.Total {
|
|
|
|
return false, true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false, false
|
|
|
|
}
|
|
|
|
|
2023-03-27 21:29:25 +02:00
|
|
|
func describeLimit(cfg *BandwidthPerPeriod, rx, tx int64) string {
|
2023-03-21 19:06:23 +01:00
|
|
|
out := ""
|
|
|
|
|
|
|
|
if cfg.Limit.Rx != Unlimited && rx > cfg.Limit.Rx {
|
|
|
|
out += fmt.Sprintf("['%v' over rx limit '%v']", util.BytesToSize(rx), util.BytesToSize(cfg.Limit.Rx))
|
|
|
|
}
|
|
|
|
if cfg.Limit.Tx != Unlimited && tx > cfg.Limit.Tx {
|
|
|
|
out += fmt.Sprintf("['%v' over tx limit '%v']", util.BytesToSize(tx), util.BytesToSize(cfg.Limit.Tx))
|
|
|
|
}
|
|
|
|
if cfg.Limit.Total != Unlimited && rx+tx > cfg.Limit.Total {
|
|
|
|
out += fmt.Sprintf("['%v' over total limit '%v']", util.BytesToSize(rx+tx), util.BytesToSize(cfg.Limit.Total))
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Warning.Rx != Unlimited && rx > cfg.Warning.Rx {
|
|
|
|
out += fmt.Sprintf("['%v' over rx warning '%v']", util.BytesToSize(rx), util.BytesToSize(cfg.Warning.Rx))
|
|
|
|
}
|
|
|
|
if cfg.Warning.Tx != Unlimited && tx > cfg.Warning.Tx {
|
|
|
|
out += fmt.Sprintf("['%v' over tx warning '%v']", util.BytesToSize(tx), util.BytesToSize(cfg.Warning.Tx))
|
|
|
|
}
|
|
|
|
if cfg.Warning.Total != Unlimited && rx+tx > cfg.Warning.Total {
|
|
|
|
out += fmt.Sprintf("['%v' over total warning '%v']", util.BytesToSize(rx+tx), util.BytesToSize(cfg.Warning.Total))
|
|
|
|
}
|
|
|
|
|
|
|
|
return out
|
2023-03-21 18:05:22 +01:00
|
|
|
}
|