package data import ( "database/sql" "errors" "time" "github.com/lib/pq" "greenlight.debuggingjon.dev/internal/validator" ) // Define a MovieModel struct type which wraps a sql.DB connection pool. type MovieModel struct { DB *sql.DB } // Add a placeholder method for inserting a new record in the movies table. func (m MovieModel) Insert(movie *Movie) error { // Define the SQL query for inserting a new record in the movies table and returning // the system-generated data. query := ` INSERT INTO movies (title, year, runtime, genres) VALUES ($1, $2, $3, $4) RETURNING id, created_at, version` // Create an args slice containing the values for the placeholder parameters from // the movie struct. Declaring this slice immediately next to our SQL query helps to // make it nice and clear *what values are being used where* in the query. args := []interface{}{movie.Title, movie.Year, movie.Runtime, pq.Array(movie.Genres)} // Use the QueryRow() method to execute the SQL query on our connection pool, // passing in the args slice as a variadic parameter and scanning the system- // generated id, created_at and version values into the movie struct. return m.DB.QueryRow(query, args...).Scan(&movie.ID, &movie.CreatedAt, &movie.Version) } func (m MovieModel) Get(id int64) (*Movie, error) { // The PostgreSQL bigserial type that we're using for the movie ID starts // auto-incrementing at 1 by default, so we know that no movies will have ID values // less than that. To avoid making an unnecessary database call, we take a shortcut // and return an ErrRecordNotFound error straight away. if id < 1 { return nil, ErrRecordNotFound } // Define the SQL query for retrieving the movie data. query := ` SELECT id, created_at, title, year, runtime, genres, version FROM movies WHERE id = $1` // Declare a Movie struct to hold the data returned by the query. var movie Movie // Execute the query using the QueryRow() method, passing in the provided id value // as a placeholder parameter, and scan the response data into the fields of the // Movie struct. Importantly, notice that we need to convert the scan target for the // genres column using the pq.Array() adapter function again. err := m.DB.QueryRow(query, id).Scan( &movie.ID, &movie.CreatedAt, &movie.Title, &movie.Year, &movie.Runtime, pq.Array(&movie.Genres), &movie.Version, ) // Handle any errors. If there was no matching movie found, Scan() will return // a sql.ErrNoRows error. We check for this and return our custom ErrRecordNotFound // error instead. if err != nil { switch { case errors.Is(err, sql.ErrNoRows): return nil, ErrRecordNotFound default: return nil, err } } // Otherwise, return a pointer to the Movie struct. return &movie, nil } func (m MovieModel) Update(movie *Movie) error { // Declare the SQL query for updating the record and returning the new version // number. query := ` UPDATE movies SET title = $1, year = $2, runtime = $3, genres = $4, version = version + 1 WHERE id = $5 RETURNING version` // Create an args slice containing the values for the placeholder parameters. args := []any{ movie.Title, movie.Year, movie.Runtime, pq.Array(movie.Genres), movie.ID, } // Use the QueryRow() method to execute the query, passing in the args slice as a // variadic parameter and scanning the new version value into the movie struct. return m.DB.QueryRow(query, args...).Scan(&movie.Version) } func (m MovieModel) Delete(id int64) error { // Return an ErrRecordNotFound error if the movie ID is less than 1. if id < 1 { return ErrRecordNotFound } // Construct the SQL query to delete the record. query := ` DELETE FROM movies WHERE id = $1` // Execute the SQL query using the Exec() method, passing in the id variable as // the value for the placeholder parameter. The Exec() method returns a sql.Result // object. result, err := m.DB.Exec(query, id) if err != nil { return err } // Call the RowsAffected() method on the sql.Result object to get the number of rows // affected by the query. rowsAffected, err := result.RowsAffected() if err != nil { return err } // If no rows were affected, we know that the movies table didn't contain a record // with the provided ID at the moment we tried to delete it. In that case we // return an ErrRecordNotFound error. if rowsAffected == 0 { return ErrRecordNotFound } return nil } type Movie struct { ID int64 `json:"id"` CreatedAt time.Time `json:"-"` // Use the - directive Title string `json:"title"` Year int32 `json:"year,omitempty"` // Add the omitempty directive Runtime Runtime `json:"runtime,omitempty"` // Add the omitempty directive Genres []string `json:"genres,omitempty"` // Add the omitempty directive Version int32 `json:"version"` } 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 be 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") v.Check(validator.Unique(movie.Genres), "genres", "must not contain duplicate values") }