-
Notifications
You must be signed in to change notification settings - Fork 1
/
auth_code.go
101 lines (89 loc) · 2.65 KB
/
auth_code.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
package gcloudcx
import (
"context"
"net/http"
"net/url"
"time"
"github.com/gildas/go-errors"
"github.com/gildas/go-request"
"github.com/google/uuid"
)
// AuthorizationCodeGrant implements Gcloud's Client Authorization Code Grants
//
// See: https://developer.mypurecloud.com/api/rest/authorization/use-authorization-code.html
type AuthorizationCodeGrant struct {
ClientID uuid.UUID
Secret string
Code string
RedirectURL *url.URL
Token AccessToken
CustomData interface{}
TokenUpdated chan UpdatedAccessToken
}
// GetID gets the client Identifier
//
// Implements core.Identifiable
func (grant *AuthorizationCodeGrant) GetID() uuid.UUID {
return grant.ClientID
}
// Authorize this Grant with Gcloud
//
// Implements Authorizable
func (grant *AuthorizationCodeGrant) Authorize(context context.Context, client *Client) (err error) {
log := client.GetLogger(context).Child(nil, "authorize", "grant", "authorization_code")
log.Infof("Authenticating with %s using Authorization Code grant", client.Region)
// Validates the Grant
if grant.ClientID == uuid.Nil {
return errors.ArgumentMissing.With("ClientID")
}
if len(grant.Secret) == 0 {
return errors.ArgumentMissing.With("Secret")
}
if len(grant.Code) == 0 {
return errors.ArgumentMissing.With("Code")
}
// Resets the token before authenticating
grant.Token.Reset()
response := struct {
AccessToken string `json:"access_token,omitempty"`
TokenType string `json:"token_type,omitempty"`
ExpiresIn int64 `json:"expires_in,omitempty"`
Error string `json:"error,omitempty"`
}{}
err = client.SendRequest(
context,
NewURI("%s/oauth/token", client.LoginURL),
&request.Options{
Method: http.MethodPost,
Authorization: request.BasicAuthorization(grant.ClientID.String(), grant.Secret),
Payload: map[string]string{
"grant_type": "authorization_code",
"code": grant.Code,
"redirect_uri": grant.RedirectURL.String(),
},
},
&response,
)
if err != nil {
return err
}
// Saves the token
grant.Token.Type = response.TokenType
grant.Token.Token = response.AccessToken
grant.Token.ExpiresOn = time.Now().Add(time.Duration(response.ExpiresIn) * time.Second)
log.Debugf("New %s token expires on %s", grant.Token.Type, grant.Token.ExpiresOn)
if grant.TokenUpdated != nil {
log.Debugf("Sending new token to TokenUpdated chan")
grant.TokenUpdated <- UpdatedAccessToken{
AccessToken: grant.Token,
CustomData: grant.CustomData,
}
}
return
}
// AccessToken gives the access Token carried by this Grant
//
// Implements Authorizable
func (grant *AuthorizationCodeGrant) AccessToken() *AccessToken {
return &grant.Token
}