-
Notifications
You must be signed in to change notification settings - Fork 31
/
protocol.h
552 lines (481 loc) · 21.4 KB
/
protocol.h
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
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
#pragma once
#include <memory>
#include <string>
#include "envoy/buffer/buffer.h"
#include "envoy/common/pure.h"
#include "envoy/config/typed_config.h"
#include "envoy/registry/registry.h"
#include "source/common/common/assert.h"
#include "source/common/config/utility.h"
#include "source/common/singleton/const_singleton.h"
#include "src/application_protocols/thrift/conn_state.h"
#include "src/application_protocols/thrift/decoder_events.h"
#include "src/application_protocols/thrift/metadata.h"
#include "src/application_protocols/thrift/thrift.h"
#include "src/application_protocols/thrift/thrift_object.h"
#include "src/application_protocols/thrift/transport.h"
#include "absl/strings/string_view.h"
namespace Envoy {
namespace Extensions {
namespace NetworkFilters {
namespace ThriftProxy {
class DirectResponse;
using DirectResponsePtr = std::unique_ptr<DirectResponse>;
/**
* Protocol represents the operations necessary to implement the a generic Thrift protocol.
* See https://github.com/apache/thrift/blob/master/doc/specs/thrift-protocol-spec.md
*/
class Protocol {
public:
virtual ~Protocol() = default;
/**
* @return const std::string& the human-readable name of the protocol
*/
virtual const std::string& name() const PURE;
/**
* @return ProtocolType the protocol type
*/
virtual ProtocolType type() const PURE;
/**
* For protocol-detecting implementations, set the underlying type based on external
* (e.g. transport-level) information).
* @param type ProtocolType to explicitly set
*/
virtual void setType(ProtocolType) {}
/**
* Reads the start of a Thrift protocol message from the buffer and updates the metadata
* parameter with values from the message header. If successful, the message header is removed
* from the buffer.
* @param buffer the buffer to read from
* @param metadata MessageMetadata to be updated with name, message type, and sequence id.
* @return true if a message header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid message header
*/
virtual bool readMessageBegin(Buffer::Instance& buffer, MessageMetadata& metadata) PURE;
/**
* Reads the end of a Thrift protocol message from the buffer. If successful, the message footer
* is removed from the buffer.
* @param buffer the buffer to read from
* @return true if a message footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid message footer
*/
virtual bool readMessageEnd(Buffer::Instance& buffer) PURE;
/**
* Peeks the start of a Thrift protocol reply payload in the buffer and updates the reply
* type parameter with the reply type of the payload.
* @param buffer the buffer to peek from
* @param reply_type ReplyType to set the payload's reply type to success or error
* @return true if reply type was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid payload
*/
virtual bool peekReplyPayload(Buffer::Instance& buffer, ReplyType& reply_type) PURE;
/**
* Reads the start of a Thrift struct from the buffer and updates the name parameter with the
* value from the struct header. If successful, the struct header is removed from the buffer.
* @param buffer the buffer to read from
* @param name updated with the struct name on success only
* @return true if a struct header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid struct header
*/
virtual bool readStructBegin(Buffer::Instance& buffer, std::string& name) PURE;
/**
* Reads the end of a Thrift struct from the buffer. If successful, the struct footer is removed
* from the buffer.
* @param buffer the buffer to read from
* @return true if a struct footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid struct footer
*/
virtual bool readStructEnd(Buffer::Instance& buffer) PURE;
/**
* Reads the start of a Thrift struct field from the buffer and updates the name, field_type, and
* field_id parameters with the values from the field header. If successful, the field header is
* removed from the buffer.
* @param buffer the buffer to read from
* @param name updated with the field name on success only
* @param field_type updated with the FieldType on success only
* @param field_id updated with the field ID on success only
* @return true if a field header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid field header
*/
virtual bool readFieldBegin(Buffer::Instance& buffer, std::string& name, FieldType& field_type,
int16_t& field_id) PURE;
/**
* Reads the end of a Thrift struct field from the buffer. If successful, the field footer is
* removed from the buffer.
* @param buffer the buffer to read from
* @return true if a field footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid field footer
*/
virtual bool readFieldEnd(Buffer::Instance& buffer) PURE;
/**
* Reads the start of a Thrift map from the buffer and updates the key_type, value_type, and size
* parameters with the values from the map header. If successful, the map header is removed from
* the buffer.
* @param buffer the buffer to read from
* @param key_type updated with map key FieldType on success only
* @param value_type updated with map value FieldType on success only
* @param size updated with the number of key-value pairs in the map on success only
* @return true if a map header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid map header
*/
virtual bool readMapBegin(Buffer::Instance& buffer, FieldType& key_type, FieldType& value_type,
uint32_t& size) PURE;
/**
* Reads the end of a Thrift map from the buffer. If successful, the map footer is removed from
* the buffer.
* @param buffer the buffer to read from
* @return true if a map footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid map footer
*/
virtual bool readMapEnd(Buffer::Instance& buffer) PURE;
/**
* Reads the start of a Thrift list from the buffer and updates the elem_type, and size
* parameters with the values from the list header. If successful, the list header is removed from
* the buffer.
* @param buffer the buffer to read from
* @param elem_type updated with list element FieldType on success only
* @param size updated with the number of list members on success only
* @return true if a list header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid list header
*/
virtual bool readListBegin(Buffer::Instance& buffer, FieldType& elem_type, uint32_t& size) PURE;
/**
* Reads the end of a Thrift list from the buffer. If successful, the list footer is removed from
* the buffer.
* @param buffer the buffer to read from
* @return true if a list footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid list footer
*/
virtual bool readListEnd(Buffer::Instance& buffer) PURE;
/**
* Reads the start of a Thrift set from the buffer and updates the elem_type, and size
* parameters with the values from the set header. If successful, the set header is removed from
* the buffer.
* @param buffer the buffer to read from
* @param elem_type updated with set element FieldType on success only
* @param size updated with the number of set members on success only
* @return true if a set header was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set header
*/
virtual bool readSetBegin(Buffer::Instance& buffer, FieldType& elem_type, uint32_t& size) PURE;
/**
* Reads the end of a Thrift set from the buffer. If successful, the set footer is removed from
* the buffer.
* @param buffer the buffer to read from
* @return true if a set footer was successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readSetEnd(Buffer::Instance& buffer) PURE;
/**
* Reads a boolean value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readBool(Buffer::Instance& buffer, bool& value) PURE;
/**
* Reads a byte value from the buffer and updates value. If successful, the value is removed from
* the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readByte(Buffer::Instance& buffer, uint8_t& value) PURE;
/**
* Reads a int16_t value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readInt16(Buffer::Instance& buffer, int16_t& value) PURE;
/**
* Reads a int32_t value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readInt32(Buffer::Instance& buffer, int32_t& value) PURE;
/**
* Reads a int64_t value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readInt64(Buffer::Instance& buffer, int64_t& value) PURE;
/**
* Reads a double value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readDouble(Buffer::Instance& buffer, double& value) PURE;
/**
* Reads a string value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readString(Buffer::Instance& buffer, std::string& value) PURE;
/**
* Reads a binary value from the buffer and updates value. If successful, the value is removed
* from the buffer.
* @param buffer the buffer to read from
* @param value updated with the value read from the buffer
* @return true if a value successfully read, false if more data is required
* @throw EnvoyException if the data is not a valid set footer
*/
virtual bool readBinary(Buffer::Instance& buffer, std::string& value) PURE;
/**
* Writes the start of a Thrift protocol message to the buffer.
* @param buffer Buffer::Instance to modify
* @param metadata MessageMetadata for the message to write.
*/
virtual void writeMessageBegin(Buffer::Instance& buffer, const MessageMetadata& metadata) PURE;
/**
* Writes the end of a Thrift protocol message to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeMessageEnd(Buffer::Instance& buffer) PURE;
/**
* Writes the start of a Thrift struct to the buffer.
* @param buffer Buffer::Instance to modify
* @param name the struct name, if known
*/
virtual void writeStructBegin(Buffer::Instance& buffer, const std::string& name) PURE;
/**
* Writes the end of a Thrift struct to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeStructEnd(Buffer::Instance& buffer) PURE;
/**
* Writes the start of a Thrift struct field to the buffer
* @param buffer Buffer::Instance to modify
* @param name the field name, if known
* @param field_type the field's FieldType
* @param field_id the field ID
*/
virtual void writeFieldBegin(Buffer::Instance& buffer, const std::string& name,
FieldType field_type, int16_t field_id) PURE;
/**
* Writes the end of a Thrift struct field to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeFieldEnd(Buffer::Instance& buffer) PURE;
/**
* Writes the start of a Thrift map to the buffer.
* @param buffer Buffer::Instance to modify
* @param key_type the map key FieldType
* @param value_type the map value FieldType
* @param size the number of key-value pairs in the map
*/
virtual void writeMapBegin(Buffer::Instance& buffer, FieldType key_type, FieldType value_type,
uint32_t size) PURE;
/**
* Writes the end of a Thrift map to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeMapEnd(Buffer::Instance& buffer) PURE;
/**
* Writes the start of a Thrift list to the buffer.
* @param buffer Buffer::Instance to modify
* @param elem_type the list element FieldType
* @param size the number of list members
*/
virtual void writeListBegin(Buffer::Instance& buffer, FieldType elem_type, uint32_t size) PURE;
/**
* Writes the end of a Thrift list to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeListEnd(Buffer::Instance& buffer) PURE;
/**
* Writes the start of a Thrift set to the buffer.
* @param buffer Buffer::Instance to modify
* @param elem_type the set element FieldType
* @param size the number of set members
*/
virtual void writeSetBegin(Buffer::Instance& buffer, FieldType elem_type, uint32_t size) PURE;
/**
* Writes the end of a Thrift set to the buffer.
* @param buffer Buffer::Instance to modify
*/
virtual void writeSetEnd(Buffer::Instance& buffer) PURE;
/**
* Writes a boolean value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value bool to write
*/
virtual void writeBool(Buffer::Instance& buffer, bool value) PURE;
/**
* Writes a byte value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value uint8_t to write
*/
virtual void writeByte(Buffer::Instance& buffer, uint8_t value) PURE;
/**
* Writes a int16_t value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value int16_t to write
*/
virtual void writeInt16(Buffer::Instance& buffer, int16_t value) PURE;
/**
* Writes a int32_t value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value int32_t to write
*/
virtual void writeInt32(Buffer::Instance& buffer, int32_t value) PURE;
/**
* Writes a int64_t value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value int64_t to write
*/
virtual void writeInt64(Buffer::Instance& buffer, int64_t value) PURE;
/**
* Writes a double value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value double to write
*/
virtual void writeDouble(Buffer::Instance& buffer, double value) PURE;
/**
* Writes a string value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value std::string to write
*/
virtual void writeString(Buffer::Instance& buffer, const std::string& value) PURE;
/**
* Writes a binary value to the buffer.
* @param buffer Buffer::Instance to modify
* @param value std::string to write
*/
virtual void writeBinary(Buffer::Instance& buffer, const std::string& value) PURE;
/**
* Indicates whether a protocol uses start-of-connection messages to negotiate protocol options.
* If this method returns true, the Protocol must invoke setProtocolUpgradeMessage during
* readMessageBegin if it detects an upgrade request.
*
* @return true for protocols that exchange messages at the start of a connection to negotiate
* protocol upgrade (or options)
*/
virtual bool supportsUpgrade() { return false; }
/**
* Creates an opaque DecoderEventHandlerSharedPtr that can decode a downstream client's upgrade
* request. When the request is complete, the decoder is passed back to writeUpgradeResponse
* to allow the Protocol to update its internal state and generate a response to the request.
*
* @return a DecoderEventHandlerSharedPtr that decodes a downstream client's upgrade request
*/
virtual DecoderEventHandlerSharedPtr upgradeRequestDecoder() { return nullptr; }
/**
* Writes a response to a downstream client's upgrade request.
* @param decoder DecoderEventHandlerSharedPtr created by upgradeRequestDecoder
* @return DirectResponsePtr containing an upgrade response
*/
virtual DirectResponsePtr upgradeResponse(const DecoderEventHandler& decoder) {
UNREFERENCED_PARAMETER(decoder);
return nullptr;
}
/**
* Checks whether a given upstream connection can be upgraded and generates an upgrade request
* message. If this method returns a ThriftObject it will be used to decode the upstream's next
* response.
*
* @param transport the Transport to use for decoding the response
* @param state ThriftConnectionState tracking whether upgrade has already been performed
* @param buffer Buffer::Instance to modify with an upgrade request
* @return a ThriftObject capable of decoding an upgrade response or nullptr if upgrade was
* already completed (successfully or not)
*/
virtual ThriftObjectPtr attemptUpgrade(Transport& transport, ThriftConnectionState& state,
Buffer::Instance& buffer) {
UNREFERENCED_PARAMETER(transport);
UNREFERENCED_PARAMETER(state);
UNREFERENCED_PARAMETER(buffer);
return nullptr;
}
/**
* Completes an upgrade previously started via attemptUpgrade.
* @param response ThriftObject created by attemptUpgrade, after the response has completed
* decoding
*/
virtual void completeUpgrade(ThriftConnectionState& state, ThriftObject& response) {
UNREFERENCED_PARAMETER(state);
UNREFERENCED_PARAMETER(response);
}
};
using ProtocolPtr = std::unique_ptr<Protocol>;
/**
* A DirectResponse manipulates a Protocol to directly create a Thrift response message.
*/
class DirectResponse {
public:
virtual ~DirectResponse() = default;
enum class ResponseType {
// DirectResponse encodes MessageType::Reply with success payload
SuccessReply,
// DirectResponse encodes MessageType::Reply with an exception payload
ErrorReply,
// DirectResponse encodes MessageType::Exception
Exception,
};
/**
* Encodes the response via the given Protocol.
* @param metadata the MessageMetadata for the request that generated this response
* @param proto the Protocol to be used for message encoding
* @param buffer the Buffer into which the message should be encoded
* @return ResponseType indicating whether the message is a successful or error reply or an
* exception
*/
virtual ResponseType encode(MessageMetadata& metadata, Protocol& proto,
Buffer::Instance& buffer) const PURE;
};
/**
* Implemented by each Thrift protocol and registered via Registry::registerFactory or the
* convenience class RegisterFactory.
*/
class NamedProtocolConfigFactory : public Config::UntypedFactory {
public:
~NamedProtocolConfigFactory() override = default;
/**
* Create a particular Thrift protocol
* @return ProtocolFactoryCb the protocol
*/
virtual ProtocolPtr createProtocol() PURE;
std::string category() const override { return "envoy.thrift_proxy.protocols"; }
/**
* Convenience method to lookup a factory by type.
* @param ProtocolType the protocol type
* @return NamedProtocolConfigFactory& for the ProtocolType
*/
static NamedProtocolConfigFactory& getFactory(ProtocolType type) {
const std::string& name = ProtocolNames::get().fromType(type);
return Envoy::Config::Utility::getAndCheckFactoryByName<NamedProtocolConfigFactory>(name);
}
};
/**
* ProtocolFactoryBase provides a template for a trivial NamedProtocolConfigFactory.
*/
template <class ProtocolImpl> class ProtocolFactoryBase : public NamedProtocolConfigFactory {
public:
ProtocolPtr createProtocol() override { return std::move(std::make_unique<ProtocolImpl>()); }
std::string name() const override { return name_; }
protected:
ProtocolFactoryBase(const std::string& name) : name_(name) {}
private:
const std::string name_;
};
} // namespace ThriftProxy
} // namespace NetworkFilters
} // namespace Extensions
} // namespace Envoy