party/internal/data/issues.go

283 lines
5.7 KiB
Go

package data
import (
"time"
"database/sql"
"errors"
"context"
"fmt"
"party.at/party/internal/validator"
)
type Issue struct {
ID int64 `json:"id"`
Title string `json:"title"`
Description string `json:"description"`
StartTime time.Time `json:"start_time"`
EndTime time.Time `json:"end_time"`
N []byte `json:"-"`
E int `json:"-"`
PrivatePem []byte `json:"-"`
Created time.Time `json:"created"`
Version int32 `json:"version"`
}
func ValidateIssue(v *validator.Validator, issue *Issue) {
v.Check(issue.Title != "", "title", "must be provided")
v.Check(len(issue.Title) <= 500, "title", "must not be more than 500 bytes long")
v.Check(issue.Description != "", "description", "must be provided")
v.Check(len(issue.Description) <= 500, "description", "must not be greater than 500 bytes long")
// v.Check(issue.StartTime != "", "start_time", "must be provided")
// v.Check(len(issue.StartTime) <= 500, "start_time", "must not be more than 500 bytes long")
// v.Check(issue.EndTime != "", "end_time", "must be provided")
// v.Check(len(issue.EndTime) <= 500, "end_time", "must not be more than 500 bytes long")
}
type IssueModel struct {
DB *sql.DB
}
func (m IssueModel) Insert(issue *Issue) error {
query :=
`INSERT INTO issues (title, description, start_time, end_time, rsa_n, rsa_e, rsa_private_pem)
VALUES ($1, $2, $3, $4, $5, $6, $7)
RETURNING id, created, version`
args := []interface{}{
issue.Title,
issue.Description,
issue.StartTime,
issue.EndTime,
issue.N,
issue.E,
issue.PrivatePem,
}
return m.DB.QueryRow(query, args...).Scan(
&issue.ID,
&issue.Created,
&issue.Version,
)
}
func (m IssueModel) InsertWithOptions(issue *Issue, options []*Option) error {
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
tx, err := m.DB.BeginTx(ctx, nil)
if err != nil {
return err
}
defer tx.Rollback()
query :=
`INSERT INTO issues (title, description, start_time, end_time, rsa_n, rsa_e, rsa_private_pem)
VALUES ($1, $2, $3, $4, $5, $6, $7)
RETURNING id, created, version`
args := []any{
issue.Title,
issue.Description,
issue.StartTime,
issue.EndTime,
issue.N,
issue.E,
issue.PrivatePem,
}
err = tx.QueryRowContext(ctx, query, args...).Scan(&issue.ID, &issue.Created, &issue.Version)
if err != nil {
return err
}
for _, option := range options {
option.IssueID = issue.ID
err = tx.QueryRowContext(ctx,
`INSERT INTO options (issue_id, label)
VALUES ($1, $2)
RETURNING id, created, version`,
option.IssueID,
option.Label,
).Scan(&option.ID, &option.Created, &option.Version)
if err != nil {
return err
}
}
err = tx.Commit()
if err != nil {
return err
}
return nil
}
func (m IssueModel) Get(id int64) (*Issue, error) {
if id < 1 {
return nil, ErrRecordNotFound
}
query :=
`SELECT id, title, description, start_time, end_time, rsa_n, rsa_e, rsa_private_pem, created, version
FROM issues
WHERE id = $1`
var issue Issue
err := m.DB.QueryRow(query, id).Scan(
&issue.ID,
&issue.Title,
&issue.Description,
&issue.StartTime,
&issue.EndTime,
&issue.N,
&issue.E,
&issue.PrivatePem,
&issue.Created,
&issue.Version,
)
if err != nil {
switch {
case errors.Is(err, sql.ErrNoRows):
return nil, ErrRecordNotFound
default:
return nil, err
}
}
return &issue, nil
}
func (m IssueModel) Update(issue *Issue) error {
query := `
UPDATE issues
SET
title = $1,
description = $2,
start_time = $3,
end_time = $4,
rsa_n = $5,
rsa_e = $6,
rsa_private_pem = $7,
version = version + 1
WHERE id = $8 AND version = $9
RETURNING version`
args := []interface{}{
issue.Title,
issue.Description,
issue.StartTime,
issue.EndTime,
issue.N,
issue.E,
issue.PrivatePem,
issue.ID,
issue.Version,
}
err := m.DB.QueryRow(query, args...).Scan(&issue.Version)
if err != nil {
switch {
case errors.Is(err, sql.ErrNoRows):
return ErrEditConflict
default:
return err
}
}
return nil
}
func (m IssueModel) Delete(id int64) error {
if id < 1 {
return ErrRecordNotFound
}
query := `
DELETE FROM issues
WHERE id = $1`
result, err := m.DB.Exec(query, id)
if err != nil {
return err
}
rowsAffected, err := result.RowsAffected()
if err != nil {
return err
}
if rowsAffected == 0 {
return ErrRecordNotFound
}
return nil
}
func (m IssueModel) GetAll(title string, filters Filters) ([]*Issue, Metadata, error) {
query := fmt.Sprintf(`
SELECT
COUNT(*) OVER(), id, title, description, start_time, end_time, rsa_n, rsa_e, rsa_private_pem, created, version
FROM
issues
WHERE
(to_tsvector('simple', title) @@ plainto_tsquery('simple', $1) OR $1 = '')
ORDER BY
%s %s, id ASC
LIMIT
$2
OFFSET
$3`,
filters.sortColumn(),
filters.sortDirection(),
)
ctx, cancel := context.WithTimeout(context.Background(), 3 * time.Second)
defer cancel()
args := []interface{}{title, filters.limit(), filters.offset()}
rows, err := m.DB.QueryContext(ctx, query, args...)
if err != nil {
return nil, Metadata{}, err
}
defer rows.Close()
totalRecords := 0
issues := []*Issue{}
for rows.Next() {
var issue Issue
err := rows.Scan(
&totalRecords,
&issue.ID,
&issue.Title,
&issue.Description,
&issue.StartTime,
&issue.EndTime,
&issue.N,
&issue.E,
&issue.PrivatePem,
&issue.Created,
&issue.Version,
)
if err != nil {
return nil, Metadata{}, err
}
issues = append(issues, &issue)
}
if err = rows.Err(); err != nil {
return nil, Metadata{}, err
}
metadata := calculateMetadata(totalRecords, filters.Page, filters.PageSize)
return issues, metadata, nil
}