mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-10 17:23:39 +00:00
1a49b7d298
* project quota added * project quota removed * add periods table * make log record generic * accumulate usage * query usage * count action run seconds * fix filter in ReportQuotaUsage * fix existing tests * fix logstore tests * fix typo * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * move notifications into debouncer and improve limit querying * cleanup * comment * fix: add quota unit tests command side * fix remaining quota usage query * implement InmemLogStorage * cleanup and linting * improve test * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * fix: add quota unit tests command side * action notifications and fixes for notifications query * revert console prefix * fix: add quota unit tests command side * fix: add quota integration tests * improve accountable requests * improve accountable requests * fix: add quota integration tests * fix: add quota integration tests * fix: add quota integration tests * comment * remove ability to store logs in db and other changes requested from review * changes requested from review * changes requested from review * Update internal/api/http/middleware/access_interceptor.go Co-authored-by: Silvan <silvan.reusser@gmail.com> * tests: fix quotas integration tests * improve incrementUsageStatement * linting * fix: delete e2e tests as intergation tests cover functionality * Update internal/api/http/middleware/access_interceptor.go Co-authored-by: Silvan <silvan.reusser@gmail.com> * backup * fix conflict * create rc * create prerelease * remove issue release labeling * fix tracing --------- Co-authored-by: Livio Spring <livio.a@gmail.com> Co-authored-by: Stefan Benz <stefan@caos.ch> Co-authored-by: adlerhurst <silvan.reusser@gmail.com>
122 lines
3.1 KiB
Go
122 lines
3.1 KiB
Go
package query
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
errs "errors"
|
|
"time"
|
|
|
|
sq "github.com/Masterminds/squirrel"
|
|
|
|
"github.com/zitadel/zitadel/internal/database"
|
|
"github.com/zitadel/zitadel/internal/errors"
|
|
"github.com/zitadel/zitadel/internal/query/projection"
|
|
"github.com/zitadel/zitadel/internal/repository/quota"
|
|
"github.com/zitadel/zitadel/internal/telemetry/tracing"
|
|
)
|
|
|
|
var (
|
|
quotasTable = table{
|
|
name: projection.QuotasProjectionTable,
|
|
instanceIDCol: projection.QuotaColumnInstanceID,
|
|
}
|
|
QuotaColumnID = Column{
|
|
name: projection.QuotaColumnID,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnInstanceID = Column{
|
|
name: projection.QuotaColumnInstanceID,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnUnit = Column{
|
|
name: projection.QuotaColumnUnit,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnAmount = Column{
|
|
name: projection.QuotaColumnAmount,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnLimit = Column{
|
|
name: projection.QuotaColumnLimit,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnInterval = Column{
|
|
name: projection.QuotaColumnInterval,
|
|
table: quotasTable,
|
|
}
|
|
QuotaColumnFrom = Column{
|
|
name: projection.QuotaColumnFrom,
|
|
table: quotasTable,
|
|
}
|
|
)
|
|
|
|
type Quota struct {
|
|
ID string
|
|
From time.Time
|
|
ResetInterval time.Duration
|
|
Amount uint64
|
|
Limit bool
|
|
CurrentPeriodStart time.Time
|
|
}
|
|
|
|
func (q *Queries) GetQuota(ctx context.Context, instanceID string, unit quota.Unit) (qu *Quota, err error) {
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
defer func() { span.EndWithError(err) }()
|
|
query, scan := prepareQuotaQuery(ctx, q.client)
|
|
stmt, args, err := query.Where(
|
|
sq.Eq{
|
|
QuotaColumnInstanceID.identifier(): instanceID,
|
|
QuotaColumnUnit.identifier(): unit,
|
|
},
|
|
).ToSql()
|
|
if err != nil {
|
|
return nil, errors.ThrowInternal(err, "QUERY-XmYn9", "Errors.Query.SQLStatement")
|
|
}
|
|
err = q.client.QueryRowContext(ctx, func(row *sql.Row) error {
|
|
qu, err = scan(row)
|
|
return err
|
|
}, stmt, args...)
|
|
return qu, err
|
|
}
|
|
|
|
func prepareQuotaQuery(ctx context.Context, db prepareDatabase) (sq.SelectBuilder, func(*sql.Row) (*Quota, error)) {
|
|
return sq.
|
|
Select(
|
|
QuotaColumnID.identifier(),
|
|
QuotaColumnFrom.identifier(),
|
|
QuotaColumnInterval.identifier(),
|
|
QuotaColumnAmount.identifier(),
|
|
QuotaColumnLimit.identifier(),
|
|
"now()",
|
|
).
|
|
From(quotasTable.identifier()).
|
|
PlaceholderFormat(sq.Dollar), func(row *sql.Row) (*Quota, error) {
|
|
q := new(Quota)
|
|
var interval database.Duration
|
|
var now time.Time
|
|
err := row.Scan(&q.ID, &q.From, &interval, &q.Amount, &q.Limit, &now)
|
|
if err != nil {
|
|
if errs.Is(err, sql.ErrNoRows) {
|
|
return nil, errors.ThrowNotFound(err, "QUERY-rDTM6", "Errors.Quota.NotExisting")
|
|
}
|
|
return nil, errors.ThrowInternal(err, "QUERY-LqySK", "Errors.Internal")
|
|
}
|
|
q.ResetInterval = time.Duration(interval)
|
|
q.CurrentPeriodStart = pushPeriodStart(q.From, q.ResetInterval, now)
|
|
return q, nil
|
|
}
|
|
}
|
|
|
|
func pushPeriodStart(from time.Time, interval time.Duration, now time.Time) time.Time {
|
|
if now.IsZero() {
|
|
now = time.Now()
|
|
}
|
|
for {
|
|
next := from.Add(interval)
|
|
if next.After(now) {
|
|
return from
|
|
}
|
|
from = next
|
|
}
|
|
}
|