Return correct status and data when client sends bad requests

This commit is contained in:
froge 2025-02-12 14:08:46 +10:00
parent eb57f3fb83
commit 10cc28e5f4
Signed by: froge
GPG key ID: A825E09930271BFA
3 changed files with 59 additions and 4 deletions

View file

@ -22,6 +22,7 @@ func setupRouter(env *Env, spotifyID string, spotifySecret string) *gin.Engine {
r.Use(spotifyAuth(spotifyID, spotifySecret)) r.Use(spotifyAuth(spotifyID, spotifySecret))
r.GET("/alive", env.alive) r.GET("/alive", env.alive)
r.GET("/artists/:artistID", env.getArtistByID) r.GET("/artists/:artistID", env.getArtistByID)
r.GET("/artists", env.getArtistByName)
return r return r
} }

View file

@ -5,6 +5,7 @@ import (
"gorm.io/gorm/clause" "gorm.io/gorm/clause"
"net/http" "net/http"
"log/slog" "log/slog"
"fmt"
) )
func (env *Env) alive(c *gin.Context) { func (env *Env) alive(c *gin.Context) {
@ -22,6 +23,15 @@ func (env *Env) getArtistByID(c *gin.Context) {
spotifyResponse, err := getSpotifyArtistData(artistID, spotifyAuthToken) spotifyResponse, err := getSpotifyArtistData(artistID, spotifyAuthToken)
if err != nil { if err != nil {
statusCode := err.(*ResponseError).StatusCode
switch statusCode {
case 404:
c.JSON(http.StatusNotFound, gin.H{"Error": "This artist does not exist"})
return
case 400:
c.JSON(http.StatusBadRequest, gin.H{"Error": "Bad request"})
return
}
slog.Error("[GOMUSIC] Failed to request latest spotify data from API", "Error", err) slog.Error("[GOMUSIC] Failed to request latest spotify data from API", "Error", err)
c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to request the latest data from spotify API"}) c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to request the latest data from spotify API"})
return return
@ -55,3 +65,21 @@ func (env *Env) getArtistByID(c *gin.Context) {
// Send back our response data // Send back our response data
c.JSON(http.StatusOK, spotifyResponse) c.JSON(http.StatusOK, spotifyResponse)
} }
func (env *Env) getArtistByName(c *gin.Context) {
artistName, exists := c.GetQuery("name")
if !exists {
c.JSON(http.StatusBadRequest, gin.H{"Error": "name parameter was not supplied"})
return
}
// Lookup this name in the DB and return any ArtistProfile objects
var artistProfiles []ArtistProfile
dbResult := env.db.Where("name LIKE ?", fmt.Sprintf("%%%s%%", artistName)).Find(&artistProfiles)
if dbResult.Error != nil {
slog.Error("[GOMUSIC] Failed to query local database for artist name", "Name", artistName)
c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to lookup name"})
return
}
c.JSON(http.StatusOK, artistProfiles)
}

View file

@ -2,6 +2,7 @@ package main
import ( import (
"fmt" "fmt"
"errors"
"net/http" "net/http"
"encoding/json" "encoding/json"
) )
@ -13,6 +14,15 @@ type SpotifyResponse struct {
Genres []string Genres []string
} }
type ResponseError struct {
StatusCode int
Err error
}
func (r ResponseError) Error() string {
return fmt.Sprintf("Response error with status code: %d, error: %v", r.StatusCode, r.Err)
}
// Makes a request to spotify API to grab artist data // Makes a request to spotify API to grab artist data
// And parse the results into a properly typed struct // And parse the results into a properly typed struct
func getSpotifyArtistData(artistID string, spotifyAuthToken string) (SpotifyResponse, error) { func getSpotifyArtistData(artistID string, spotifyAuthToken string) (SpotifyResponse, error) {
@ -20,22 +30,38 @@ func getSpotifyArtistData(artistID string, spotifyAuthToken string) (SpotifyResp
artistEndpoint := fmt.Sprintf("https://api.spotify.com/v1/artists/%s", artistID) artistEndpoint := fmt.Sprintf("https://api.spotify.com/v1/artists/%s", artistID)
req, err := http.NewRequest("GET", artistEndpoint, nil) req, err := http.NewRequest("GET", artistEndpoint, nil)
if err != nil { if err != nil {
return SpotifyResponse{}, fmt.Errorf("Failed to build HTTP request to spotify: %w", err) respErr := &ResponseError{
StatusCode: 500,
Err: errors.New("Failed to build request"),
}
return SpotifyResponse{}, respErr
} }
req.Header.Add("Authorization", spotifyAuthToken) req.Header.Add("Authorization", spotifyAuthToken)
// Send off request // Send off request
resp, err := http.DefaultClient.Do(req) resp, err := http.DefaultClient.Do(req)
if err != nil { if err != nil {
return SpotifyResponse{}, fmt.Errorf("Failed to get artist data from spotify API: %w", err) respErr := &ResponseError{
StatusCode: 500,
Err: errors.New("Failed to send request to spotify"),
}
return SpotifyResponse{}, respErr
} else if resp.StatusCode != 200 { } else if resp.StatusCode != 200 {
return SpotifyResponse{}, fmt.Errorf("Failed to get artist data from spotify API due to response status: %s", resp.Status) respErr := &ResponseError{
StatusCode: resp.StatusCode,
Err: errors.New("Failed to get artist data from spotify API"),
}
return SpotifyResponse{}, respErr
} }
var spotifyResponse SpotifyResponse var spotifyResponse SpotifyResponse
err = json.NewDecoder(resp.Body).Decode(&spotifyResponse) err = json.NewDecoder(resp.Body).Decode(&spotifyResponse)
if err != nil { if err != nil {
return SpotifyResponse{}, fmt.Errorf("Failed to read response body data from spotify: %w", err) respErr := &ResponseError{
StatusCode: 500,
Err: errors.New("Failed to decode response body"),
}
return SpotifyResponse{}, respErr
} }
// Close this immediately since it's unused now // Close this immediately since it's unused now