-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathamqp-proton.html
293 lines (258 loc) · 12.7 KB
/
amqp-proton.html
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
<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>Kevin Boone: Using the Qpid Proton C++ library to understand AMQP</title>
<link rel="shortcut icon" href="https://kevinboone.me/img/favicon.ico">
<meta name="msvalidate.01" content="894212EEB3A89CC8B4E92780079B68E9"/>
<meta name="google-site-verification" content="DXS4cMAJ8VKUgK84_-dl0J1hJK9HQdYU4HtimSr_zLE" />
<meta name="description" content="%%DESC%%">
<meta name="author" content="Kevin Boone">
<meta name="viewport" content="width=device-width; initial-scale=1; maximum-scale=1">
<link rel="stylesheet" href="css/main.css">
</head>
<body>
<div id="myname">
Kevin Boone
</div>
<div id="menu">
<a class="menu_entry" href="index.html">Home</a>
<a class="menu_entry" href="contact.html">Contact</a>
<a class="menu_entry" href="cv.html">CV</a>
<a class="menu_entry" href="software.html">Software</a>
<a class="menu_entry" href="articles.html">Articles</a>
<form id="search_form" method="get" action="https://duckduckgo.com/" target="_blank"><input type="text" name="q" placeholder="Search" size="5" id="search_input" /><button type="submit" id="search_submit">🔍</button><input type="hidden" name="sites" value="kevinboone.me" /><input type="hidden" name="kn" value="1" /></form>
</div>
<div id="content">
<h1>Using the Qpid Proton C++ library to understand AMQP</h1>
<p>
<img class="article-top-image" src="img/antenna.png"
alt="Antenna logo"/>
</p>
<p>
<a href="https://qpid.apache.org/proton"
target="_blank">Apache Qpid Proton</a> is a library in C and C++ for
carrying out messaging operations using AMQP (there's also a very
similar Python library). Very often these messaging operations
will be between some kind of message broker (e.g., Apache Artemis) and
its clients, but AMQP can be used for peer-to-peer communication, as will
be demonstrated in this article.
</p>
<p>
AMQP is not a trivially-straightforward protocol and, for better or worse,
developers sometimes need to understand the low-level details to
make effective use
of it. This understanding is, perhaps, less pressing for
Java developers, who may just use a
JMS wrapper around AMQP, like Qpid-JMS. But even in that case, it helps
to understand the protocol, if only to understand the error messages that
the client runtime produces. For example, it's difficult to know what to
do about the notorious "remote did not respond to a drain request in time"
message, if you don't know what a "drain request is"; and it's difficult
to understand a drain request unless you understand the <i>flow performative</i> and <i>link credit</i>.
</p>
<p>
It's not really my intention to explain any of those things in this article;
my much more modest goal is to demonstrate that a good way to investigate
how AMQP works is to use simple Qpid Proton applications, along with the
built-in packet trace that Proton provides. This approach can give a
useful insight into what goes on at the wire protocol level, and there
are plenty of simple Proton applications to experiment with.
</p>
<p>
To this end, I am maintaining a small collection of simple, fully-documented
Proton C+ examples. The
code is available from <a href="https://github.com/kevinboone/amqp-monitor">my GitHub repository</a> as usual. Instructions on building and running the code
are in the README file in the repository -- you'll need the Proton development
headers and the Proton library and its dependencies. It can be illustrative
to run the examples whilst capturing raw network data, using a tool like
WireShark or tcpdump. However, I won't be showing raw TCP data in
this article.
</p>
<p>
I'm not going to explain here how to obtain
and install the Proton C++ library -- I did that in
<a href="amqp-monitor.html">an earlier article</a>. I'm assuming here that
you have the library and development headers available, and that you
have some familiarity with C++ programming. I'm also using Linux
as my demonstration platform, although Proton is available for other
platforms.
</p>
<h2>The code examples</h2>
<p>
For demonstration purposes, I'll show a simple AMQP client and server.
In the repository I referenced earlier, the source code is
<code>server.cpp</code> for the server, and <code>send_lots.cpp</code>
for the client. The server code just responds to incoming requests to
create a messaging session on an address called <code>foo</code>.
The client sends a number of messages to that address.
</p>
<p>
As supplied, the client and server use port 5672. That's the default for
AMQP, so you may need to ensure that you aren't running any other AMQP-aware
software on the same host as the server process. Also by default, the client
connects to the server on <code>localhost</code>, but that's easy to change
in the source.
</p>
The Proton library primarily works through callbacks. That is, the application
provides a class that implements certain methods, and the library calls
those methods at the appropriate stages of the AMQP conversation.
In my code examples, most of these methods -- even when they are not
relevant to the specific application -- generate some log output, to make
it easier to trace what is going on.
<p>
</p>
<h2>Running the client and server</h2>
<p>
Assuming you've built the samples (which, in most cases, should
amount to running <code>make</code>) you can run both the client and
server with AMQP tracing enabled, like this:
</p>
<pre class="codeblock">
$ PN_TRACE_FRM=1 ./bin/server
</pre>
<p>
And in a different terminal:
</p>
<pre class="codeblock">
$ PN_TRACE_FRM=1 ./bin/send_lots
</pre>
<h2>Examining the output</h2>
<p>
Here is the sample output from <code>send_lots</code> -- I've removed some
duplicate and irrelevant lines, to make it easier to follow.
</p>
<pre class="codeblock">
creating sender
created sender
[0x11fd500]: SASL:FRAME: -> SASL
[0x11fd500]: SASL:FRAME: <- SASL
[0x11fd500]: AMQP:FRAME:0 <- @sasl-mechanisms(64) [sasl-server-mechanisms=@<symbol>[:GSS-SPNEGO, :GSSAPI, :ANONYMOUS]]
[0x11fd500]: AMQP:FRAME:0 -> @sasl-init(65) [mechanism=:ANONYMOUS, initial-response=b"anonymous@fedora"]
[0x11fd500]: AMQP:FRAME:0 <- @sasl-outcome(68) [code=0x0]
[0x11fd500]: AMQP:FRAME: -> AMQP
[0x11fd500]: AMQP:FRAME:0 -> @open(16) [container-id="4f021f4e-e4a8-4a5d-a637-14219a74bca0", hostname="127.0.0.1", channel-max=0x7fff]
[0x11fd500]: AMQP:FRAME:0 -> @begin(17) [next-outgoing-id=0x0, incoming-window=0x7fffffff, outgoing-window=0x7fffffff]
[0x11fd500]: AMQP:FRAME:0 -> @attach(18) [name="00a31a02-cb86-48d9-b20d-437ddc6289ef", handle=0x0, role=false, snd-settle-mode=0x2, rcv-settle-mode=0x0, source=@source(40) [durable=0x0, timeout=0x0, dynamic=false], target=@target(41) [address="foo", durable=0x0, timeout=0x0, dynamic=false], initial-delivery-count=0x0, max-message-size=0x0]
[0x11fd500]: AMQP:FRAME: <- AMQP
[0x11fd500]: AMQP:FRAME:0 <- @open(16) [container-id="38ebe19c-cb6f-4f4c-8cb6-19a3739ccd30", channel-max=0x7fff]
[0x11fd500]: AMQP:FRAME:0 <- @begin(17) [remote-channel=0x0, next-outgoing-id=0x0, incoming-window=0x7fffffff, outgoing-window=0x7fffffff]
[0x11fd500]: AMQP:FRAME:0 <- @attach(18) [name="00a31a02-cb86-48d9-b20d-437ddc6289ef", handle=0x0, role=true, snd-settle-mode=0x2, rcv-settle-mode=0x0, target=@target(41) [address="foo", durable=0x0, timeout=0x0, dynamic=false], initial-delivery-count=0x0, max-message-size=0x0]
[0x11fd500]: AMQP:FRAME:0 <- @flow(19) [next-incoming-id=0x0, incoming-window=0x7fffffff, next-outgoing-id=0x0, outgoing-window=0x7fffffff, handle=0x0, delivery-count=0x0, link-credit=0xa, drain=false]
on_transport_open
on_connection_open
on_session_open
on_sender_open
on_sendable
my link credit is now 10
Sending message
sent messages = 1
Sending message
sent messages = 2
... (more) ...
Sending message
sent messages = 10
[0x11fd500]: AMQP:FRAME:0 -> @transfer(20) [handle=0x0, delivery-id=0x0, delivery-tag=b"\x01\x00\x00\x00\x00\x00\x00\x00", message-format=0x0] (32) \x00SpE\x00Ss\xc0\x06\x01\xa1\x03foo\x00Sw\xa1\x0cHello, world
[0x11fd500]: AMQP:FRAME:0 -> @transfer(20) [handle=0x0, delivery-id=0x1, delivery-tag=b"\x02\x00\x00\x00\x00\x00\x00\x00", message-format=0x0] (32) \x00SpE\x00Ss\xc0\x06\x01\xa1\x03foo\x00Sw\xa1\x0cHello, world
... (more) ...
[0x11fd500]: AMQP:FRAME:0 -> @transfer(20) [handle=0x0, delivery-id=0x9, delivery-tag=b"\x0a\x00\x00\x00\x00\x00\x00\x00", message-format=0x0] (32) \x00SpE\x00Ss\xc0\x06\x01\xa1\x03foo\x00Sw\xa1\x0cHello, world
[0x11fd500]: AMQP:FRAME:0 <- @flow(19) [next-incoming-id=0xa, incoming-window=0x7fffffff, next-outgoing-id=0x0, outgoing-window=0x7fffffff, handle=0x0, delivery-count=0xa, link-credit=0xa, drain=false]
[0x11fd500]: AMQP:FRAME:0 <- @disposition(21) [role=true, first=0x0, last=0x9, settled=true, state=@accepted(36) []]
on_sendable
my link credit is now 10
on_tracker_accept
Closing connection
on_tracker_settle
on_tracker_accept
on_tracker_settle
on_tracker_accept
on_tracker_settle
... (more) ...
on_tracker_accept
on_tracker_settle
[0x11fd500]: AMQP:FRAME:0 -> @close(24) []
[0x11fd500]: IO:FRAME: -> EOS
[0x11fd500]: AMQP:FRAME:0 <- @close(24) []
[0x11fd500]: IO:FRAME: <- EOS
on_connection_close
</pre>
<p>
I'm not going to explain the AMQP protocol in this article, but a few
interesting features are worth some attention.
</p>
<p>
• The conversation starts with SASL authentication, even though the
server does not require credentials. This is, therefore, 'anonymous' authentication, as shown by the token <code>mechanism=:ANONYMOUS</code>.
</p>
<p>
• The client (sender) then sends the <code>@open</code>,
<code>@begin</code>, and <code>@link</code> frames. The tokens
'open', etc., are called 'performatives' in AMQP jargon.
<code>open</code> initiates the communication, and exchanges capabilities
and limits. <code>begin</code> opens a session -- a 'session' in AMQP
terminology is a pair of unidirectional communication channels, one inbound
and one outbound. TCP is, of course, full-duplex; but AMQP could conceivably
be used with other communications infrastructure. Note that the
receiver responds with its own <code>@open</code>,
<code>@begin</code>, and <code>@link</code>, but the sender does not
wait for each response before sending its next frame. This willingness
to proceed without specific confirmation is called 'pipelining' in AMQP.
</p>
<p>
• The sender can not proceed until the receiver has sent a
<code>@flow</code> packet, setting out the amount of link credit it will
allow. Link credit is usually specified in terms of messages, so the
token <code>link-credit=0xa</code> means "you can send ten messages
without checking with me again" (hexadecimal 0x0A = 10).
</p>
<p>
• As the sender proceeds to send messages, the Proton library
generates a bunch of <code>@transfer</code> frames. These contain the
actual message data. Note that the sender does not wait for any
confirmation -- in fact, all ten of the message payloads in this example
will probably be packed into a single TCP packet.
So when does the sender get an acknowledgement from the
receiver?
</p>
<p>
• The receiver sends a bunch of <code>@disposition</code> frames,
indicating what it has done with the messages. <code>state=@accepted</code>
represents a positive acknowledgement -- the message was received and
processed. Of course, what 'processed' amounts to depends on the application.
</p>
<p>
• The sender and receiver both send <code>@close</code> frames, to
indicate that they are finished with the current
connection. They might send <code>@detach</code> and
<code>@end</code> as well, but the specification does not require this.
</p>
<p>
If you're programming with Proton, it's helpful to note how the way that
Proton invokes the application's callback handlers
(like <code>on_sendable()</code>) is interleaved with the actual AMQP frames.
In particular, these callbacks are not necessarily called exactly when
the corresponding AMQP action takes place. Notice, for example, how
a bunch of "accepted" disposition frames are received before the
application's <code>on_tracker_accept()</code> method gets called for the
first time.
</p>
<h2>Closing remarks</h2>
<p>
The AMQP specification does not make for easy reading and, if you're
using high-level AMQP libraries like Qpid-JMS the details might not be
important. For low-level programming -- particularly when programming
Proton server applications -- knowing the fine details really matters.
Using Proton's built-in packet trace, along with at least an outline
understanding of the AMQP protocol, is a good way to debug this kind
of application.
</p>
<p><span class="footer-clearance-para"/></p>
</div>
<div id="footer">
<a href="rss.html"><img src="img/rss.png" width="24px" height="24px"/></a>
Categories: <a href="C-groupindex.html">C</a>, <a href="middleware-groupindex.html">middleware</a>
<span class="last-updated">Last update Apr 23 2023
</span>
</div>
</body>
</html>