package main import ( "fmt" "github.com/gin-gonic/gin" "gorm.io/gorm/clause" "log/slog" "net/http" ) func (env *Env) alive(c *gin.Context) { c.String(http.StatusOK, "yes!") } func (env *Env) getArtistByID(c *gin.Context) { artistID := c.Params.ByName("artistID") spotifyAuthToken := c.GetString("spotifyAuthToken") if artistID == "" || spotifyAuthToken == "Bearer " { c.JSON(http.StatusBadRequest, gin.H{"Error": "Could not find required parameters and/or required authentication tokens"}) return } spotifyResponse, err := getSpotifyArtistData(artistID, spotifyAuthToken) if err != nil { // cast the error back into our custom type // so we can get context on why things failed 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) c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to request the latest data from spotify API"}) return } // Update DB here var genreList []Genre for _, val := range spotifyResponse.Genres { genreList = append(genreList, Genre{Name: val}) } artistProfile := &ArtistProfile{ SpotifyID: spotifyResponse.ID, Name: spotifyResponse.Name, Popularity: spotifyResponse.Popularity, Genres: genreList, } dbResult := env.db.Clauses(clause.OnConflict{ Columns: []clause.Column{{Name: "spotify_id"}}, UpdateAll: true, }).Create(&artistProfile) if dbResult.Error != nil { slog.Error("[GOMUSIC] Failed to store response in local database", "Error", err) } c.JSON(http.StatusOK, artistProfile) } 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 // NOTE: This is case insensitive for ascii on sqlite // NOTE: However unicode is treated case sensitive due to very difficult conversions for some languages/characters // NOTE: In future better DBs with unicode support such as postgres should be used var artistProfiles []ArtistProfile dbResult := env.db.Where("name LIKE ?", fmt.Sprintf("%%%s%%", artistName)).Preload("Genres").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) } func (env *Env) getGenres(c *gin.Context) { var genres []Genre dbResult := env.db.Find(&genres) if dbResult.Error != nil { slog.Error("[GOMUSIC] Failed to query local database for genre data") c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to query database for genre data"}) return } recordNum := dbResult.RowsAffected // Construct simple slice for response genreList := make([]string, 0, recordNum) if recordNum > 0 { for _, g := range genres { genreList = append(genreList, g.Name) } } c.JSON(http.StatusOK, genreList) } func (env *Env) createGenre(c * gin.Context) { var userGenre Genre if c.ShouldBind(&userGenre) == nil { // If we didn't get the correct data abort early if userGenre.Name == "" { c.JSON(http.StatusBadRequest, gin.H{"Error":"Could not find genre name"}) return } dbResult := env.db.Clauses(clause.OnConflict{ DoNothing: true, }).Create(&userGenre) if dbResult.Error != nil { slog.Error("[GOMUSIC] Failed to create new genre record in DB") c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to create new genre record"}) return } c.String(http.StatusOK, "Success") return } c.JSON(http.StatusInternalServerError, gin.H{"Error":"Failed to parse input data"}) }