1 // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
2 // vim: ts=8 sw=2 smarttab
5 #include "include/ceph_assert.h"
6 #include "include/stringify.h"
7 #include "common/Formatter.h"
11 namespace instance_watcher
{
15 class EncodePayloadVisitor
: public boost::static_visitor
<void> {
17 explicit EncodePayloadVisitor(bufferlist
&bl
) : m_bl(bl
) {}
19 template <typename Payload
>
20 inline void operator()(const Payload
&payload
) const {
22 encode(static_cast<uint32_t>(Payload::NOTIFY_OP
), m_bl
);
30 class DecodePayloadVisitor
: public boost::static_visitor
<void> {
32 DecodePayloadVisitor(__u8 version
, bufferlist::const_iterator
&iter
)
33 : m_version(version
), m_iter(iter
) {}
35 template <typename Payload
>
36 inline void operator()(Payload
&payload
) const {
37 payload
.decode(m_version
, m_iter
);
42 bufferlist::const_iterator
&m_iter
;
45 class DumpPayloadVisitor
: public boost::static_visitor
<void> {
47 explicit DumpPayloadVisitor(Formatter
*formatter
) : m_formatter(formatter
) {}
49 template <typename Payload
>
50 inline void operator()(const Payload
&payload
) const {
51 NotifyOp notify_op
= Payload::NOTIFY_OP
;
52 m_formatter
->dump_string("notify_op", stringify(notify_op
));
53 payload
.dump(m_formatter
);
57 ceph::Formatter
*m_formatter
;
60 } // anonymous namespace
62 void PayloadBase::encode(bufferlist
&bl
) const {
64 encode(request_id
, bl
);
67 void PayloadBase::decode(__u8 version
, bufferlist::const_iterator
&iter
) {
69 decode(request_id
, iter
);
72 void PayloadBase::dump(Formatter
*f
) const {
73 f
->dump_unsigned("request_id", request_id
);
76 void ImagePayloadBase::encode(bufferlist
&bl
) const {
78 PayloadBase::encode(bl
);
79 encode(global_image_id
, bl
);
82 void ImagePayloadBase::decode(__u8 version
, bufferlist::const_iterator
&iter
) {
84 PayloadBase::decode(version
, iter
);
85 decode(global_image_id
, iter
);
88 void ImagePayloadBase::dump(Formatter
*f
) const {
90 f
->dump_string("global_image_id", global_image_id
);
93 void PeerImageRemovedPayload::encode(bufferlist
&bl
) const {
95 PayloadBase::encode(bl
);
96 encode(global_image_id
, bl
);
97 encode(peer_mirror_uuid
, bl
);
100 void PeerImageRemovedPayload::decode(__u8 version
, bufferlist::const_iterator
&iter
) {
102 PayloadBase::decode(version
, iter
);
103 decode(global_image_id
, iter
);
104 decode(peer_mirror_uuid
, iter
);
107 void PeerImageRemovedPayload::dump(Formatter
*f
) const {
108 PayloadBase::dump(f
);
109 f
->dump_string("global_image_id", global_image_id
);
110 f
->dump_string("peer_mirror_uuid", peer_mirror_uuid
);
113 void SyncPayloadBase::encode(bufferlist
&bl
) const {
115 PayloadBase::encode(bl
);
119 void SyncPayloadBase::decode(__u8 version
, bufferlist::const_iterator
&iter
) {
121 PayloadBase::decode(version
, iter
);
122 decode(sync_id
, iter
);
125 void SyncPayloadBase::dump(Formatter
*f
) const {
126 PayloadBase::dump(f
);
127 f
->dump_string("sync_id", sync_id
);
130 void UnknownPayload::encode(bufferlist
&bl
) const {
134 void UnknownPayload::decode(__u8 version
, bufferlist::const_iterator
&iter
) {
137 void UnknownPayload::dump(Formatter
*f
) const {
140 void NotifyMessage::encode(bufferlist
& bl
) const {
141 ENCODE_START(2, 2, bl
);
142 boost::apply_visitor(EncodePayloadVisitor(bl
), payload
);
146 void NotifyMessage::decode(bufferlist::const_iterator
& iter
) {
147 DECODE_START(2, iter
);
150 decode(notify_op
, iter
);
152 // select the correct payload variant based upon the encoded op
154 case NOTIFY_OP_IMAGE_ACQUIRE
:
155 payload
= ImageAcquirePayload();
157 case NOTIFY_OP_IMAGE_RELEASE
:
158 payload
= ImageReleasePayload();
160 case NOTIFY_OP_PEER_IMAGE_REMOVED
:
161 payload
= PeerImageRemovedPayload();
163 case NOTIFY_OP_SYNC_REQUEST
:
164 payload
= SyncRequestPayload();
166 case NOTIFY_OP_SYNC_START
:
167 payload
= SyncStartPayload();
170 payload
= UnknownPayload();
174 apply_visitor(DecodePayloadVisitor(struct_v
, iter
), payload
);
178 void NotifyMessage::dump(Formatter
*f
) const {
179 apply_visitor(DumpPayloadVisitor(f
), payload
);
182 void NotifyMessage::generate_test_instances(std::list
<NotifyMessage
*> &o
) {
183 o
.push_back(new NotifyMessage(ImageAcquirePayload()));
184 o
.push_back(new NotifyMessage(ImageAcquirePayload(1, "gid")));
186 o
.push_back(new NotifyMessage(ImageReleasePayload()));
187 o
.push_back(new NotifyMessage(ImageReleasePayload(1, "gid")));
189 o
.push_back(new NotifyMessage(PeerImageRemovedPayload()));
190 o
.push_back(new NotifyMessage(PeerImageRemovedPayload(1, "gid", "uuid")));
192 o
.push_back(new NotifyMessage(SyncRequestPayload()));
193 o
.push_back(new NotifyMessage(SyncRequestPayload(1, "sync_id")));
195 o
.push_back(new NotifyMessage(SyncStartPayload()));
196 o
.push_back(new NotifyMessage(SyncStartPayload(1, "sync_id")));
199 std::ostream
&operator<<(std::ostream
&out
, const NotifyOp
&op
) {
201 case NOTIFY_OP_IMAGE_ACQUIRE
:
202 out
<< "ImageAcquire";
204 case NOTIFY_OP_IMAGE_RELEASE
:
205 out
<< "ImageRelease";
207 case NOTIFY_OP_PEER_IMAGE_REMOVED
:
208 out
<< "PeerImageRemoved";
210 case NOTIFY_OP_SYNC_REQUEST
:
211 out
<< "SyncRequest";
213 case NOTIFY_OP_SYNC_START
:
217 out
<< "Unknown (" << static_cast<uint32_t>(op
) << ")";
223 void NotifyAckPayload::encode(bufferlist
&bl
) const {
225 encode(instance_id
, bl
);
226 encode(request_id
, bl
);
230 void NotifyAckPayload::decode(bufferlist::const_iterator
&iter
) {
232 decode(instance_id
, iter
);
233 decode(request_id
, iter
);
234 decode(ret_val
, iter
);
237 void NotifyAckPayload::dump(Formatter
*f
) const {
238 f
->dump_string("instance_id", instance_id
);
239 f
->dump_unsigned("request_id", request_id
);
240 f
->dump_int("request_id", ret_val
);
243 } // namespace instance_watcher
244 } // namespace mirror