apollo-backend/internal/repository/postgres_account.go

309 lines
8 KiB
Go
Raw Normal View History

2021-07-26 16:34:26 +00:00
package repository
2021-07-24 20:17:54 +00:00
import (
"context"
2022-03-28 21:05:01 +00:00
"time"
2021-07-24 20:17:54 +00:00
2022-11-03 17:43:10 +00:00
"go.opentelemetry.io/otel"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/codes"
"go.opentelemetry.io/otel/trace"
2021-07-26 16:34:26 +00:00
"github.com/christianselig/apollo-backend/internal/domain"
2021-07-24 20:17:54 +00:00
)
type postgresAccountRepository struct {
2022-11-03 17:43:10 +00:00
conn Connection
tracer trace.Tracer
2021-07-24 20:17:54 +00:00
}
func NewPostgresAccount(conn Connection) domain.AccountRepository {
2022-11-03 17:43:10 +00:00
tracer := otel.Tracer("db:postgres:accounts")
return &postgresAccountRepository{conn: conn, tracer: tracer}
2021-07-24 20:17:54 +00:00
}
func (p *postgresAccountRepository) fetch(ctx context.Context, query string, args ...interface{}) ([]domain.Account, error) {
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
rows, err := p.conn.Query(ctx, query, args...)
2021-07-24 20:17:54 +00:00
if err != nil {
2022-11-03 17:43:10 +00:00
span.SetStatus(codes.Error, "failed querying accounts")
span.RecordError(err)
2021-07-24 20:17:54 +00:00
return nil, err
}
defer rows.Close()
var accs []domain.Account
for rows.Next() {
var acc domain.Account
if err := rows.Scan(
&acc.ID,
&acc.Username,
&acc.AccountID,
&acc.AccessToken,
&acc.RefreshToken,
2022-03-28 21:05:01 +00:00
&acc.TokenExpiresAt,
2021-07-24 20:17:54 +00:00
&acc.LastMessageID,
2022-03-28 21:05:01 +00:00
&acc.NextNotificationCheckAt,
&acc.NextStuckNotificationCheckAt,
&acc.CheckCount,
2021-07-24 20:17:54 +00:00
); err != nil {
return nil, err
}
accs = append(accs, acc)
}
return accs, nil
}
func (p *postgresAccountRepository) GetByID(ctx context.Context, id int64) (domain.Account, error) {
query := `
2022-03-28 21:27:07 +00:00
SELECT id, username, reddit_account_id, access_token, refresh_token, token_expires_at,
2022-03-28 21:05:01 +00:00
last_message_id, next_notification_check_at, next_stuck_notification_check_at,
check_count
2021-07-24 20:17:54 +00:00
FROM accounts
2022-11-01 16:45:11 +00:00
WHERE id = $1 AND is_deleted IS FALSE`
2021-07-24 20:17:54 +00:00
accs, err := p.fetch(ctx, query, id)
if err != nil {
return domain.Account{}, err
}
if len(accs) == 0 {
return domain.Account{}, domain.ErrNotFound
}
return accs[0], nil
}
func (p *postgresAccountRepository) GetByRedditID(ctx context.Context, id string) (domain.Account, error) {
query := `
2022-03-28 21:27:07 +00:00
SELECT id, username, reddit_account_id, access_token, refresh_token, token_expires_at,
2022-03-28 21:05:01 +00:00
last_message_id, next_notification_check_at, next_stuck_notification_check_at,
check_count
2021-07-24 20:17:54 +00:00
FROM accounts
2022-11-01 16:45:11 +00:00
WHERE reddit_account_id = $1 AND is_deleted IS FALSE`
2021-07-24 20:17:54 +00:00
accs, err := p.fetch(ctx, query, id)
if err != nil {
return domain.Account{}, err
}
if len(accs) == 0 {
return domain.Account{}, domain.ErrNotFound
}
return accs[0], nil
}
2021-07-26 16:34:26 +00:00
func (p *postgresAccountRepository) CreateOrUpdate(ctx context.Context, acc *domain.Account) error {
query := `
2022-03-28 21:27:07 +00:00
INSERT INTO accounts (username, reddit_account_id, access_token, refresh_token, token_expires_at,
2022-11-01 16:45:11 +00:00
last_message_id, next_notification_check_at, next_stuck_notification_check_at, is_deleted)
2022-11-05 19:55:07 +00:00
VALUES ($1, $2, $3, $4, $5, $6, NOW(), NOW(), FALSE)
2021-07-26 16:34:26 +00:00
ON CONFLICT(username) DO
UPDATE SET access_token = $3,
refresh_token = $4,
2022-11-01 16:45:11 +00:00
token_expires_at = $5,
2022-11-01 17:14:33 +00:00
is_deleted = FALSE
2021-07-26 16:34:26 +00:00
RETURNING id`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if err := p.conn.QueryRow(
2021-07-26 16:34:26 +00:00
ctx,
query,
acc.Username,
acc.AccountID,
acc.AccessToken,
acc.RefreshToken,
2022-03-28 21:05:01 +00:00
acc.TokenExpiresAt,
2022-11-05 19:55:07 +00:00
acc.LastMessageID,
2022-11-03 17:43:10 +00:00
).Scan(&acc.ID); err != nil {
span.SetStatus(codes.Error, "failed upserting account")
span.RecordError(err)
return err
}
return nil
2021-07-26 16:34:26 +00:00
}
2021-07-24 20:17:54 +00:00
func (p *postgresAccountRepository) Create(ctx context.Context, acc *domain.Account) error {
query := `
INSERT INTO accounts
2022-03-28 21:27:07 +00:00
(username, reddit_account_id, access_token, refresh_token, token_expires_at,
2022-11-01 16:45:11 +00:00
last_message_id, next_notification_check_at, next_stuck_notification_check_at, is_deleted)
2022-11-01 17:14:33 +00:00
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, FALSE)
2021-07-24 20:17:54 +00:00
RETURNING id`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if err := p.conn.QueryRow(
2021-07-24 20:17:54 +00:00
ctx,
query,
acc.Username,
acc.AccountID,
acc.AccessToken,
acc.RefreshToken,
2022-03-28 21:05:01 +00:00
acc.TokenExpiresAt,
2021-07-24 20:17:54 +00:00
acc.LastMessageID,
2022-03-28 21:05:01 +00:00
acc.NextNotificationCheckAt,
acc.NextStuckNotificationCheckAt,
2022-11-03 17:43:10 +00:00
).Scan(&acc.ID); err != nil {
span.SetStatus(codes.Error, "failed inserting account")
span.RecordError(err)
return err
}
return nil
2021-07-24 20:17:54 +00:00
}
func (p *postgresAccountRepository) Update(ctx context.Context, acc *domain.Account) error {
query := `
UPDATE accounts
SET username = $2,
2022-03-28 21:27:07 +00:00
reddit_account_id = $3,
2021-07-24 20:17:54 +00:00
access_token = $4,
refresh_token = $5,
2022-03-28 21:05:01 +00:00
token_expires_at = $6,
2021-07-24 20:17:54 +00:00
last_message_id = $7,
2022-03-28 21:05:01 +00:00
next_notification_check_at = $8,
next_stuck_notification_check_at = $9,
check_count = $10
2021-07-24 20:17:54 +00:00
WHERE id = $1`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if _, err := p.conn.Exec(
2021-07-24 20:17:54 +00:00
ctx,
query,
2021-08-14 17:42:28 +00:00
acc.ID,
acc.Username,
2021-07-24 20:17:54 +00:00
acc.AccountID,
acc.AccessToken,
acc.RefreshToken,
2022-03-28 21:05:01 +00:00
acc.TokenExpiresAt,
2021-07-24 20:17:54 +00:00
acc.LastMessageID,
2022-03-28 21:05:01 +00:00
acc.NextNotificationCheckAt,
acc.NextStuckNotificationCheckAt,
acc.CheckCount,
2022-11-03 17:43:10 +00:00
); err != nil {
span.SetStatus(codes.Error, "failed to update account")
span.RecordError(err)
return err
}
2021-07-24 20:17:54 +00:00
2022-11-03 17:43:10 +00:00
return nil
2021-07-24 20:17:54 +00:00
}
func (p *postgresAccountRepository) Delete(ctx context.Context, id int64) error {
2022-11-01 16:45:11 +00:00
query := `UPDATE accounts SET is_deleted = TRUE WHERE id = $1`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if _, err := p.conn.Exec(ctx, query, id); err != nil {
span.SetStatus(codes.Error, "failed to delete account")
span.RecordError(err)
return err
}
return nil
2021-07-24 20:17:54 +00:00
}
2021-07-27 14:05:50 +00:00
func (p *postgresAccountRepository) Associate(ctx context.Context, acc *domain.Account, dev *domain.Device) error {
2021-08-08 18:19:47 +00:00
query := `
INSERT INTO devices_accounts
(account_id, device_id)
VALUES ($1, $2)
ON CONFLICT(account_id, device_id) DO NOTHING`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if _, err := p.conn.Exec(ctx, query, acc.ID, dev.ID); err != nil {
span.SetStatus(codes.Error, "failed to associate account to device")
span.RecordError(err)
return err
}
return nil
2021-08-08 18:19:47 +00:00
}
func (p *postgresAccountRepository) Disassociate(ctx context.Context, acc *domain.Account, dev *domain.Device) error {
query := `DELETE FROM devices_accounts WHERE account_id = $1 AND device_id = $2`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
if _, err := p.conn.Exec(ctx, query, acc.ID, dev.ID); err != nil {
span.SetStatus(codes.Error, "failed to disassociate account from device")
span.RecordError(err)
return err
}
return nil
2021-08-08 18:19:47 +00:00
}
func (p *postgresAccountRepository) GetByAPNSToken(ctx context.Context, token string) ([]domain.Account, error) {
query := `
2022-03-28 21:27:07 +00:00
SELECT accounts.id, username, accounts.reddit_account_id, access_token, refresh_token, token_expires_at,
2022-03-28 21:05:01 +00:00
last_message_id, next_notification_check_at, next_stuck_notification_check_at,
check_count
2021-08-08 18:19:47 +00:00
FROM accounts
INNER JOIN devices_accounts ON accounts.id = devices_accounts.account_id
INNER JOIN devices ON devices.id = devices_accounts.device_id
2022-11-01 16:45:11 +00:00
WHERE devices.apns_token = $1
AND accounts.is_deleted IS FALSE`
2021-08-08 18:19:47 +00:00
return p.fetch(ctx, query, token)
}
2022-03-28 21:05:01 +00:00
func (p *postgresAccountRepository) PruneStale(ctx context.Context, expiry time.Time) (int64, error) {
2021-08-14 15:59:13 +00:00
query := `
2022-11-01 16:45:11 +00:00
UPDATE accounts
SET is_deleted = TRUE
2022-03-28 21:05:01 +00:00
WHERE token_expires_at < $1`
2021-08-14 15:59:13 +00:00
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
res, err := p.conn.Exec(ctx, query, expiry)
2022-11-03 17:43:10 +00:00
if err != nil {
span.SetStatus(codes.Error, "failed to prune stale accounts")
span.RecordError(err)
}
span.SetAttributes(attribute.Int64("db.result.rows_affected", res.RowsAffected()))
2021-08-14 15:59:13 +00:00
return res.RowsAffected(), err
}
func (p *postgresAccountRepository) PruneOrphaned(ctx context.Context) (int64, error) {
2021-08-08 18:19:47 +00:00
query := `
WITH accounts_with_device_count AS (
SELECT accounts.id, COUNT(device_id) AS device_count
FROM accounts
LEFT JOIN devices_accounts ON accounts.id = devices_accounts.account_id
GROUP BY accounts.id
)
2022-11-01 16:45:11 +00:00
UPDATE accounts
SET is_deleted = TRUE
WHERE id IN (
2021-08-08 18:19:47 +00:00
SELECT id
FROM accounts_with_device_count
WHERE device_count = 0
)`
2022-11-03 17:43:10 +00:00
ctx, span := spanWithQuery(ctx, p.tracer, query)
defer span.End()
res, err := p.conn.Exec(ctx, query)
2022-11-03 17:43:10 +00:00
if err != nil {
span.SetStatus(codes.Error, "failed to prune orphaned accounts")
span.RecordError(err)
}
span.SetAttributes(attribute.Int64("db.result.rows_affected", res.RowsAffected()))
2021-08-08 18:19:47 +00:00
return res.RowsAffected(), err
2021-07-27 14:05:50 +00:00
}