209 lines
5.3 KiB
Go
209 lines
5.3 KiB
Go
package data
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/lib/pq"
|
|
"greenlight.alexedwards.net/internal/validator"
|
|
)
|
|
|
|
type Movie struct {
|
|
ID int64 `json:"id"`
|
|
CreateAt time.Time `json:"-"`
|
|
Title string `json:"title"`
|
|
Year int32 `json:"year,omitempty"`
|
|
Runtime Runtime `json:"-"`
|
|
Genres []string `json:"genres,omitempty"`
|
|
Version int32 `json:"version"`
|
|
}
|
|
|
|
func (m Movie) MarshalJSON() ([]byte, error) {
|
|
var runtime string
|
|
if m.Runtime != 0 {
|
|
runtime = fmt.Sprintf("%d mins", m.Runtime)
|
|
}
|
|
type MovieAlis Movie
|
|
aux := struct {
|
|
MovieAlis
|
|
Runtime string `json:"runtime,omitempty"`
|
|
}{
|
|
MovieAlis: (MovieAlis)(m),
|
|
Runtime: runtime,
|
|
}
|
|
return json.Marshal(aux)
|
|
}
|
|
|
|
func ValidateMovie(v *validator.Validator, movie *Movie) {
|
|
v.Check(movie.Title != "", "title", "must be provided")
|
|
v.Check(len(movie.Title) <= 500, "title", "must not be more than 500 bytes long")
|
|
v.Check(movie.Year != 0, "year", "must be provided")
|
|
v.Check(movie.Year >= 1888, "year", "must be greater than 1888")
|
|
v.Check(movie.Year <= int32(time.Now().Year()), "year", "must not bve in the future")
|
|
v.Check(movie.Runtime != 0, "runtime", "must be provided")
|
|
v.Check(movie.Runtime > 0, "runtime", "must be a positive integer")
|
|
v.Check(movie.Genres != nil, "genres", "must be provided")
|
|
v.Check(len(movie.Genres) >= 1, "genres", "must contain at least 1 genre")
|
|
v.Check(len(movie.Genres) <= 5, "genres", "must not contain more than 5 genres")
|
|
}
|
|
|
|
type MovieModel struct {
|
|
DB *sql.DB
|
|
}
|
|
|
|
func (m MovieModel) Insert(movie *Movie) error {
|
|
query := `
|
|
INSERT INTO movies (title, year, runtime, genres)
|
|
VALUES ($1, $2, $3, $4)
|
|
RETURNING id, created_at, version
|
|
`
|
|
args := []any{movie.Title, movie.Year, movie.Runtime, pq.Array(movie.Genres)}
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
defer cancel()
|
|
return m.DB.QueryRowContext(ctx, query, args...).Scan(&movie.ID, &movie.CreateAt, &movie.Version)
|
|
}
|
|
func (m MovieModel) Get(id int64) (*Movie, error) {
|
|
if id < 1 {
|
|
return nil, ErrRecordNotFound
|
|
}
|
|
query := `
|
|
SELECT id, created_at, title, year, runtime, genres, version
|
|
FROM movies
|
|
WHERE id = $1`
|
|
var movie Movie
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
defer cancel()
|
|
err := m.DB.QueryRowContext(ctx, query, id).Scan(
|
|
&movie.ID,
|
|
&movie.CreateAt,
|
|
&movie.Title,
|
|
&movie.Year,
|
|
&movie.Runtime,
|
|
pq.Array(&movie.Genres),
|
|
&movie.Version,
|
|
)
|
|
if err != nil {
|
|
switch {
|
|
case errors.Is(err, sql.ErrNoRows):
|
|
return nil, ErrRecordNotFound
|
|
default:
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return &movie, nil
|
|
}
|
|
func (m MovieModel) Update(movie *Movie) error {
|
|
query := `
|
|
UPDATE movies
|
|
SET title = $1, year = $2, runtime = $3, genres = $4, version = version + 1
|
|
WHERE id = $5 AND version = $6
|
|
RETURNING version`
|
|
args := []any{
|
|
movie.Title,
|
|
movie.Year,
|
|
movie.Runtime,
|
|
pq.Array(movie.Genres),
|
|
movie.ID,
|
|
movie.Version,
|
|
}
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
defer cancel()
|
|
err := m.DB.QueryRowContext(ctx, query, args...).Scan(&movie.Version)
|
|
if err != nil {
|
|
switch {
|
|
case errors.Is(err, sql.ErrNoRows):
|
|
return ErrEditConflict
|
|
default:
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
func (m MovieModel) Delete(id int64) error {
|
|
if id < 1 {
|
|
return ErrRecordNotFound
|
|
}
|
|
query := `
|
|
DELETE FROM movies
|
|
WHERE id = $1`
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
defer cancel()
|
|
result, err := m.DB.ExecContext(ctx, 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 MovieModel) GetAll(title string, genres []string, filters Filters) ([]*Movie, Metadata, error) {
|
|
query := fmt.Sprintf(`
|
|
SELECT count(*) OVER(), id, created_at, title, year, runtime, genres, version
|
|
FROM movies
|
|
WHERE (to_tsvector('simple', title) @@ plainto_tsquery($1) or $1 = '')
|
|
AND (genres @> $2 OR $2 = '{}')
|
|
ORDER BY %s %s, id ASC
|
|
LIMIT $3 OFFSET $4`, filters.sortColumn(), filters.sortDirection())
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
defer cancel()
|
|
args := []any{title, pq.Array(genres), filters.limit(), filters.offset()}
|
|
rows, err := m.DB.QueryContext(ctx, query, args...)
|
|
if err != nil {
|
|
return nil, Metadata{}, err
|
|
}
|
|
defer rows.Close()
|
|
totalRecords := 0
|
|
movies := []*Movie{}
|
|
for rows.Next() {
|
|
var movie Movie
|
|
err := rows.Scan(
|
|
&totalRecords,
|
|
&movie.ID,
|
|
&movie.CreateAt,
|
|
&movie.Title,
|
|
&movie.Year,
|
|
&movie.Runtime,
|
|
pq.Array(&movie.Genres),
|
|
&movie.Version,
|
|
)
|
|
if err != nil {
|
|
return nil, Metadata{}, err
|
|
}
|
|
movies = append(movies, &movie)
|
|
}
|
|
if err = rows.Err(); err != nil {
|
|
return nil, Metadata{}, err
|
|
}
|
|
metadata := calculateMetadata(totalRecords, filters.Page, filters.PageSize)
|
|
return movies, metadata, nil
|
|
}
|
|
|
|
type MockMovieModel struct{}
|
|
|
|
func (m MockMovieModel) Insert(movie *Movie) error {
|
|
return nil
|
|
}
|
|
func (m MockMovieModel) Get(id int64) (*Movie, error) {
|
|
return nil, nil
|
|
}
|
|
func (m MockMovieModel) GetAll(title string, genres []string, filters Filters) ([]*Movie, Metadata, error) {
|
|
return nil, Metadata{}, nil
|
|
}
|
|
func (m MockMovieModel) Update(movie *Movie) error {
|
|
return nil
|
|
}
|
|
func (m MockMovieModel) Delete(id int64) error {
|
|
return nil
|
|
}
|