forked from zmb3/spotify
-
Notifications
You must be signed in to change notification settings - Fork 0
/
track.go
177 lines (152 loc) · 5.86 KB
/
track.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
package spotify
import (
"context"
"errors"
"fmt"
"strings"
"time"
)
type TrackExternalIDs struct {
ISRC string `json:"isrc"`
EAN string `json:"ean"`
UPC string `json:"upc"`
}
// SimpleTrack contains basic info about a track.
type SimpleTrack struct {
Album SimpleAlbum `json:"album"`
Artists []SimpleArtist `json:"artists"`
// A list of the countries in which the track can be played,
// identified by their ISO 3166-1 alpha-2 codes.
AvailableMarkets []string `json:"available_markets"`
// The disc number (usually 1 unless the album consists of more than one disc).
DiscNumber int `json:"disc_number"`
// The length of the track, in milliseconds.
Duration int `json:"duration_ms"`
// Whether or not the track has explicit lyrics.
// true => yes, it does; false => no, it does not.
Explicit bool `json:"explicit"`
// External URLs for this track.
ExternalURLs map[string]string `json:"external_urls"`
// ExternalIDs are IDs for this track in other databases
ExternalIDs TrackExternalIDs `json:"external_ids"`
// A link to the Web API endpoint providing full details for this track.
Endpoint string `json:"href"`
ID ID `json:"id"`
Name string `json:"name"`
// A URL to a 30 second preview (MP3) of the track.
PreviewURL string `json:"preview_url"`
// The number of the track. If an album has several
// discs, the track number is the number on the specified
// DiscNumber.
TrackNumber int `json:"track_number"`
URI URI `json:"uri"`
// Type of the track
Type string `json:"type"`
}
func (st SimpleTrack) String() string {
return fmt.Sprintf("TRACK<[%s] [%s]>", st.ID, st.Name)
}
// LinkedFromInfo
// See: https://developer.spotify.com/documentation/general/guides/track-relinking-guide/
type LinkedFromInfo struct {
// ExternalURLs are the known external APIs for this track or album
ExternalURLs map[string]string `json:"external_urls"`
// Href is a link to the Web API endpoint providing full details
Href string `json:"href"`
// ID of the linked track
ID ID `json:"id"`
// Type of the link: album of the track
Type string `json:"type"`
// URI is the Spotify URI of the track/album
URI string `json:"uri"`
}
// FullTrack provides extra track data in addition to what is provided by SimpleTrack.
type FullTrack struct {
SimpleTrack
// The album on which the track appears. The album object includes a link in href to full information about the album.
Album SimpleAlbum `json:"album"`
// Known external IDs for the track.
ExternalIDs map[string]string `json:"external_ids"`
// Popularity of the track. The value will be between 0 and 100,
// with 100 being the most popular. The popularity is calculated from
// both total plays and most recent plays.
Popularity int `json:"popularity"`
// IsPlayable defines if the track is playable. It's reported when the "market" parameter is passed to the tracks
// listing API.
// See: https://developer.spotify.com/documentation/general/guides/track-relinking-guide/
IsPlayable *bool `json:"is_playable"`
// LinkedFrom points to the linked track. It's reported when the "market" parameter is passed to the tracks listing
// API.
LinkedFrom *LinkedFromInfo `json:"linked_from"`
}
// PlaylistTrack contains info about a track in a playlist.
type PlaylistTrack struct {
// The date and time the track was added to the playlist.
// You can use the TimestampLayout constant to convert
// this field to a time.Time value.
// Warning: very old playlists may not populate this value.
AddedAt string `json:"added_at"`
// The Spotify user who added the track to the playlist.
// Warning: vary old playlists may not populate this value.
AddedBy User `json:"added_by"`
// Whether this track is a local file or not.
IsLocal bool `json:"is_local"`
// Information about the track.
Track FullTrack `json:"track"`
}
// SavedTrack provides info about a track saved to a user's account.
type SavedTrack struct {
// The date and time the track was saved, represented as an ISO
// 8601 UTC timestamp with a zero offset (YYYY-MM-DDTHH:MM:SSZ).
// You can use the TimestampLayout constant to convert this to
// a time.Time value.
AddedAt string `json:"added_at"`
FullTrack `json:"track"`
}
// TimeDuration returns the track's duration as a time.Duration value.
func (t *SimpleTrack) TimeDuration() time.Duration {
return time.Duration(t.Duration) * time.Millisecond
}
// GetTrack gets Spotify catalog information for
// a single track identified by its unique Spotify ID.
//
// API Doc: https://developer.spotify.com/documentation/web-api/reference/tracks/get-track/
//
// Supported options: Market
func (c *Client) GetTrack(ctx context.Context, id ID, opts ...RequestOption) (*FullTrack, error) {
spotifyURL := c.baseURL + "tracks/" + string(id)
var t FullTrack
if params := processOptions(opts...).urlParams.Encode(); params != "" {
spotifyURL += "?" + params
}
err := c.get(ctx, spotifyURL, &t)
if err != nil {
return nil, err
}
return &t, nil
}
// GetTracks gets Spotify catalog information for multiple tracks based on their
// Spotify IDs. It supports up to 50 tracks in a single call. Tracks are
// returned in the order requested. If a track is not found, that position in the
// result will be nil. Duplicate ids in the query will result in duplicate
// tracks in the result.
//
// API Doc: https://developer.spotify.com/documentation/web-api/reference/tracks/get-several-tracks/
//
// Supported options: Market
func (c *Client) GetTracks(ctx context.Context, ids []ID, opts ...RequestOption) ([]*FullTrack, error) {
if len(ids) > 50 {
return nil, errors.New("spotify: FindTracks supports up to 50 tracks")
}
params := processOptions(opts...).urlParams
params.Set("ids", strings.Join(toStringSlice(ids), ","))
spotifyURL := c.baseURL + "tracks?" + params.Encode()
var t struct {
Tracks []*FullTrack `json:"tracks"`
}
err := c.get(ctx, spotifyURL, &t)
if err != nil {
return nil, err
}
return t.Tracks, nil
}