-
-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathclient.go
181 lines (153 loc) · 4.94 KB
/
client.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
178
179
180
181
// Copyright © 2019 OpenFoodFacts. All rights reserved.
// Use of this source code is governed by the MIT license which can be found in the LICENSE.txt file.
// This is a go library used to access the OpenFoodFacts.org database for food product, ingredients and
// nutritional data from within your go application.
//
// The main method of using this library is to create a DataOperator and call methods on it.
package openfoodfacts
import (
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
"time"
)
const defaultUserAgent = "OpenFoodFacts - Go - v0.0 - https://github.com/openfoodfacts/openfoodfacts-go"
var (
// ErrNoProduct is an error returned by Client.Product when the product could not be
// retrieved successfully.
// It is not a transient error, the product does not exist.
ErrNoProduct = errors.New("Product retrieval failure")
// ErrUnauthorized is an error returned by Client methods that require a valid user account, but none
// was provided when the Client was instantiated.
ErrUnauthorized = errors.New("Action requires user account")
)
// Client is an OpenFoodFacts client.
// It uses the official API as data source.
type Client struct {
locale string
username string
password string
live bool
client *http.Client
userAgent string
}
// NewClient returns a Client that is capable of talking to the official OpenFoodFacts database via
// the HTTP API, or the dev server if live is false.
//
// The locale should be one of "world" or the country level code for the locale you wish to use.
//
// The username and password should be set to your OpenFoodFacts credentials if you need WRITE access, else provide
// them both as empty strings for anonymous access.
//
// Sandbox mode
//
// If you are testing your application, you should use the test server in order to use the sandbox environment instead
// of the live servers. See the Sandbox() method for more detail and an example.
//
// Timeout
//
// By default the HTTP client doesn't sets a timeout. See the Timeout(time.Duration) method for more detail.
//
// UserAgent
//
// Please set a UserAgent HTTP Header with the name of the app/service querying, the version, system and a URL if
// you have one, so that you are not blocked by mistake
// (e.g. CoolFoodApp - Go - Version 1.0 - https://coolfoodapp.com)
func NewClient(locale, username, password string) Client {
return Client{
locale: locale,
username: username,
password: password,
live: true,
client: &http.Client{},
userAgent: defaultUserAgent,
}
}
// Product returns a new Product for the given code, retrieved from the server.
//
// It will return an error on a failed retrieval, if the retrieval is successful but the API result status is not 1,
// then will return a "ProductRetrievalError" error. This indicates the product is not available.
func (h *Client) Product(code string) (*Product, error) {
request := h.newRequest("GET", "/api/v0/product/%s.json", code)
resp, err := h.client.Do(request)
if err != nil {
return nil, err
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return nil, err
}
productResult := &ProductResult{}
err = json.Unmarshal(body, productResult)
if err != nil {
offs := int64(-1)
if e, ok := err.(*json.UnmarshalTypeError); ok {
offs = e.Offset
} else if e, ok := err.(*json.SyntaxError); ok {
offs = e.Offset
}
if offs > -1 {
a := offs - 50
b := offs + 20
n := int64(len(body))
if a < 0 {
a = 0
}
if b > n {
b = n
}
err = fmt.Errorf("%s at:\n %s⚠️ %s",
string(err.Error()),
string(body[a:offs]), string(body[offs:b]),
)
}
return nil, err
}
if productResult.Status != 1 {
return nil, ErrNoProduct
}
return productResult.Product, nil
}
// Sandbox configures this operator to use the sandbox server at http://world.openfoodfacts.net instead of the live
// server. This is used for testing purposes instead of operating on the live server.
func (h *Client) Sandbox() {
h.live = false
}
// Timeout configures the HTTP client timeout. As the net/http specifies a 0 timeout means no timeout.
func (h *Client) Timeout(timeout time.Duration) {
h.client.Timeout = timeout
}
// UserAgent configures the HTTP User-Agent Header.
func (h *Client) UserAgent(ua string) {
h.userAgent = ua
}
// newRequest is an internal function to setup the request based on the given
// locale/liveness of the given Client.
func (h *Client) newRequest(method, format string, args ...interface{}) *http.Request {
path := fmt.Sprintf(format, args...)
const scheme string = "https"
sub := "ssl-api"
tld := "org"
if !h.live {
sub = "world"
tld = "net"
}
if h.locale != "world" {
sub = h.locale
}
url := fmt.Sprintf("%s://%s.openfoodfacts.%s%s", scheme, sub, tld, path)
request, err := http.NewRequest(method, url, nil)
if err != nil {
return nil
}
if !h.live {
request.SetBasicAuth("off", "off")
}
if h.userAgent != "" {
request.Header.Set("User-Agent", h.userAgent)
}
return request
}