187 lines
5.2 KiB
Go
187 lines
5.2 KiB
Go
package main
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
|
|
"greenlight.debuggingjon.dev/internal/data"
|
|
"greenlight.debuggingjon.dev/internal/validator"
|
|
)
|
|
|
|
func (app *application) createMovieHandler(w http.ResponseWriter, r *http.Request) {
|
|
var input struct {
|
|
Title string `json:"title"`
|
|
Year int32 `json:"year"`
|
|
Runtime data.Runtime `json:"runtime"`
|
|
Genres []string `json:"genres"`
|
|
}
|
|
|
|
err := app.readJSON(w, r, &input)
|
|
if err != nil {
|
|
app.badRequestResponse(w, r, err)
|
|
return
|
|
}
|
|
|
|
// Copy the values from the input struct to a new Movie struct.
|
|
movie := &data.Movie{
|
|
Title: input.Title,
|
|
Year: input.Year,
|
|
Runtime: input.Runtime,
|
|
Genres: input.Genres,
|
|
}
|
|
|
|
// Initialize a new Validator.
|
|
v := validator.New()
|
|
|
|
// Call the ValidateMovie() function and return a response containing the errors if
|
|
// any of the checks fail.
|
|
if data.ValidateMovie(v, movie); !v.Valid() {
|
|
app.failedValidationResponse(w, r, v.Errors)
|
|
return
|
|
}
|
|
|
|
// Call the Insert() method on our movies model, passing in a pointer to the
|
|
// validated movie struct. This will create a record in the database and update the
|
|
// movie struct with the system-generated information.”
|
|
err = app.models.Movies.Insert(movie)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
return
|
|
}
|
|
|
|
// When sending a HTTP response, we want to include a Location header to let the
|
|
// client know which URL they can find the newly-created resource at. We make an
|
|
// empty http.Header map and then use the Set() method to add a new Location header,
|
|
// interpolating the system-generated ID for our new movie in the URL.
|
|
headers := make(http.Header)
|
|
headers.Set("Location", fmt.Sprintf("/v1/movies/%d", movie.ID))
|
|
|
|
// Write a JSON response with a 201 Created status code, the movie data in the
|
|
// response body, and the Location header.
|
|
err = app.writeJSON(w, http.StatusCreated, envelope{"movie": movie}, headers)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
}
|
|
|
|
func (app *application) showMovieHandler(w http.ResponseWriter, r *http.Request) {
|
|
id, err := app.readIDParam(r)
|
|
if err != nil {
|
|
app.notFoundResponse(w, r)
|
|
return
|
|
}
|
|
|
|
// Call the Get() method to fetch the data for a specific movie. We also need to
|
|
// use the errors.Is() function to check if it returns a data.ErrRecordNotFound
|
|
// error, in which case we send a 404 Not Found response to the client.
|
|
movie, err := app.models.Movies.Get(id)
|
|
if err != nil {
|
|
switch {
|
|
case errors.Is(err, data.ErrRecordNotFound):
|
|
app.notFoundResponse(w, r)
|
|
default:
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
return
|
|
}
|
|
|
|
err = app.writeJSON(w, http.StatusOK, envelope{"movie": movie}, nil)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
}
|
|
|
|
func (app *application) updateMovieHandler(w http.ResponseWriter, r *http.Request) {
|
|
// Extract the movie ID from the URL.
|
|
id, err := app.readIDParam(r)
|
|
if err != nil {
|
|
app.notFoundResponse(w, r)
|
|
return
|
|
}
|
|
|
|
// Fetch the existing movie record from the database, sending a 404 Not Found
|
|
// response to the client if we couldn't find a matching record.
|
|
movie, err := app.models.Movies.Get(id)
|
|
if err != nil {
|
|
switch {
|
|
case errors.Is(err, data.ErrRecordNotFound):
|
|
app.notFoundResponse(w, r)
|
|
default:
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
return
|
|
}
|
|
// Declare an input struct to hold the expected data from the client.
|
|
var input struct {
|
|
Title string `json:"title"`
|
|
Year int32 `json:"year"`
|
|
Runtime data.Runtime `json:"runtime"`
|
|
Genres []string `json:"genres"`
|
|
}
|
|
|
|
// Read the JSON request body data into the input struct.
|
|
err = app.readJSON(w, r, &input)
|
|
if err != nil {
|
|
app.badRequestResponse(w, r, err)
|
|
return
|
|
}
|
|
|
|
// Copy the values from the request body to the appropriate fields of the movie
|
|
// record.
|
|
movie.Title = input.Title
|
|
movie.Year = input.Year
|
|
movie.Runtime = input.Runtime
|
|
movie.Genres = input.Genres
|
|
|
|
// Validate the updated movie record, sending the client a 422 Unprocessable Entity
|
|
// response if any checks fail.
|
|
v := validator.New()
|
|
|
|
if data.ValidateMovie(v, movie); !v.Valid() {
|
|
app.failedValidationResponse(w, r, v.Errors)
|
|
return
|
|
}
|
|
|
|
// Pass the updated movie record to our new Update() method.
|
|
err = app.models.Movies.Update(movie)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
return
|
|
}
|
|
|
|
// Write the updated movie record in a JSON response.
|
|
err = app.writeJSON(w, http.StatusOK, envelope{"movie": movie}, nil)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
}
|
|
|
|
func (app *application) deleteMovieHandler(w http.ResponseWriter, r *http.Request) {
|
|
// Extract the movie ID from the URL.
|
|
id, err := app.readIDParam(r)
|
|
if err != nil {
|
|
app.notFoundResponse(w, r)
|
|
return
|
|
}
|
|
|
|
// Delete the movie from the database, sending a 404 Not Found response to the
|
|
// client if there isn't a matching record.
|
|
err = app.models.Movies.Delete(id)
|
|
if err != nil {
|
|
switch {
|
|
case errors.Is(err, data.ErrRecordNotFound):
|
|
app.notFoundResponse(w, r)
|
|
default:
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
return
|
|
}
|
|
|
|
// Return a 200 OK status code along with a success message.
|
|
err = app.writeJSON(w, http.StatusOK, envelope{"message": "movie successfully deleted"}, nil)
|
|
if err != nil {
|
|
app.serverErrorResponse(w, r, err)
|
|
}
|
|
}
|