2016-06-12 22:39:00 +00:00
|
|
|
package footballdata
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
2016-06-13 17:46:27 +00:00
|
|
|
"errors"
|
2016-06-12 22:39:00 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
Provides a high-level client to talk to the API that football-data.org offers.
|
|
|
|
|
|
|
|
To create an instance please use NewClient(h).
|
|
|
|
*/
|
2016-06-25 10:15:58 +00:00
|
|
|
type Client interface {
|
|
|
|
Fixture(id uint64) FixtureRequest
|
|
|
|
Fixtures() FixturesRequest
|
|
|
|
SoccerSeason(id uint64) SoccerSeasonRequest
|
|
|
|
FixturesOfSoccerSeason(soccerSeasonId uint64) SoccerSeasonFixturesRequest
|
|
|
|
LeagueTableOfSoccerSeason(soccerSeasonId uint64) SoccerSeasonLeagueTableRequest
|
|
|
|
TeamsOfSoccerSeason(soccerSeasonId uint64) SoccerSeasonTeamsRequest
|
|
|
|
SoccerSeasons() SoccerSeasonsRequest
|
|
|
|
Team(id uint64) TeamRequest
|
|
|
|
FixturesOfTeam(id uint64) TeamFixturesRequest
|
|
|
|
|
|
|
|
SetToken(authToken string)
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Provides a high-level client implementation to talk to the API that football-data.org offers.
|
|
|
|
|
|
|
|
To create an instance please use NewClient(h).
|
|
|
|
*/
|
|
|
|
type client struct {
|
|
|
|
httpClient *http.Client
|
2016-06-12 22:39:00 +00:00
|
|
|
|
|
|
|
// Insert an API token here if you have one. It will be sent across with all requests.
|
2016-06-25 21:20:59 +00:00
|
|
|
AuthToken string
|
2016-06-12 22:39:00 +00:00
|
|
|
}
|
|
|
|
|
2016-06-25 21:22:52 +00:00
|
|
|
// NewClient creates a new Client instance that wraps around the given HTTP client.
|
2016-06-25 10:15:58 +00:00
|
|
|
//
|
|
|
|
// Call SetToken to add your token.
|
|
|
|
func NewClient(h *http.Client) Client {
|
|
|
|
return &client{httpClient: h}
|
2016-06-12 22:39:00 +00:00
|
|
|
}
|
|
|
|
|
2016-06-25 21:22:52 +00:00
|
|
|
// SetToken sets the authentication token.
|
|
|
|
// Calling this method is *optional*.
|
2016-06-25 21:20:59 +00:00
|
|
|
func (c *client) SetToken(authToken string) {
|
|
|
|
c.AuthToken = authToken
|
2016-06-25 10:15:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *client) req(path string, pathValues ...interface{}) request {
|
2016-06-12 22:39:00 +00:00
|
|
|
return request{c, fmt.Sprintf(path, pathValues...), url.Values{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Executes an HTTP request with given parameters and on success returns the response wrapped in a JSON decoder.
|
2016-06-25 10:15:58 +00:00
|
|
|
func (c *client) doJson(method string, path string, values url.Values) (j *json.Decoder, meta ResponseMeta, err error) {
|
2016-06-12 22:39:00 +00:00
|
|
|
// Create request
|
|
|
|
req := &http.Request{
|
|
|
|
Method: method,
|
|
|
|
URL: resolveRelativeUrl(path, values),
|
2016-06-13 17:46:46 +00:00
|
|
|
Header: http.Header{},
|
2016-06-12 22:39:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set request headers
|
|
|
|
if len(c.AuthToken) > 0 {
|
|
|
|
req.Header.Set("X-Auth-Token", c.AuthToken)
|
|
|
|
}
|
|
|
|
req.Header.Set("X-Response-Control", "minified")
|
|
|
|
req.Header.Set("User-Agent", "go-footballdata/0.0")
|
|
|
|
|
|
|
|
// Execute request
|
2016-06-25 10:15:58 +00:00
|
|
|
resp, err := c.httpClient.Do(req)
|
2016-06-12 22:39:00 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2016-06-13 17:46:27 +00:00
|
|
|
defer resp.Body.Close()
|
2016-06-12 22:39:00 +00:00
|
|
|
|
|
|
|
// Save metadata from headers
|
|
|
|
meta = responseMetaFromHeaders(resp.Header)
|
|
|
|
|
2016-06-13 17:46:27 +00:00
|
|
|
// Expect the request to be successful, otherwise throw an error
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
err = errors.New(resp.Status)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-06-12 22:39:00 +00:00
|
|
|
// Download to buffer to allow passing back a fully prepared decoder
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
io.Copy(buf, resp.Body)
|
|
|
|
|
|
|
|
// Wrap JSON decoder around buffered data
|
|
|
|
j = json.NewDecoder(buf)
|
|
|
|
return
|
|
|
|
}
|