2023-02-15 02:52:11 +01:00
|
|
|
package access
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-08-22 12:49:22 +02:00
|
|
|
"database/sql"
|
2023-09-15 16:58:45 +02:00
|
|
|
"errors"
|
2023-02-15 02:52:11 +01:00
|
|
|
"time"
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/api/authz"
|
|
|
|
"github.com/zitadel/zitadel/internal/command"
|
2023-02-27 22:36:43 +01:00
|
|
|
"github.com/zitadel/zitadel/internal/database"
|
2023-02-15 02:52:11 +01:00
|
|
|
"github.com/zitadel/zitadel/internal/logstore"
|
2023-09-15 16:58:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/logstore/record"
|
|
|
|
"github.com/zitadel/zitadel/internal/query"
|
|
|
|
"github.com/zitadel/zitadel/internal/query/projection"
|
2023-02-15 02:52:11 +01:00
|
|
|
"github.com/zitadel/zitadel/internal/repository/quota"
|
2023-09-15 16:58:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/telemetry/tracing"
|
2023-02-15 02:52:11 +01:00
|
|
|
)
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
var _ logstore.UsageStorer[*record.AccessLog] = (*databaseLogStorage)(nil)
|
2023-02-15 02:52:11 +01:00
|
|
|
|
|
|
|
type databaseLogStorage struct {
|
2023-02-27 22:36:43 +01:00
|
|
|
dbClient *database.DB
|
2023-09-15 16:58:45 +02:00
|
|
|
commands *command.Commands
|
|
|
|
queries *query.Queries
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
func NewDatabaseLogStorage(dbClient *database.DB, commands *command.Commands, queries *query.Queries) *databaseLogStorage {
|
|
|
|
return &databaseLogStorage{dbClient: dbClient, commands: commands, queries: queries}
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (l *databaseLogStorage) QuotaUnit() quota.Unit {
|
|
|
|
return quota.RequestsAllAuthenticated
|
|
|
|
}
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
func (l *databaseLogStorage) Emit(ctx context.Context, bulk []*record.AccessLog) error {
|
2023-03-01 18:05:12 +01:00
|
|
|
if len(bulk) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2023-09-15 16:58:45 +02:00
|
|
|
return l.incrementUsage(ctx, bulk)
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
func (l *databaseLogStorage) incrementUsage(ctx context.Context, bulk []*record.AccessLog) (err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2023-02-15 02:52:11 +01:00
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
byInstance := make(map[string][]*record.AccessLog)
|
|
|
|
for _, r := range bulk {
|
|
|
|
if r.InstanceID != "" {
|
|
|
|
byInstance[r.InstanceID] = append(byInstance[r.InstanceID], r)
|
|
|
|
}
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
2023-09-15 16:58:45 +02:00
|
|
|
for instanceID, instanceBulk := range byInstance {
|
|
|
|
q, getQuotaErr := l.queries.GetQuota(ctx, instanceID, quota.RequestsAllAuthenticated)
|
|
|
|
if errors.Is(getQuotaErr, sql.ErrNoRows) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
err = errors.Join(err, getQuotaErr)
|
|
|
|
if getQuotaErr != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
sum, incrementErr := l.incrementUsageFromAccessLogs(ctx, instanceID, q.CurrentPeriodStart, instanceBulk)
|
|
|
|
err = errors.Join(err, incrementErr)
|
|
|
|
if incrementErr != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
notifications, getNotificationErr := l.queries.GetDueQuotaNotifications(ctx, instanceID, quota.RequestsAllAuthenticated, q, q.CurrentPeriodStart, sum)
|
|
|
|
err = errors.Join(err, getNotificationErr)
|
|
|
|
if getNotificationErr != nil || len(notifications) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ctx = authz.WithInstanceID(ctx, instanceID)
|
|
|
|
reportErr := l.commands.ReportQuotaUsage(ctx, notifications)
|
|
|
|
err = errors.Join(err, reportErr)
|
|
|
|
if reportErr != nil {
|
|
|
|
continue
|
|
|
|
}
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
2023-09-15 16:58:45 +02:00
|
|
|
return err
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
func (l *databaseLogStorage) incrementUsageFromAccessLogs(ctx context.Context, instanceID string, periodStart time.Time, records []*record.AccessLog) (sum uint64, err error) {
|
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
2023-02-15 02:52:11 +01:00
|
|
|
|
2023-09-15 16:58:45 +02:00
|
|
|
var count uint64
|
|
|
|
for _, r := range records {
|
|
|
|
if r.IsAuthenticated() {
|
|
|
|
count++
|
|
|
|
}
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|
2023-09-15 16:58:45 +02:00
|
|
|
return projection.QuotaProjection.IncrementUsage(ctx, quota.RequestsAllAuthenticated, instanceID, periodStart, count)
|
2023-02-15 02:52:11 +01:00
|
|
|
}
|