8
8
"net/http"
9
9
10
10
"github.com/pkg/errors"
11
- "github.com/tomasen/realip"
12
11
)
13
12
14
13
// FilterEndpoint defines the filter URL castle.io side
@@ -17,80 +16,13 @@ var FilterEndpoint = "https://api.castle.io/v1/filter"
17
16
// RiskEndpoint defines the risk URL castle.io side
18
17
var RiskEndpoint = "https://api.castle.io/v1/risk"
19
18
20
- type Event struct {
21
- EventType EventType
22
- EventStatus EventStatus
23
- }
24
-
25
- // EventType is an enum defining types of event castle tracks
26
- type EventType string
27
-
28
- // See https://docs.castle.io/docs/events
29
- const (
30
- EventTypeLogin EventType = "$login"
31
- EventTypeRegistration EventType = "$registration"
32
- EventTypeProfileUpdate EventType = "$profile_update"
33
- EventTypeProfileReset EventType = "$profile_reset"
34
- EventTypePasswordResetRequest EventType = "$password_reset_request"
35
- EventTypeChallenge EventType = "$challenge"
36
- EventTypeLogout EventType = "&logout"
37
- )
38
-
39
- // EventStatus is an enum defining the statuses for a given event.
40
- type EventStatus string
41
-
42
- // See https://docs.castle.io/docs/events
43
- const (
44
- EventStatusAttempted EventStatus = "$attempted"
45
- EventStatusSucceeded EventStatus = "$succeeded"
46
- EventStatusFailed EventStatus = "$failed"
47
- EventStatusRequested EventStatus = "$requested"
48
- )
49
-
50
- // RecommendedAction encapsulates the 3 possible responses from auth call (allow, challenge, deny)
51
- type RecommendedAction string
52
-
53
- // See https://castle.io/docs/authentication
54
- const (
55
- RecommendedActionNone RecommendedAction = ""
56
- RecommendedActionAllow RecommendedAction = "allow"
57
- RecommendedActionChallenge RecommendedAction = "challenge"
58
- RecommendedActionDeny RecommendedAction = "deny"
59
- )
60
-
61
19
// New creates a new castle client
62
20
func New (secret string ) (* Castle , error ) {
63
21
client := & http.Client {}
64
22
65
23
return NewWithHTTPClient (secret , client )
66
24
}
67
25
68
- // HeaderAllowList keeps a list of headers that will be forwarded to castle
69
- var HeaderAllowList = []string {
70
- "Accept" ,
71
- "Accept-Charset" ,
72
- "Accept-Datetime" ,
73
- "Accept-Encoding" ,
74
- "Accept-Language" ,
75
- "Cache-Control" ,
76
- "Connection" ,
77
- "Content-Length" ,
78
- "Content-Type" ,
79
- "Dnt" ,
80
- "Host" ,
81
- "Origin" ,
82
- "Pragma" ,
83
- "Referer" ,
84
- "Sec-Fetch-Dest" ,
85
- "Sec-Fetch-Mode" ,
86
- "Sec-Fetch-Site" ,
87
- "Sec-Fetch-User" ,
88
- "Te" ,
89
- "Upgrade-Insecure-Requests" ,
90
- "User-Agent" ,
91
- "X-Castle-Request-Token" ,
92
- }
93
-
94
26
// NewWithHTTPClient same as New but allows passing of http.Client with custom config
95
27
func NewWithHTTPClient (secret string , client * http.Client ) (* Castle , error ) {
96
28
return & Castle {client : client , apiSecret : secret }, nil
@@ -102,82 +34,6 @@ type Castle struct {
102
34
apiSecret string
103
35
}
104
36
105
- // Context captures data from HTTP request
106
- type Context struct {
107
- IP string `json:"ip"`
108
- Headers map [string ]string `json:"headers"`
109
- RequestToken string `json:"request_token"`
110
- }
111
-
112
- func isHeaderAllowed (header string ) bool {
113
- for _ , allowedHeader := range HeaderAllowList {
114
- if header == http .CanonicalHeaderKey (allowedHeader ) {
115
- return true
116
- }
117
- }
118
- return false
119
- }
120
-
121
- // ContextFromRequest builds castle context from current http.Request
122
- func ContextFromRequest (r * http.Request ) * Context {
123
- headers := make (map [string ]string )
124
-
125
- for requestHeader := range r .Header {
126
- if isHeaderAllowed (requestHeader ) {
127
- headers [requestHeader ] = r .Header .Get (requestHeader )
128
- }
129
- }
130
-
131
- requestToken := getRequestToken (r )
132
-
133
- return & Context {IP : realip .FromRequest (r ), Headers : headers , RequestToken : requestToken }
134
- }
135
-
136
- func getRequestToken (r * http.Request ) string {
137
- // RequestToken is X-Castle-Request-Token
138
- return r .Header .Get ("HTTP_X_CASTLE_REQUEST_TOKEN" )
139
- }
140
-
141
- type Request struct {
142
- Context * Context
143
- Event Event
144
- User User
145
- Properties map [string ]string
146
- }
147
-
148
- type User struct {
149
- ID string `json:"id"`
150
- Email string `json:"email,omitempty"`
151
- Phone string `json:"phone,omitempty"`
152
- Name string `json:"name,omitempty"`
153
- RegisteredAt string `json:"registered_at,omitempty"`
154
- Traits map [string ]string `json:"traits,omitempty"`
155
- }
156
-
157
- type castleAPIRequest struct {
158
- Type EventType `json:"type"`
159
- Status EventStatus `json:"status"`
160
- RequestToken string `json:"request_token"`
161
- User User `json:"user"`
162
- Context * Context `json:"context"`
163
- Properties map [string ]string `json:"properties,omitempty"`
164
- }
165
-
166
- type castleAPIResponse struct {
167
- Type string `json:"type"`
168
- Message string `json:"message"`
169
- Risk float32 `json:"risk"`
170
- Policy struct {
171
- Name string `json:"name"`
172
- ID string `json:"id"`
173
- RevisionID string `json:"revision_id"`
174
- Action string `json:"action"`
175
- } `json:"policy"`
176
- Device struct {
177
- Token string `json:"token"`
178
- } `json:"device"`
179
- }
180
-
181
37
// Filter sends a filter request to castle.io
182
38
// see https://reference.castle.io/#operation/filter for details
183
39
func (c * Castle ) Filter (ctx context.Context , req * Request ) (RecommendedAction , error ) {
0 commit comments