-
Notifications
You must be signed in to change notification settings - Fork 34
/
storage.go
177 lines (142 loc) · 5.27 KB
/
storage.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 zoho
import (
"encoding/gob"
"errors"
"fmt"
"net/http"
"os"
"time"
"google.golang.org/appengine"
"google.golang.org/appengine/datastore"
)
// TokenLoaderSaver is an interface that can be implemented when using a system that does
// not allow disk persistence, or a different type of persistence is required.
// The use case that was in mind was AppEngine where datastore is the only persistence option.
type TokenLoaderSaver interface {
SaveTokens(t AccessTokenResponse) error
LoadAccessAndRefreshToken() (AccessTokenResponse, error)
}
// SaveTokens will check for a provided 'TokenManager' interface
// if one exists it will use its provided method
func (z Zoho) SaveTokens(t AccessTokenResponse) error {
if z.tokenManager != nil {
return z.tokenManager.SaveTokens(t)
}
// Save the token response as GOB to file
file, err := os.OpenFile(z.tokensFile, os.O_WRONLY|os.O_CREATE, 0666)
if err != nil {
return fmt.Errorf("Failed to open file '%s': %s", z.tokensFile, err)
}
enc := gob.NewEncoder(file)
v := TokenWrapper{
Token: z.oauth.token,
}
v.SetExpiry()
err = enc.Encode(v)
if err != nil {
return fmt.Errorf("Failed to encode tokens to file '%s': %s", z.tokensFile, err)
}
return nil
}
// LoadAccessAndRefreshToken will check for a provided 'TokenManager' interface
// if one exists it will use its provided method
func (z Zoho) LoadAccessAndRefreshToken() (AccessTokenResponse, error) {
if z.tokenManager != nil {
return z.tokenManager.LoadAccessAndRefreshToken()
}
// Load the GOB and decode to AccessToken
file, err := os.OpenFile(z.tokensFile, os.O_RDONLY|os.O_CREATE, 0666)
if err != nil {
return AccessTokenResponse{}, fmt.Errorf("Failed to open file '%s': %s", z.tokensFile, err)
}
dec := gob.NewDecoder(file)
var v TokenWrapper
err = dec.Decode(&v)
if err != nil {
return AccessTokenResponse{}, fmt.Errorf("Failed to decode tokens from file '%s': %s", z.tokensFile, err)
}
if v.CheckExpiry() {
return v.Token, ErrTokenExpired
}
return v.Token, nil
}
// ErrTokenExpired should be returned when the token is expired but still exists in persistence
var ErrTokenExpired = errors.New("zoho: oAuth2 token already expired")
// ErrTokenInvalidCode is turned when the autorization code in a request is invalid
var ErrTokenInvalidCode = errors.New("zoho: authorization-code is invalid ")
// ErrClientSecretInvalidCode is turned when the client secret used is invalid
var ErrClientSecretInvalidCode = errors.New("zoho: client secret used in authorization is invalid")
// TokenWrapper should be used to provide the time.Time corresponding to the expiry of an access token
type TokenWrapper struct {
Token AccessTokenResponse
Expires time.Time
}
// SetExpiry sets the TokenWrappers expiry time to now + seconds until expiry
func (t *TokenWrapper) SetExpiry() {
t.Expires = time.Now().Add(time.Duration(t.Token.ExpiresIn) * time.Second)
}
// CheckExpiry if the token expired before this instant
func (t *TokenWrapper) CheckExpiry() bool {
return t.Expires.Before(time.Now())
}
func (z *Zoho) CheckForSavedTokens() error {
t, err := z.LoadAccessAndRefreshToken()
z.oauth.token = t
if err != nil && err == ErrTokenExpired {
return err
}
if (t != AccessTokenResponse{}) && err != ErrTokenExpired {
return nil
}
return fmt.Errorf("No saved tokens")
}
// DatastoreManager is an example TokenManager that satisfies the TokenManager interface
// When instantiating, user must provide the *http.Request for the current app engine request
// and the token key where the tokens are to be saved to/loaded from.
type DatastoreManager struct {
Request *http.Request
EntityNamespace string
TokensKey string
}
// LoadAccessAndRefreshToken will use datastore package to get tokens from the datastore under the entity namespace
// 'ZohoAccessTokens' unless a value is provided to the EntityNamespace field
func (d DatastoreManager) LoadAccessAndRefreshToken() (AccessTokenResponse, error) {
t := TokenWrapper{}
if d.Request == nil || d.TokensKey == "" {
return AccessTokenResponse{}, fmt.Errorf("Must provide the *http.Request for the current request and a valid token key")
}
entity := "ZohoAccessTokens"
if d.EntityNamespace != "" {
entity = d.EntityNamespace
}
ctx := appengine.NewContext(d.Request)
k := datastore.NewKey(ctx, entity, d.TokensKey, 0, nil)
if err := datastore.Get(ctx, k, &t); err != nil {
return AccessTokenResponse{}, fmt.Errorf("Failed to retrieve tokens from datastore: %s", err)
}
if t.CheckExpiry() {
return AccessTokenResponse{}, ErrTokenExpired
}
return t.Token, nil
}
// SaveTokens will use datastore package to put tokens to the datastore under the entity namespace
// 'ZohoAccessTokens' unless a value is provided to the EntityNamespace field
func (d DatastoreManager) SaveTokens(t AccessTokenResponse) error {
if d.Request == nil || d.TokensKey == "" {
return fmt.Errorf("Must provide the *http.Request for the current request and a valid token key")
}
entity := "ZohoAccessTokens"
if d.EntityNamespace != "" {
entity = d.EntityNamespace
}
ctx := appengine.NewContext(d.Request)
k := datastore.NewKey(ctx, entity, d.TokensKey, 0, nil)
v := TokenWrapper{
Token: t,
}
v.SetExpiry()
if _, err := datastore.Put(ctx, k, v); err != nil {
return fmt.Errorf("Failed to save tokens to datastore: %s", err)
}
return nil
}