forked from keighl/postmark
-
Notifications
You must be signed in to change notification settings - Fork 0
/
postmark.go
122 lines (104 loc) · 3.33 KB
/
postmark.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
// Package postmark ...
package postmark
import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
)
var (
postmarkURL = `https://api.postmarkapp.com`
)
// Client provides a connection to the Postmark API
type Client struct {
// HTTPClient is &http.Client{} by default
HTTPClient *http.Client
// Server Token: Used for requests that require server level privileges. This token can be found on the Credentials tab under your Postmark server.
ServerToken string
// AccountToken: Used for requests that require account level privileges. This token is only accessible by the account owner, and can be found on the Account tab of your Postmark account.
AccountToken string
// BaseURL is the root API endpoint
BaseURL string
}
const (
server_token = "server"
account_token = "account"
)
// Options is an object to hold variable parameters to perform request.
type parameters struct {
// Method is HTTP method type.
Method string
// Path is postfix for URI.
Path string
// Payload for the request.
Payload interface{}
// TokenType defines which token to use
TokenType string
}
// NewClient builds a new Client pointer using the provided tokens, a default HTTPClient, and a default API base URL
// Accepts `Server Token`, and `Account Token` as arguments
// http://developer.postmarkapp.com/developer-api-overview.html#authentication
func NewClient(serverToken string, accountToken string) *Client {
return &Client{
HTTPClient: &http.Client{},
ServerToken: serverToken,
AccountToken: accountToken,
BaseURL: postmarkURL,
}
}
func (client *Client) doRequest(opts parameters, dst interface{}) error {
return client.withRetry(opts, dst, 1)
}
func (client *Client) withRetry(opts parameters, dst interface{}, retries int) error {
url := fmt.Sprintf("%s/%s", client.BaseURL, opts.Path)
req, err := http.NewRequest(opts.Method, url, nil)
if err != nil {
return err
}
if opts.Payload != nil {
payloadData, err := json.Marshal(opts.Payload)
if err != nil {
return fmt.Errorf("error marshalling payload %+v - %s", opts.Payload, err.Error())
}
req.Body = ioutil.NopCloser(bytes.NewBuffer(payloadData))
}
req.Header.Add("Accept", "application/json")
req.Header.Add("Content-Type", "application/json")
switch opts.TokenType {
case account_token:
req.Header.Add("X-Postmark-Account-Token", client.AccountToken)
default:
req.Header.Add("X-Postmark-Server-Token", client.ServerToken)
}
req.Close = true
res, err := client.HTTPClient.Do(req)
if err != nil {
return fmt.Errorf("error executing request - %s", err.Error())
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
if err != nil {
return fmt.Errorf("error reading body - %s", err.Error())
}
if res.StatusCode == http.StatusGatewayTimeout {
if retries > 0 {
return client.withRetry(opts, dst, retries-1)
}
}
if err := json.Unmarshal(body, dst); err != nil {
return fmt.Errorf("error unmarshalling response %s - %s - have status %d", string(body), err.Error(), res.StatusCode)
}
return nil
}
// APIError represents errors returned by Postmark
type APIError struct {
// ErrorCode: see error codes here (http://developer.postmarkapp.com/developer-api-overview.html#error-codes)
ErrorCode int64
// Message contains error details
Message string
}
// Error returns the error message details
func (res APIError) Error() string {
return res.Message
}