-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathinterfaces.go
348 lines (271 loc) · 11.4 KB
/
interfaces.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
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
// Copyright 2019 Aporeto Inc.
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package bahamut
import (
"context"
"crypto/tls"
"net/http"
"go.aporeto.io/elemental"
)
type processorFinderFunc func(identity elemental.Identity) (Processor, error)
type eventPusherFunc func(...*elemental.Event)
type retrieveHandlersFunc func() map[string]http.HandlerFunc
// AuthAction is the type of action an Authenticator or an Authorizer can return.
type AuthAction int
const (
// AuthActionOK means the authenticator/authorizer takes the responsibility
// to grant the request. The execution in the chain will
// stop and will be considered as a success.
AuthActionOK AuthAction = iota
// AuthActionKO means the authenticator/authorizer takes the responsibility
// to reject the request. The execution in the chain will
// stop and will be considered as a success.
AuthActionKO
// AuthActionContinue means the authenticator/authorizer does not take
// any responsabolity and let the chain continue.
// If the last authenticator in the chain returns AuthActionContinue,
// Then the request will be considered as a success.
AuthActionContinue
)
// Server is the interface of a bahamut server.
type Server interface {
// RegisterProcessor registers a new Processor for a particular Identity.
RegisterProcessor(Processor, elemental.Identity) error
// UnregisterProcessor unregisters a registered Processor for a particular identity.
UnregisterProcessor(elemental.Identity) error
// ProcessorForIdentity returns the registered Processor for a particular identity.
ProcessorForIdentity(elemental.Identity) (Processor, error)
// ProcessorsCount returns the number of registered processors.
ProcessorsCount() int
// RegisterCustomRouteHandler registers a generic HTTP handler for a given
// path. Users are responsible for all processing in this path.
RegisterCustomRouteHandler(path string, handler http.HandlerFunc) error
// UnregisterCustomRouteHandler unregisters a generic HTTP handler for a path.
UnregisterCustomRouteHandler(path string) error
// CustomHandlers returns a map of all the custom handlers.
CustomHandlers() map[string]http.HandlerFunc
// Push pushes the given events to all active sessions.
// It will use the PubSubClient configured in the pushConfig.
Push(...*elemental.Event)
// RoutesInfo returns the routing information of the server.
RoutesInfo() map[int][]RouteInfo
// VersionsInfo returns additional versioning info.
VersionsInfo() map[string]any
// PushEndpoint returns the configured push endpoints.
// If the push server is not active, it will return an
// empty string.
PushEndpoint() string
// Run runs the server using the given context.Context.
// You can stop the server by canceling the context.
Run(context.Context)
}
// A ResponseWriter is a function you can use in
// the Context to handle the writing of the response by
// yourself. You are responsible for the full handling of the response,
// including encoding, setting the CORS headers etc.
type ResponseWriter func(w http.ResponseWriter) int
// A Context contains all information about a current operation.
type Context interface {
// Identifier returns the internal unique identifier of the context.
Identifier() string
// Context returns the underlying context.Context.
Context() context.Context
// Request returns the underlying *elemental.Request.
Request() *elemental.Request
// InputData returns the data sent by the client
InputData() any
// SetInputData replaces the current input data.
SetInputData(any)
// OutputData returns the current output data.
OutputData() any
// SetOutputData sets the data that will be returned to the client.
//
// If you use SetOutputData after having already used SetResponseWriter,
// the call will panic.
SetOutputData(any)
// SetDisableOutputDataPush will instruct the bahamut server to
// not automatically push the content of OutputData.
SetDisableOutputDataPush(bool)
// SetResponseWriter sets the ResponseWriter function to use to write the response back to the client.
//
// No additional operation or check will be performed by Bahamut. You are responsible
// for correctly encoding the response, setting the header etc. This is useful when
// you want to handle a route that is note really fitting in the handling of an elemental Model
// like for instance handling file download, response streaming etc.
//
// If you use SetResponseWriter after having already used SetOutputData,
// the call will panic.
SetResponseWriter(ResponseWriter)
// Set count sets the count.
SetCount(int)
// Count returns the current count.
Count() int
// SetRedirect sets the redirect URL.
SetRedirect(string)
// Redirect returns the current value for redirection
Redirect() string
// SetStatusCode sets the status code that will be returned to the client.
SetStatusCode(int)
// StatusCode returns the current status code.
StatusCode() int
// AddMessage adds a custom message that will be sent as repponse header.
AddMessage(string)
// SetClaims sets the claims.
SetClaims(claims []string)
// Claims returns the list of claims.
Claims() []string
// Claims returns claims in a map.
ClaimsMap() map[string]string
// Duplicate creates a copy of the Context.
Duplicate() Context
// SetNext can be use to give the next pagination token.
SetNext(string)
// EnqueueEvents enqueues the given event to the Context.
//
// Bahamut will automatically generate events on the currently processed object.
// But if your processor creates other objects alongside with the main one and you want to
// send a push to the user, then you can use this method.
//
// The events you enqueue using EnqueueEvents will be sent in order to the enqueueing, and
// *before* the main object related event.
EnqueueEvents(...*elemental.Event)
// SetMetadata sets opaque metadata that can be reteieved by Metadata().
SetMetadata(key, value any)
// Metadata returns the opaque data set by using SetMetadata().
Metadata(key any) any
// outputCookies adds cookies to the response that
// will be returned to the client.
AddOutputCookies(cookies ...*http.Cookie)
}
// Processor is the interface for a Processor Unit
type Processor any
// RetrieveManyProcessor is the interface a processor must implement
// in order to be able to manage OperationRetrieveMany.
type RetrieveManyProcessor interface {
ProcessRetrieveMany(Context) error
}
// RetrieveProcessor is the interface a processor must implement
// in order to be able to manage OperationRetrieve.
type RetrieveProcessor interface {
ProcessRetrieve(Context) error
}
// CreateProcessor is the interface a processor must implement
// in order to be able to manage OperationCreate.
type CreateProcessor interface {
ProcessCreate(Context) error
}
// UpdateProcessor is the interface a processor must implement
// in order to be able to manage OperationUpdate.
type UpdateProcessor interface {
ProcessUpdate(Context) error
}
// DeleteProcessor is the interface a processor must implement
// in order to be able to manage OperationDelete.
type DeleteProcessor interface {
ProcessDelete(Context) error
}
// PatchProcessor is the interface a processor must implement
// in order to be able to manage OperationPatch.
type PatchProcessor interface {
ProcessPatch(Context) error
}
// InfoProcessor is the interface a processor must implement
// in order to be able to manage OperationInfo.
type InfoProcessor interface {
ProcessInfo(Context) error
}
// RequestAuthenticator is the interface that must be implemented in order to
// to be used as the Bahamut Authenticator.
type RequestAuthenticator interface {
AuthenticateRequest(Context) (AuthAction, error)
}
// SessionAuthenticator is the interface that must be implemented in order to
// be used as the initial Web socket session Authenticator.
type SessionAuthenticator interface {
AuthenticateSession(Session) (AuthAction, error)
}
// Authorizer is the interface that must be implemented in order to
// to be used as the Bahamut Authorizer.
type Authorizer interface {
IsAuthorized(Context) (AuthAction, error)
}
// PushDispatchHandler is the interface that must be implemented in order to
// to be used as the Bahamut Push Dispatch handler.
type PushDispatchHandler interface {
// OnPushSessionInit is called when a new push session wants to connect.
// If it returns false, the push session will be considered
// as forbidden. If it returns an error, it will be returned
// to the client.
OnPushSessionInit(PushSession) (bool, error)
// OnPushSessionStart is called when a new push session starts.
OnPushSessionStart(PushSession)
// OnPushSessionStart is called when a new push session terminated.
OnPushSessionStop(PushSession)
// ShouldDispatch is called to decide if the given event should be sent to the given session.
// The last parameter will contain whatever has been returned by SummarizeEvent.
// It is NOT safe to modify the given *elemental.Event. This would cause
// race conditions. You can only safely read from it.
ShouldDispatch(PushSession, *elemental.Event, any) (bool, error)
// RelatedEventIdentities allows to return a list of related identities
// associated to the main event identity. This allows to pass filtering
// in case a push on identity A must also trigger a push on identity B and C.
RelatedEventIdentities(string) []string
// SummarizeEvent is called once per event and allows the implementation
// to return an interface that will be passed to ShouldDispatch.
// If you need to decode an event to read some information to make a
// dispatch decision, this is a good place as it will allow you to only
// do this once.
SummarizeEvent(event *elemental.Event) (any, error)
}
// PushPublishHandler is the interface that must be implemented in order to
// to be used as the Bahamut Push Publish handler.
type PushPublishHandler interface {
ShouldPublish(*elemental.Event) (bool, error)
}
// Auditer is the interface an object must implement in order to handle
// audit traces.
type Auditer interface {
Audit(Context, error)
}
// A RateLimiter is the interface an object must implement in order to
// limit the rate of the incoming requests.
type RateLimiter interface {
RateLimit(*http.Request) (bool, error)
}
// Session is the interface of a generic websocket session.
type Session interface {
Identifier() string
Parameter(string) string
Header(string) string
PushConfig() *elemental.PushConfig
SetClaims([]string)
Claims() []string
ClaimsMap() map[string]string
Token() string
TLSConnectionState() *tls.ConnectionState
Metadata() any
SetMetadata(any)
Context() context.Context
ClientIP() string
Cookie(string) (*http.Cookie, error)
}
// PushSession is a Push Session
type PushSession interface {
Session
DirectPush(...*elemental.Event)
}
// A CORSPolicyController allows to return
// the CORS policy for a given http.Request.
type CORSPolicyController interface {
// PolicyForRequest returns the CORSPolicy to
// apply for the given http.Request.
PolicyForRequest(*http.Request) *CORSPolicy
}