forked from envoyproxy/envoy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhot_restarting_base.h
95 lines (80 loc) · 4.19 KB
/
hot_restarting_base.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
#pragma once
#include <fcntl.h>
#include <sys/un.h>
#include <array>
#include <atomic>
#include <cstdint>
#include <string>
#include "envoy/common/platform.h"
#include "envoy/server/hot_restart.h"
#include "envoy/server/options.h"
#include "envoy/stats/scope.h"
#include "common/common/assert.h"
namespace Envoy {
namespace Server {
/**
* Logic shared by the implementations of both sides of the child<-->parent hot restart protocol:
* domain socket communication, and our ad hoc RPC protocol.
*/
class HotRestartingBase {
protected:
HotRestartingBase(uint64_t base_id) : base_id_(base_id) {}
~HotRestartingBase();
void initDomainSocketAddress(sockaddr_un* address);
sockaddr_un createDomainSocketAddress(uint64_t id, const std::string& role);
void bindDomainSocket(uint64_t id, const std::string& role);
int myDomainSocket() const { return my_domain_socket_; }
// Protocol description:
//
// In each direction between parent<-->child, a series of pairs of:
// A uint64 'length' (bytes in network order),
// followed by 'length' bytes of a serialized HotRestartMessage.
// Each new message must start in a new sendmsg datagram, i.e. 'length' must always start at byte
// 0. Each sendmsg datagram can be up to 4096 bytes (including 'length' if present). When the
// serialized protobuf is longer than 4096-8 bytes, and so cannot fit in just one datagram, it is
// delivered by a series of datagrams. In each of these continuation datagrams, the protobuf data
// starts at byte 0.
//
// There is no mechanism to explicitly pair responses to requests. However, the child initiates
// all exchanges, and blocks until a reply is received, so there is implicit pairing.
void sendHotRestartMessage(sockaddr_un& address, const envoy::HotRestartMessage& proto);
enum class Blocking { Yes, No };
// Receive data, possibly enough to build one of our protocol messages.
// If block is true, blocks until a full protocol message is available.
// If block is false, returns nullptr if we run out of data to receive before a full protocol
// message is available. In either case, the HotRestartingBase may end up buffering some data for
// the next protocol message, even if the function returns a protobuf.
std::unique_ptr<envoy::HotRestartMessage> receiveHotRestartMessage(Blocking block);
bool replyIsExpectedType(const envoy::HotRestartMessage* proto,
envoy::HotRestartMessage::Reply::ReplyCase oneof_type) const;
// Returns a Gauge that tracks hot-restart generation, where every successive
// child increments this number.
static Stats::Gauge& hotRestartGeneration(Stats::Scope& scope);
private:
void getPassedFdIfPresent(envoy::HotRestartMessage* out, msghdr* message);
std::unique_ptr<envoy::HotRestartMessage> parseProtoAndResetState();
void initRecvBufIfNewMessage();
// An int in [0, MaxConcurrentProcesses). As hot restarts happen, each next process gets the
// next of 0,1,2,0,1,...
// A HotRestartingBase's domain socket's name contains its base_id_ value, and so we can use
// this value to determine which domain socket name to treat as our parent, and which to treat as
// our child. (E.g. if we are 2, 1 is parent and 0 is child).
const uint64_t base_id_;
int my_domain_socket_{-1};
// State for the receiving half of the protocol.
//
// When filled, the size in bytes that the in-flight HotRestartMessage should be.
// When empty, we're ready to start receiving a new message (starting with a uint64 'length').
absl::optional<uint64_t> expected_proto_length_;
// How much of the current in-flight message (including both the uint64 'length', plus the proto
// itself) we have received. Once this equals expected_proto_length_ + sizeof(uint64_t), we're
// ready to parse the HotRestartMessage. Should be set to 0 in between messages, to indicate
// readiness for a new message.
uint64_t cur_msg_recvd_bytes_{};
// The first 8 bytes will always be the raw net-order bytes of the current value of
// expected_proto_length_. The protobuf partial data starts at byte 8.
// Should be resized to 0 in between messages, to indicate readiness for a new message.
std::vector<uint8_t> recv_buf_;
};
} // namespace Server
} // namespace Envoy