2023-09-03 00:23:48 +02:00
|
|
|
package sql
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"emperror.dev/errors"
|
|
|
|
"git.sleepycat.moe/sam/mercury/internal/database"
|
|
|
|
"github.com/jackc/pgx/v5"
|
|
|
|
"github.com/jackc/pgx/v5/pgconn"
|
|
|
|
"github.com/keegancsmith/sqlf"
|
|
|
|
"github.com/oklog/ulid/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// BlogStore is the interface to blogs in the database.
|
|
|
|
type BlogStore struct {
|
|
|
|
q Querier
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlogStore creates a new BlogStore instance.
|
|
|
|
func NewBlogStore(q Querier) *BlogStore {
|
|
|
|
return &BlogStore{q: q}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ByID gets a blog by its ID.
|
|
|
|
func (s *BlogStore) ByID(ctx context.Context, id ulid.ULID) (b database.Blog, err error) {
|
|
|
|
q := sqlf.Sprintf("SELECT * FROM blogs WHERE id = %s", id)
|
|
|
|
|
|
|
|
b, err = Get[database.Blog](ctx, s.q, q)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Cause(err) == pgx.ErrNoRows {
|
|
|
|
return b, ErrNotFound
|
|
|
|
}
|
|
|
|
return b, errors.Wrap(err, "getting blog")
|
|
|
|
}
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ByName gets a blog by name. If domain is empty, it gets a local blog.
|
|
|
|
func (s *BlogStore) ByName(ctx context.Context, name, domain string) (b database.Blog, err error) {
|
|
|
|
q := sqlf.Sprintf("SELECT * FROM blogs WHERE name = %s", name)
|
|
|
|
if domain != "" {
|
|
|
|
q = sqlf.Sprintf("%v AND domain = %s", q, domain)
|
|
|
|
} else {
|
|
|
|
q = sqlf.Sprintf("%v AND domain IS NULL", q)
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err = Get[database.Blog](ctx, s.q, q)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Cause(err) == pgx.ErrNoRows {
|
|
|
|
return b, ErrNotFound
|
|
|
|
}
|
|
|
|
return b, errors.Wrap(err, "getting blog")
|
|
|
|
}
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create creates a blog.
|
|
|
|
func (s *BlogStore) Create(ctx context.Context, accountID ulid.ULID, name string) (b database.Blog, err error) {
|
|
|
|
q := sqlf.Sprintf(
|
|
|
|
"INSERT INTO blogs (id, name, domain, bio, account_id) VALUES (%s, %s, NULL, '', %s) RETURNING *",
|
|
|
|
makeULID(), name, accountID)
|
|
|
|
|
|
|
|
b, err = Get[database.Blog](ctx, s.q, q)
|
|
|
|
if err != nil {
|
|
|
|
pge := &pgconn.PgError{}
|
|
|
|
if errors.As(err, &pge) {
|
|
|
|
// unique constraint violation
|
|
|
|
if pge.Code == "23505" {
|
|
|
|
return b, ErrUsernameTaken
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, errors.Wrap(err, "creating blog")
|
|
|
|
}
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *BlogStore) LocalCount(ctx context.Context) (count int64, err error) {
|
|
|
|
q := sqlf.Sprintf("SELECT count(*) FROM blogs WHERE domain IS NULL")
|
|
|
|
|
|
|
|
err = s.q.QueryRow(ctx, q.Query(sqlf.PostgresBindVar)).Scan(&count)
|
|
|
|
if err != nil {
|
|
|
|
return count, errors.Wrap(err, "getting count")
|
|
|
|
}
|
|
|
|
return count, nil
|
|
|
|
}
|
2023-09-06 16:32:33 +02:00
|
|
|
|
2023-10-15 21:34:13 +02:00
|
|
|
type BlogFollower struct {
|
|
|
|
AccountID ulid.ULID
|
|
|
|
IsLocal bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *BlogStore) Followers(ctx context.Context, id ulid.ULID) ([]BlogFollower, error) {
|
|
|
|
q := sqlf.Sprintf(`SELECT f.account_id, a.domain IS NOT NULL AS is_local
|
|
|
|
FROM account_follows f JOIN accounts a ON a.id = f.account_id WHERE f.blog_id = %s`, id)
|
2023-09-06 16:32:33 +02:00
|
|
|
|
2023-10-15 21:34:13 +02:00
|
|
|
follows, err := Select[BlogFollower](ctx, s.q, q)
|
2023-09-06 16:32:33 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "executing query")
|
|
|
|
}
|
|
|
|
return follows, nil
|
|
|
|
}
|