2024-02-24 21:00:04 +00:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2024-03-20 16:36:45 +00:00
|
|
|
accountColumns = "id, uid, username, cookies, profile_image, api_key"
|
2024-02-24 21:00:04 +00:00
|
|
|
accountTable = "account"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Account struct {
|
2024-03-20 16:36:45 +00:00
|
|
|
ID *int64 `json:"id"`
|
|
|
|
UID *string `json:"uid"`
|
|
|
|
Username *string `json:"username"`
|
|
|
|
Cookies *string `json:"cookies"`
|
|
|
|
ProfileImage *string `json:"profile_image"`
|
|
|
|
ApiKey *string `json:"api_key"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Account) values() []any {
|
|
|
|
return []any{a.ID, a.UID, a.Username, a.Cookies, a.ProfileImage, a.ApiKey}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Account) valuesNoID() []any {
|
|
|
|
return a.values()[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Account) valuesEndID() []any {
|
|
|
|
vals := a.values()
|
|
|
|
return append(vals[1:], vals[0])
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type sqlAccount struct {
|
2024-03-20 16:36:45 +00:00
|
|
|
id sql.NullInt64
|
|
|
|
uid sql.NullString
|
|
|
|
username sql.NullString
|
|
|
|
cookies sql.NullString
|
|
|
|
profileImage sql.NullString
|
|
|
|
apiKey sql.NullString
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sa *sqlAccount) scan(r Row) error {
|
2024-03-20 16:36:45 +00:00
|
|
|
return r.Scan(&sa.id, &sa.uid, &sa.username, &sa.cookies, &sa.profileImage, &sa.apiKey)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sa sqlAccount) toAccount() *Account {
|
|
|
|
var a Account
|
|
|
|
a.ID = toInt64(sa.id)
|
2024-03-20 16:36:45 +00:00
|
|
|
a.UID = toString(sa.uid)
|
2024-02-24 21:00:04 +00:00
|
|
|
a.Username = toString(sa.username)
|
|
|
|
a.Cookies = toString(sa.cookies)
|
2024-03-20 16:36:45 +00:00
|
|
|
a.ProfileImage = toString(sa.profileImage)
|
|
|
|
a.ApiKey = toString(sa.apiKey)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
|
|
|
type AccountService interface {
|
2024-03-20 16:36:45 +00:00
|
|
|
All() ([]Account, error)
|
2024-02-24 21:00:04 +00:00
|
|
|
AutoMigrate() error
|
|
|
|
ByUsername(username string) (*Account, error)
|
|
|
|
Create(a *Account) error
|
|
|
|
DestructiveReset() error
|
|
|
|
Update(a *Account) error
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAccountService(db *sql.DB) AccountService {
|
|
|
|
return &accountService{
|
|
|
|
Database: db,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ AccountService = &accountService{}
|
|
|
|
|
|
|
|
type accountService struct {
|
|
|
|
Database *sql.DB
|
|
|
|
}
|
|
|
|
|
2024-03-20 16:36:45 +00:00
|
|
|
func (as *accountService) All() ([]Account, error) {
|
|
|
|
selectQ := fmt.Sprintf(`
|
|
|
|
SELECT %s
|
|
|
|
FROM "%s"
|
|
|
|
`, accountColumns, accountTable)
|
|
|
|
|
|
|
|
rows, err := as.Database.Query(selectQ)
|
|
|
|
if err != nil {
|
|
|
|
return nil, pkgErr("error executing select query", err)
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
accounts := []Account{}
|
|
|
|
for rows.Next() {
|
|
|
|
sa := &sqlAccount{}
|
|
|
|
|
|
|
|
err = sa.scan(rows)
|
|
|
|
if err != nil {
|
|
|
|
return nil, pkgErr("error scanning row", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
accounts = append(accounts, *sa.toAccount())
|
|
|
|
}
|
|
|
|
err = rows.Err()
|
|
|
|
if err != nil && err != sql.ErrNoRows {
|
|
|
|
return nil, pkgErr("error iterating over rows", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return accounts, nil
|
|
|
|
}
|
|
|
|
|
2024-02-24 21:00:04 +00:00
|
|
|
func (as *accountService) AutoMigrate() error {
|
|
|
|
err := as.createAccountTable()
|
|
|
|
if err != nil {
|
2024-03-20 16:36:45 +00:00
|
|
|
return pkgErr(fmt.Sprintf("error creating %s table", accountTable), err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) createAccountTable() error {
|
|
|
|
createQ := fmt.Sprintf(`
|
|
|
|
CREATE TABLE IF NOT EXISTS "%s" (
|
|
|
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
2024-03-20 16:36:45 +00:00
|
|
|
uid TEXT UNIQUE,
|
2024-02-24 21:00:04 +00:00
|
|
|
username TEXT UNIQUE NOT NULL,
|
2024-03-20 16:36:45 +00:00
|
|
|
cookies TEXT,
|
|
|
|
profile_image TEXT,
|
|
|
|
api_key TEXT
|
2024-02-24 21:00:04 +00:00
|
|
|
)
|
|
|
|
`, accountTable)
|
|
|
|
|
|
|
|
_, err := as.Database.Exec(createQ)
|
|
|
|
if err != nil {
|
2024-03-20 16:36:45 +00:00
|
|
|
return fmt.Errorf("error executing create query: %v", err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) ByUsername(username string) (*Account, error) {
|
|
|
|
err := runAccountValFuncs(
|
|
|
|
&Account{Username: &username},
|
|
|
|
accountRequireUsername,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, pkgErr("", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
selectQ := fmt.Sprintf(`
|
|
|
|
SELECT %s
|
|
|
|
FROM "%s"
|
|
|
|
WHERE username=?
|
|
|
|
`, accountColumns, accountTable)
|
|
|
|
|
|
|
|
var sa sqlAccount
|
|
|
|
row := as.Database.QueryRow(selectQ, username)
|
|
|
|
err = sa.scan(row)
|
|
|
|
if err != nil {
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2024-03-20 16:36:45 +00:00
|
|
|
return nil, pkgErr("error executing select query", err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return sa.toAccount(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) Create(a *Account) error {
|
|
|
|
err := runAccountValFuncs(
|
|
|
|
a,
|
|
|
|
accountRequireUsername,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return pkgErr("invalid account", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
columns := columnsNoID(accountColumns)
|
|
|
|
insertQ := fmt.Sprintf(`
|
|
|
|
INSERT INTO "%s" (%s)
|
|
|
|
VALUES (%s)
|
|
|
|
`, accountTable, columns, values(columns))
|
|
|
|
|
2024-03-20 16:36:45 +00:00
|
|
|
_, err = as.Database.Exec(insertQ, a.valuesNoID()...)
|
2024-02-24 21:00:04 +00:00
|
|
|
if err != nil {
|
2024-03-20 16:36:45 +00:00
|
|
|
return pkgErr("error executing insert query", err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) DestructiveReset() error {
|
2024-03-20 16:36:45 +00:00
|
|
|
err := as.dropAccountTable()
|
|
|
|
if err != nil {
|
|
|
|
return pkgErr(fmt.Sprintf("error dropping %s table", accountTable), err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) dropAccountTable() error {
|
2024-02-24 21:00:04 +00:00
|
|
|
dropQ := fmt.Sprintf(`
|
|
|
|
DROP TABLE IF EXISTS "%s"
|
|
|
|
`, accountTable)
|
|
|
|
|
|
|
|
_, err := as.Database.Exec(dropQ)
|
|
|
|
if err != nil {
|
2024-03-20 16:36:45 +00:00
|
|
|
return fmt.Errorf("error executing drop query: %v", err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (as *accountService) Update(a *Account) error {
|
|
|
|
err := runAccountValFuncs(
|
|
|
|
a,
|
|
|
|
accountRequireID,
|
|
|
|
accountRequireUsername,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return pkgErr("invalid account", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
columns := columnsNoID(accountColumns)
|
|
|
|
updateQ := fmt.Sprintf(`
|
|
|
|
UPDATE "%s"
|
|
|
|
SET %s
|
|
|
|
WHERE id=?
|
|
|
|
`, accountTable, set(columns))
|
|
|
|
|
2024-03-20 16:36:45 +00:00
|
|
|
_, err = as.Database.Exec(updateQ, a.valuesEndID()...)
|
2024-02-24 21:00:04 +00:00
|
|
|
if err != nil {
|
2024-03-20 16:36:45 +00:00
|
|
|
return pkgErr(fmt.Sprintf("error executing update query", accountTable), err)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type accountValFunc func(*Account) error
|
|
|
|
|
|
|
|
func runAccountValFuncs(a *Account, fns ...accountValFunc) error {
|
|
|
|
if a == nil {
|
|
|
|
return fmt.Errorf("account cannot be nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fn := range fns {
|
|
|
|
err := fn(a)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func accountRequireID(a *Account) error {
|
|
|
|
if a.ID == nil || *a.ID < 1 {
|
|
|
|
return ErrAccountInvalidID
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func accountRequireUsername(a *Account) error {
|
|
|
|
if a.Username == nil || *a.Username == "" {
|
|
|
|
return ErrAccountInvalidUsername
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|