Run go fmt, fix pagination bug, clean up code
This commit is contained in:
parent
0c123c415b
commit
dc3820d6f4
4 changed files with 37 additions and 30 deletions
32
database.go
32
database.go
|
@ -1,9 +1,9 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"gorm.io/driver/sqlite"
|
"gorm.io/driver/sqlite"
|
||||||
"encoding/json"
|
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
"log/slog"
|
"log/slog"
|
||||||
"time"
|
"time"
|
||||||
|
@ -48,37 +48,37 @@ func setupDatabase() *gorm.DB {
|
||||||
|
|
||||||
// Some functions to control JSON marshalling of DB structs
|
// Some functions to control JSON marshalling of DB structs
|
||||||
func (p *ArtistProfile) MarshalJSON() ([]byte, error) {
|
func (p *ArtistProfile) MarshalJSON() ([]byte, error) {
|
||||||
type CleanedProfile struct {
|
type CleanedProfile struct {
|
||||||
UpdatedAt time.Time
|
UpdatedAt time.Time
|
||||||
SpotifyID string
|
SpotifyID string
|
||||||
Name string
|
Name string
|
||||||
Popularity int
|
Popularity int
|
||||||
Genres []string
|
Genres []string
|
||||||
}
|
}
|
||||||
|
|
||||||
artistGenres := make([]string, 0, len(p.Genres))
|
artistGenres := make([]string, 0, len(p.Genres))
|
||||||
for _, val := range p.Genres {
|
for _, val := range p.Genres {
|
||||||
artistGenres = append(artistGenres, val.Name)
|
artistGenres = append(artistGenres, val.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
t := CleanedProfile{
|
t := CleanedProfile{
|
||||||
UpdatedAt: p.UpdatedAt,
|
UpdatedAt: p.UpdatedAt,
|
||||||
SpotifyID: p.SpotifyID,
|
SpotifyID: p.SpotifyID,
|
||||||
Name: p.Name,
|
Name: p.Name,
|
||||||
Popularity: p.Popularity,
|
Popularity: p.Popularity,
|
||||||
Genres: artistGenres,
|
Genres: artistGenres,
|
||||||
}
|
}
|
||||||
return json.Marshal(&t)
|
return json.Marshal(&t)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *Genre) MarshalJSON() ([]byte, error) {
|
func (g *Genre) MarshalJSON() ([]byte, error) {
|
||||||
type CleanedGenre struct {
|
type CleanedGenre struct {
|
||||||
Name string
|
Name string
|
||||||
CreatedAt time.Time
|
CreatedAt time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
t := CleanedGenre{
|
t := CleanedGenre{
|
||||||
Name: g.Name,
|
Name: g.Name,
|
||||||
CreatedAt: g.CreatedAt,
|
CreatedAt: g.CreatedAt,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
2
main.go
2
main.go
|
@ -27,6 +27,8 @@ func setupRouter(env *Env, spotifyID string, spotifySecret string) *gin.Engine {
|
||||||
r.GET("/artists/:artistID", env.getArtistByID)
|
r.GET("/artists/:artistID", env.getArtistByID)
|
||||||
r.GET("/artists", env.getArtistByName)
|
r.GET("/artists", env.getArtistByName)
|
||||||
r.GET("/genres", env.getGenres)
|
r.GET("/genres", env.getGenres)
|
||||||
|
|
||||||
|
// POST create endpoint
|
||||||
r.POST("/genres", env.createGenre)
|
r.POST("/genres", env.createGenre)
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
19
routes.go
19
routes.go
|
@ -17,7 +17,7 @@ func paginator() gin.HandlerFunc {
|
||||||
queryPageSize := c.DefaultQuery("page_size", "10")
|
queryPageSize := c.DefaultQuery("page_size", "10")
|
||||||
|
|
||||||
// add some defaults here if user gives us empty values
|
// add some defaults here if user gives us empty values
|
||||||
if queryPage == ""{
|
if queryPage == "" {
|
||||||
queryPage = "1"
|
queryPage = "1"
|
||||||
}
|
}
|
||||||
if queryPageSize == "" {
|
if queryPageSize == "" {
|
||||||
|
@ -38,8 +38,12 @@ func paginator() gin.HandlerFunc {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if page < 1 || pageSize < 1 {
|
if page < 1 {
|
||||||
page, pageSize = 1, 10
|
page = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
if pageSize < 10 {
|
||||||
|
pageSize = 10
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate the correct SQL offset for this page
|
// Calculate the correct SQL offset for this page
|
||||||
|
@ -123,7 +127,8 @@ func (env *Env) getArtistByName(c *gin.Context) {
|
||||||
// NOTE: However unicode is treated case sensitive due to very difficult conversions for some languages/characters
|
// 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
|
// NOTE: In future better DBs with unicode support such as postgres should be used
|
||||||
var artistProfiles []ArtistProfile
|
var artistProfiles []ArtistProfile
|
||||||
dbResult := env.db.Where("name LIKE ?", fmt.Sprintf("%%%s%%", artistName)).Preload("Genres").Offset(pageOffset).Limit(pageSize).Find(&artistProfiles)
|
searchString := fmt.Sprintf("%%%s%%"), artistName)
|
||||||
|
dbResult := env.db.Where("name LIKE ?", searchString).Preload("Genres").Offset(pageOffset).Limit(pageSize).Find(&artistProfiles)
|
||||||
if dbResult.Error != nil {
|
if dbResult.Error != nil {
|
||||||
slog.Error("[GOMUSIC] Failed to query local database for artist name", "Name", artistName)
|
slog.Error("[GOMUSIC] Failed to query local database for artist name", "Name", artistName)
|
||||||
c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to lookup name"})
|
c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to lookup name"})
|
||||||
|
@ -156,12 +161,12 @@ func (env *Env) getGenres(c *gin.Context) {
|
||||||
c.JSON(http.StatusOK, genreList)
|
c.JSON(http.StatusOK, genreList)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (env *Env) createGenre(c * gin.Context) {
|
func (env *Env) createGenre(c *gin.Context) {
|
||||||
var userGenre Genre
|
var userGenre Genre
|
||||||
if c.ShouldBind(&userGenre) == nil {
|
if c.ShouldBind(&userGenre) == nil {
|
||||||
// If we didn't get the correct data abort early
|
// If we didn't get the correct data abort early
|
||||||
if userGenre.Name == "" {
|
if userGenre.Name == "" {
|
||||||
c.JSON(http.StatusBadRequest, gin.H{"Error":"Could not find genre name"})
|
c.JSON(http.StatusBadRequest, gin.H{"Error": "Could not find genre name"})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
dbResult := env.db.Clauses(clause.OnConflict{
|
dbResult := env.db.Clauses(clause.OnConflict{
|
||||||
|
@ -175,5 +180,5 @@ func (env *Env) createGenre(c * gin.Context) {
|
||||||
c.String(http.StatusOK, "Success")
|
c.String(http.StatusOK, "Success")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.JSON(http.StatusInternalServerError, gin.H{"Error":"Failed to parse input data"})
|
c.JSON(http.StatusInternalServerError, gin.H{"Error": "Failed to parse input data"})
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,16 +6,16 @@ import (
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"testing"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Define some custom types to make response parsing easier
|
// Define some custom types to make response parsing easier
|
||||||
type SearchResponse struct {
|
type SearchResponse struct {
|
||||||
SpotifyID string
|
SpotifyID string
|
||||||
Name string
|
Name string
|
||||||
Popularity int
|
Popularity int
|
||||||
Genres []string
|
Genres []string
|
||||||
}
|
}
|
||||||
type SearchResponseList []SearchResponse
|
type SearchResponseList []SearchResponse
|
||||||
|
|
||||||
|
@ -213,10 +213,10 @@ func TestGetPaginatedArtistByName(t *testing.T) {
|
||||||
for i := 0; i < 100; i++ {
|
for i := 0; i < 100; i++ {
|
||||||
testName := fmt.Sprintf("test%d", i)
|
testName := fmt.Sprintf("test%d", i)
|
||||||
testArtist := ArtistProfile{
|
testArtist := ArtistProfile{
|
||||||
Name: testName,
|
Name: testName,
|
||||||
SpotifyID: testName,
|
SpotifyID: testName,
|
||||||
Popularity: i,
|
Popularity: i,
|
||||||
Genres: []Genre{ Genre{Name: testName} },
|
Genres: []Genre{Genre{Name: testName}},
|
||||||
}
|
}
|
||||||
testData = append(testData, testArtist)
|
testData = append(testData, testArtist)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue