Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
L
libzmq
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
libzmq
Commits
3a27be3b
Commit
3a27be3b
authored
Jun 24, 2015
by
Pieter Hintjens
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1448 from jbreams/heartbeats
Add ZMTP connection heartbeats
parents
4b4e00bd
cbb3b176
Hide whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
540 additions
and
7 deletions
+540
-7
Makefile.am
Makefile.am
+4
-2
zmq.h
include/zmq.h
+3
-0
curve_client.cpp
src/curve_client.cpp
+4
-0
curve_server.cpp
src/curve_server.cpp
+4
-0
gssapi_mechanism_base.cpp
src/gssapi_mechanism_base.cpp
+4
-0
options.cpp
src/options.cpp
+46
-1
options.hpp
src/options.hpp
+9
-0
session_base.cpp
src/session_base.cpp
+2
-0
stream_engine.cpp
src/stream_engine.cpp
+120
-4
stream_engine.hpp
src/stream_engine.hpp
+14
-0
CMakeLists.txt
tests/CMakeLists.txt
+1
-0
test_heartbeats.cpp
tests/test_heartbeats.cpp
+329
-0
No files found.
Makefile.am
View file @
3a27be3b
...
@@ -358,7 +358,8 @@ test_apps = \
...
@@ -358,7 +358,8 @@ test_apps = \
tests/test_server_drop_more
\
tests/test_server_drop_more
\
tests/test_client_drop_more
\
tests/test_client_drop_more
\
tests/test_thread_safe
\
tests/test_thread_safe
\
tests/test_socketopt_hwm
tests/test_socketopt_hwm
\
tests/test_heartbeats
tests_test_system_SOURCES
=
tests/test_system.cpp
tests_test_system_SOURCES
=
tests/test_system.cpp
tests_test_system_LDADD
=
src/libzmq.la
tests_test_system_LDADD
=
src/libzmq.la
...
@@ -554,7 +555,8 @@ tests_test_thread_safe_LDADD = src/libzmq.la
...
@@ -554,7 +555,8 @@ tests_test_thread_safe_LDADD = src/libzmq.la
tests_test_socketopt_hwm_SOURCES
=
tests/test_sockopt_hwm.cpp
tests_test_socketopt_hwm_SOURCES
=
tests/test_sockopt_hwm.cpp
tests_test_socketopt_hwm_LDADD
=
src/libzmq.la
tests_test_socketopt_hwm_LDADD
=
src/libzmq.la
tests_test_heartbeats_SOURCES
=
tests/test_heartbeats.cpp
tests_test_heartbeats_LDADD
=
src/libzmq.la
if
!ON_MINGW
if
!ON_MINGW
if
!ON_CYGWIN
if
!ON_CYGWIN
...
...
include/zmq.h
View file @
3a27be3b
...
@@ -316,6 +316,9 @@ ZMQ_EXPORT uint32_t zmq_msg_get_routing_id(zmq_msg_t *msg);
...
@@ -316,6 +316,9 @@ ZMQ_EXPORT uint32_t zmq_msg_get_routing_id(zmq_msg_t *msg);
#define ZMQ_XPUB_WELCOME_MSG 72
#define ZMQ_XPUB_WELCOME_MSG 72
#define ZMQ_STREAM_NOTIFY 73
#define ZMQ_STREAM_NOTIFY 73
#define ZMQ_INVERT_MATCHING 74
#define ZMQ_INVERT_MATCHING 74
#define ZMQ_HEARTBEAT_IVL 75
#define ZMQ_HEARTBEAT_TTL 76
#define ZMQ_HEARTBEAT_TIMEOUT 77
/* Message options */
/* Message options */
#define ZMQ_MORE 1
#define ZMQ_MORE 1
...
...
src/curve_client.cpp
View file @
3a27be3b
...
@@ -130,6 +130,8 @@ int zmq::curve_client_t::encode (msg_t *msg_)
...
@@ -130,6 +130,8 @@ int zmq::curve_client_t::encode (msg_t *msg_)
uint8_t
flags
=
0
;
uint8_t
flags
=
0
;
if
(
msg_
->
flags
()
&
msg_t
::
more
)
if
(
msg_
->
flags
()
&
msg_t
::
more
)
flags
|=
0x01
;
flags
|=
0x01
;
if
(
msg_
->
flags
()
&
msg_t
::
command
)
flags
|=
0x02
;
uint8_t
message_nonce
[
crypto_box_NONCEBYTES
];
uint8_t
message_nonce
[
crypto_box_NONCEBYTES
];
memcpy
(
message_nonce
,
"CurveZMQMESSAGEC"
,
16
);
memcpy
(
message_nonce
,
"CurveZMQMESSAGEC"
,
16
);
...
@@ -223,6 +225,8 @@ int zmq::curve_client_t::decode (msg_t *msg_)
...
@@ -223,6 +225,8 @@ int zmq::curve_client_t::decode (msg_t *msg_)
const
uint8_t
flags
=
message_plaintext
[
crypto_box_ZEROBYTES
];
const
uint8_t
flags
=
message_plaintext
[
crypto_box_ZEROBYTES
];
if
(
flags
&
0x01
)
if
(
flags
&
0x01
)
msg_
->
set_flags
(
msg_t
::
more
);
msg_
->
set_flags
(
msg_t
::
more
);
if
(
flags
&
0x02
)
msg_
->
set_flags
(
msg_t
::
command
);
memcpy
(
msg_
->
data
(),
memcpy
(
msg_
->
data
(),
message_plaintext
+
crypto_box_ZEROBYTES
+
1
,
message_plaintext
+
crypto_box_ZEROBYTES
+
1
,
...
...
src/curve_server.cpp
View file @
3a27be3b
...
@@ -142,6 +142,8 @@ int zmq::curve_server_t::encode (msg_t *msg_)
...
@@ -142,6 +142,8 @@ int zmq::curve_server_t::encode (msg_t *msg_)
uint8_t
flags
=
0
;
uint8_t
flags
=
0
;
if
(
msg_
->
flags
()
&
msg_t
::
more
)
if
(
msg_
->
flags
()
&
msg_t
::
more
)
flags
|=
0x01
;
flags
|=
0x01
;
if
(
msg_
->
flags
()
&
msg_t
::
command
)
flags
|=
0x02
;
uint8_t
*
message_plaintext
=
static_cast
<
uint8_t
*>
(
malloc
(
mlen
));
uint8_t
*
message_plaintext
=
static_cast
<
uint8_t
*>
(
malloc
(
mlen
));
alloc_assert
(
message_plaintext
);
alloc_assert
(
message_plaintext
);
...
@@ -232,6 +234,8 @@ int zmq::curve_server_t::decode (msg_t *msg_)
...
@@ -232,6 +234,8 @@ int zmq::curve_server_t::decode (msg_t *msg_)
const
uint8_t
flags
=
message_plaintext
[
crypto_box_ZEROBYTES
];
const
uint8_t
flags
=
message_plaintext
[
crypto_box_ZEROBYTES
];
if
(
flags
&
0x01
)
if
(
flags
&
0x01
)
msg_
->
set_flags
(
msg_t
::
more
);
msg_
->
set_flags
(
msg_t
::
more
);
if
(
flags
&
0x02
)
msg_
->
set_flags
(
msg_t
::
command
);
memcpy
(
msg_
->
data
(),
memcpy
(
msg_
->
data
(),
message_plaintext
+
crypto_box_ZEROBYTES
+
1
,
message_plaintext
+
crypto_box_ZEROBYTES
+
1
,
...
...
src/gssapi_mechanism_base.cpp
View file @
3a27be3b
...
@@ -80,6 +80,8 @@ int zmq::gssapi_mechanism_base_t::encode_message (msg_t *msg_)
...
@@ -80,6 +80,8 @@ int zmq::gssapi_mechanism_base_t::encode_message (msg_t *msg_)
uint8_t
flags
=
0
;
uint8_t
flags
=
0
;
if
(
msg_
->
flags
()
&
msg_t
::
more
)
if
(
msg_
->
flags
()
&
msg_t
::
more
)
flags
|=
0x01
;
flags
|=
0x01
;
if
(
msg
->
flags
()
&
msg_t
::
command
)
flags
|=
0x02
;
uint8_t
*
plaintext_buffer
=
static_cast
<
uint8_t
*>
(
malloc
(
msg_
->
size
()
+
1
));
uint8_t
*
plaintext_buffer
=
static_cast
<
uint8_t
*>
(
malloc
(
msg_
->
size
()
+
1
));
plaintext_buffer
[
0
]
=
flags
;
plaintext_buffer
[
0
]
=
flags
;
...
@@ -177,6 +179,8 @@ int zmq::gssapi_mechanism_base_t::decode_message (msg_t *msg_)
...
@@ -177,6 +179,8 @@ int zmq::gssapi_mechanism_base_t::decode_message (msg_t *msg_)
const
uint8_t
flags
=
static_cast
<
char
*>
(
plaintext
.
value
)[
0
];
const
uint8_t
flags
=
static_cast
<
char
*>
(
plaintext
.
value
)[
0
];
if
(
flags
&
0x01
)
if
(
flags
&
0x01
)
msg_
->
set_flags
(
msg_t
::
more
);
msg_
->
set_flags
(
msg_t
::
more
);
if
(
flags
&
0x02
)
msg_
->
set_flags
(
msg_t
::
command
);
memcpy
(
msg_
->
data
(),
static_cast
<
char
*>
(
plaintext
.
value
)
+
1
,
plaintext
.
length
-
1
);
memcpy
(
msg_
->
data
(),
static_cast
<
char
*>
(
plaintext
.
value
)
+
1
,
plaintext
.
length
-
1
);
...
...
src/options.cpp
View file @
3a27be3b
...
@@ -69,7 +69,10 @@ zmq::options_t::options_t () :
...
@@ -69,7 +69,10 @@ zmq::options_t::options_t () :
socket_id
(
0
),
socket_id
(
0
),
conflate
(
false
),
conflate
(
false
),
handshake_ivl
(
30000
),
handshake_ivl
(
30000
),
connected
(
false
)
connected
(
false
),
heartbeat_ttl
(
0
),
heartbeat_interval
(
0
),
heartbeat_timeout
(
0
)
{
{
}
}
...
@@ -519,6 +522,27 @@ int zmq::options_t::setsockopt (int option_, const void *optval_,
...
@@ -519,6 +522,27 @@ int zmq::options_t::setsockopt (int option_, const void *optval_,
}
}
break
;
break
;
case
ZMQ_HEARTBEAT_IVL
:
if
(
is_int
&&
value
>=
0
)
{
heartbeat_interval
=
value
;
return
0
;
}
break
;
case
ZMQ_HEARTBEAT_TTL
:
if
(
is_int
&&
value
>=
0
&&
value
<
0xffff
)
{
heartbeat_ttl
=
(
uint16_t
)
value
;
return
0
;
}
break
;
case
ZMQ_HEARTBEAT_TIMEOUT
:
if
(
is_int
&&
value
>=
0
)
{
heartbeat_timeout
=
value
;
return
0
;
}
break
;
default
:
default
:
#if defined (ZMQ_ACT_MILITANT)
#if defined (ZMQ_ACT_MILITANT)
// There are valid scenarios for probing with unknown socket option
// There are valid scenarios for probing with unknown socket option
...
@@ -872,6 +896,27 @@ int zmq::options_t::getsockopt (int option_, void *optval_, size_t *optvallen_)
...
@@ -872,6 +896,27 @@ int zmq::options_t::getsockopt (int option_, void *optval_, size_t *optvallen_)
}
}
break
;
break
;
case
ZMQ_HEARTBEAT_IVL
:
if
(
is_int
)
{
*
value
=
heartbeat_interval
;
return
0
;
}
break
;
case
ZMQ_HEARTBEAT_TTL
:
if
(
is_int
)
{
*
(
uint16_t
*
)
value
=
heartbeat_ttl
;
return
0
;
}
break
;
case
ZMQ_HEARTBEAT_TIMEOUT
:
if
(
is_int
)
{
*
value
=
heartbeat_timeout
;
return
0
;
}
break
;
default
:
default
:
#if defined (ZMQ_ACT_MILITANT)
#if defined (ZMQ_ACT_MILITANT)
malformed
=
false
;
malformed
=
false
;
...
...
src/options.hpp
View file @
3a27be3b
...
@@ -198,6 +198,15 @@ namespace zmq
...
@@ -198,6 +198,15 @@ namespace zmq
int
handshake_ivl
;
int
handshake_ivl
;
bool
connected
;
bool
connected
;
// If remote peer receives a PING message and doesn't receive another
// message within the ttl value, it should close the connection
// (measured in tenths of a second)
uint16_t
heartbeat_ttl
;
// Time in milliseconds between sending heartbeat PING messages.
int
heartbeat_interval
;
// Time in milliseconds to wait for a PING response before disconnecting
int
heartbeat_timeout
;
};
};
}
}
...
...
src/session_base.cpp
View file @
3a27be3b
...
@@ -137,6 +137,8 @@ int zmq::session_base_t::pull_msg (msg_t *msg_)
...
@@ -137,6 +137,8 @@ int zmq::session_base_t::pull_msg (msg_t *msg_)
int
zmq
::
session_base_t
::
push_msg
(
msg_t
*
msg_
)
int
zmq
::
session_base_t
::
push_msg
(
msg_t
*
msg_
)
{
{
if
(
msg_
->
flags
()
&
msg_t
::
command
)
return
0
;
if
(
pipe
&&
pipe
->
write
(
msg_
))
{
if
(
pipe
&&
pipe
->
write
(
msg_
))
{
int
rc
=
msg_
->
init
();
int
rc
=
msg_
->
init
();
errno_assert
(
rc
==
0
);
errno_assert
(
rc
==
0
);
...
...
src/stream_engine.cpp
View file @
3a27be3b
...
@@ -95,6 +95,9 @@ zmq::stream_engine_t::stream_engine_t (fd_t fd_, const options_t &options_,
...
@@ -95,6 +95,9 @@ zmq::stream_engine_t::stream_engine_t (fd_t fd_, const options_t &options_,
input_stopped
(
false
),
input_stopped
(
false
),
output_stopped
(
false
),
output_stopped
(
false
),
has_handshake_timer
(
false
),
has_handshake_timer
(
false
),
has_ttl_timer
(
false
),
has_timeout_timer
(
false
),
has_heartbeat_timer
(
false
),
socket
(
NULL
)
socket
(
NULL
)
{
{
int
rc
=
tx_msg
.
init
();
int
rc
=
tx_msg
.
init
();
...
@@ -250,6 +253,20 @@ void zmq::stream_engine_t::unplug ()
...
@@ -250,6 +253,20 @@ void zmq::stream_engine_t::unplug ()
has_handshake_timer
=
false
;
has_handshake_timer
=
false
;
}
}
if
(
has_ttl_timer
)
{
cancel_timer
(
heartbeat_ttl_timer_id
);
has_ttl_timer
=
false
;
}
if
(
has_timeout_timer
)
{
cancel_timer
(
heartbeat_timeout_timer_id
);
has_timeout_timer
=
false
;
}
if
(
has_heartbeat_timer
)
{
cancel_timer
(
heartbeat_ivl_timer_id
);
has_heartbeat_timer
=
false
;
}
// Cancel all fd subscriptions.
// Cancel all fd subscriptions.
if
(
!
io_error
)
if
(
!
io_error
)
rm_fd
(
handle
);
rm_fd
(
handle
);
...
@@ -686,6 +703,11 @@ bool zmq::stream_engine_t::handshake ()
...
@@ -686,6 +703,11 @@ bool zmq::stream_engine_t::handshake ()
}
}
next_msg
=
&
stream_engine_t
::
next_handshake_command
;
next_msg
=
&
stream_engine_t
::
next_handshake_command
;
process_msg
=
&
stream_engine_t
::
process_handshake_command
;
process_msg
=
&
stream_engine_t
::
process_handshake_command
;
if
(
options
.
heartbeat_interval
>
0
)
{
add_timer
(
options
.
heartbeat_interval
,
heartbeat_ivl_timer_id
);
has_heartbeat_timer
=
true
;
}
}
}
// Start polling for output if necessary.
// Start polling for output if necessary.
...
@@ -883,6 +905,23 @@ int zmq::stream_engine_t::decode_and_push (msg_t *msg_)
...
@@ -883,6 +905,23 @@ int zmq::stream_engine_t::decode_and_push (msg_t *msg_)
if
(
mechanism
->
decode
(
msg_
)
==
-
1
)
if
(
mechanism
->
decode
(
msg_
)
==
-
1
)
return
-
1
;
return
-
1
;
if
(
has_timeout_timer
)
{
has_timeout_timer
=
false
;
cancel_timer
(
heartbeat_timeout_timer_id
);
}
if
(
has_ttl_timer
)
{
has_ttl_timer
=
false
;
cancel_timer
(
heartbeat_ttl_timer_id
);
}
if
(
msg_
->
flags
()
&
msg_t
::
command
)
{
uint8_t
cmd_id
=
*
((
uint8_t
*
)
msg_
->
data
());
if
(
cmd_id
==
4
)
process_heartbeat_message
(
msg_
);
}
if
(
metadata
)
if
(
metadata
)
msg_
->
set_metadata
(
metadata
);
msg_
->
set_metadata
(
metadata
);
if
(
session
->
push_msg
(
msg_
)
==
-
1
)
{
if
(
session
->
push_msg
(
msg_
)
==
-
1
)
{
...
@@ -954,9 +993,86 @@ bool zmq::stream_engine_t::init_properties (properties_t & properties) {
...
@@ -954,9 +993,86 @@ bool zmq::stream_engine_t::init_properties (properties_t & properties) {
void
zmq
::
stream_engine_t
::
timer_event
(
int
id_
)
void
zmq
::
stream_engine_t
::
timer_event
(
int
id_
)
{
{
zmq_assert
(
id_
==
handshake_timer_id
);
if
(
id_
==
handshake_timer_id
)
{
has_handshake_timer
=
false
;
has_handshake_timer
=
false
;
// handshake timer expired before handshake completed, so engine fail
error
(
timeout_error
);
}
else
if
(
id_
==
heartbeat_ivl_timer_id
)
{
next_msg
=
&
stream_engine_t
::
produce_ping_message
;
out_event
();
add_timer
(
options
.
heartbeat_interval
,
heartbeat_ivl_timer_id
);
}
else
if
(
id_
==
heartbeat_ttl_timer_id
)
{
has_ttl_timer
=
false
;
error
(
timeout_error
);
}
else
if
(
id_
==
heartbeat_timeout_timer_id
)
{
has_timeout_timer
=
false
;
error
(
timeout_error
);
}
else
// There are no other valid timer ids!
assert
(
false
);
}
int
zmq
::
stream_engine_t
::
produce_ping_message
(
msg_t
*
msg_
)
{
int
rc
=
0
;
zmq_assert
(
mechanism
!=
NULL
);
// 16-bit TTL + \4PING == 7
msg_
->
init_size
(
7
);
msg_
->
set_flags
(
msg_t
::
command
);
// Copy in the command message
memcpy
(
msg_
->
data
(),
"
\4
PING"
,
5
);
uint16_t
ttl_val
=
htons
(
options
.
heartbeat_ttl
);
memcpy
(((
uint8_t
*
)
msg_
->
data
())
+
5
,
&
ttl_val
,
sizeof
(
ttl_val
));
rc
=
mechanism
->
encode
(
msg_
);
next_msg
=
&
stream_engine_t
::
pull_and_encode
;
if
(
!
has_timeout_timer
&&
options
.
heartbeat_timeout
>
0
)
{
add_timer
(
options
.
heartbeat_timeout
,
heartbeat_timeout_timer_id
);
has_timeout_timer
=
true
;
}
return
rc
;
}
int
zmq
::
stream_engine_t
::
produce_pong_message
(
msg_t
*
msg_
)
{
int
rc
=
0
;
zmq_assert
(
mechanism
!=
NULL
);
msg_
->
init_size
(
5
);
msg_
->
set_flags
(
msg_t
::
command
);
memcpy
(
msg_
->
data
(),
"
\4
PONG"
,
5
);
rc
=
mechanism
->
encode
(
msg_
);
next_msg
=
&
stream_engine_t
::
pull_and_encode
;
return
rc
;
}
int
zmq
::
stream_engine_t
::
process_heartbeat_message
(
msg_t
*
msg_
)
{
if
(
memcmp
(
msg_
->
data
(),
"
\4
PING"
,
5
)
==
0
)
{
uint16_t
remote_heartbeat_ttl
;
// Get the remote heartbeat TTL to setup the timer
memcpy
(
&
remote_heartbeat_ttl
,
(
uint8_t
*
)
msg_
->
data
()
+
5
,
2
);
remote_heartbeat_ttl
=
ntohs
(
remote_heartbeat_ttl
);
// The remote heartbeat is in 10ths of a second
// so we multiply it by 10 to get the timer interval.
remote_heartbeat_ttl
*=
10
;
if
(
!
has_ttl_timer
&&
remote_heartbeat_ttl
>
0
)
{
add_timer
(
remote_heartbeat_ttl
,
heartbeat_ttl_timer_id
);
has_ttl_timer
=
true
;
}
// handshake timer expired before handshake completed, so engine fails
next_msg
=
&
stream_engine_t
::
produce_pong_message
;
error
(
timeout_error
);
out_event
();
}
return
0
;
}
}
src/stream_engine.hpp
View file @
3a27be3b
...
@@ -127,6 +127,10 @@ namespace zmq
...
@@ -127,6 +127,10 @@ namespace zmq
typedef
metadata_t
::
dict_t
properties_t
;
typedef
metadata_t
::
dict_t
properties_t
;
bool
init_properties
(
properties_t
&
properties
);
bool
init_properties
(
properties_t
&
properties
);
int
produce_ping_message
(
msg_t
*
msg_
);
int
process_heartbeat_message
(
msg_t
*
msg_
);
int
produce_pong_message
(
msg_t
*
msg_
);
// Underlying socket.
// Underlying socket.
fd_t
s
;
fd_t
s
;
...
@@ -206,6 +210,16 @@ namespace zmq
...
@@ -206,6 +210,16 @@ namespace zmq
// True is linger timer is running.
// True is linger timer is running.
bool
has_handshake_timer
;
bool
has_handshake_timer
;
// Heartbeat stuff
enum
{
heartbeat_ivl_timer_id
=
0x80
,
heartbeat_timeout_timer_id
=
0x81
,
heartbeat_ttl_timer_id
=
0x82
};
bool
has_ttl_timer
;
bool
has_timeout_timer
;
bool
has_heartbeat_timer
;
// Socket
// Socket
zmq
::
socket_base_t
*
socket
;
zmq
::
socket_base_t
*
socket
;
...
...
tests/CMakeLists.txt
View file @
3a27be3b
...
@@ -49,6 +49,7 @@ set(tests
...
@@ -49,6 +49,7 @@ set(tests
test_thread_safe
test_thread_safe
test_client_server
test_client_server
test_sockopt_hwm
test_sockopt_hwm
test_heartbeats
)
)
if
(
NOT WIN32
)
if
(
NOT WIN32
)
list
(
APPEND tests
list
(
APPEND tests
...
...
tests/test_heartbeats.cpp
0 → 100644
View file @
3a27be3b
/*
Copyright (c) 2007-2015 Contributors as noted in the AUTHORS file
This file is part of 0MQ.
0MQ is free software; you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
0MQ is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "testutil.hpp"
#if defined (ZMQ_HAVE_WINDOWS)
# include <winsock2.h>
# include <ws2tcpip.h>
# include <stdexcept>
# define close closesocket
#else
# include <sys/socket.h>
# include <netinet/in.h>
# include <arpa/inet.h>
# include <unistd.h>
#endif
// Read one event off the monitor socket; return value and address
// by reference, if not null, and event number by value. Returns -1
// in case of error.
static
int
get_monitor_event
(
void
*
monitor
)
{
for
(
int
i
=
0
;
i
<
2
;
i
++
)
{
// First frame in message contains event number and value
zmq_msg_t
msg
;
zmq_msg_init
(
&
msg
);
if
(
zmq_msg_recv
(
&
msg
,
monitor
,
ZMQ_DONTWAIT
)
==
-
1
)
{
msleep
(
150
);
continue
;
// Interruped, presumably
}
assert
(
zmq_msg_more
(
&
msg
));
uint8_t
*
data
=
(
uint8_t
*
)
zmq_msg_data
(
&
msg
);
uint16_t
event
=
*
(
uint16_t
*
)
(
data
);
// Second frame in message contains event address
zmq_msg_init
(
&
msg
);
if
(
zmq_msg_recv
(
&
msg
,
monitor
,
0
)
==
-
1
)
{
return
-
1
;
// Interruped, presumably
}
assert
(
!
zmq_msg_more
(
&
msg
));
return
event
;
}
return
-
1
;
}
static
void
mock_handshake
(
int
fd
)
{
const
uint8_t
zmtp_greeting
[
33
]
=
{
0xff
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0x7f
,
3
,
0
,
'N'
,
'U'
,
'L'
,
'L'
,
0
};
char
buffer
[
128
];
memset
(
buffer
,
0
,
sizeof
(
buffer
));
memcpy
(
buffer
,
zmtp_greeting
,
sizeof
(
zmtp_greeting
));
int
rc
=
send
(
fd
,
buffer
,
64
,
0
);
assert
(
rc
==
64
);
rc
=
recv
(
fd
,
buffer
,
64
,
0
);
assert
(
rc
==
64
);
const
uint8_t
zmtp_ready
[
43
]
=
{
4
,
41
,
5
,
'R'
,
'E'
,
'A'
,
'D'
,
'Y'
,
11
,
'S'
,
'o'
,
'c'
,
'k'
,
'e'
,
't'
,
'-'
,
'T'
,
'y'
,
'p'
,
'e'
,
0
,
0
,
0
,
6
,
'D'
,
'E'
,
'A'
,
'L'
,
'E'
,
'R'
,
8
,
'I'
,
'd'
,
'e'
,
'n'
,
't'
,
'i'
,
't'
,
'y'
,
0
,
0
,
0
,
0
};
memset
(
buffer
,
0
,
sizeof
(
buffer
));
memcpy
(
buffer
,
zmtp_ready
,
43
);
rc
=
send
(
fd
,
buffer
,
43
,
0
);
assert
(
rc
==
43
);
rc
=
recv
(
fd
,
buffer
,
43
,
0
);
assert
(
rc
==
43
);
}
static
void
setup_curve
(
void
*
socket
,
int
is_server
)
{
const
char
*
secret_key
;
const
char
*
public_key
;
const
char
*
server_key
;
if
(
is_server
)
{
secret_key
=
"JTKVSB%%)wK0E.X)V>+}o?pNmC{O&4W4b!Ni{Lh6"
;
public_key
=
"rq:rM>}U?@Lns47E1%kR.o@n%FcmmsL/@{H8]yf7"
;
server_key
=
NULL
;
}
else
{
secret_key
=
"D:)Q[IlAW!ahhC2ac:9*A}h:p?([4%wOTJ%JR%cs"
;
public_key
=
"Yne@$w-vo<fVvi]a<NY6T1ed:M$fCG*[IaLV{hID"
;
server_key
=
"rq:rM>}U?@Lns47E1%kR.o@n%FcmmsL/@{H8]yf7"
;
}
zmq_setsockopt
(
socket
,
ZMQ_CURVE_SECRETKEY
,
secret_key
,
strlen
(
secret_key
));
zmq_setsockopt
(
socket
,
ZMQ_CURVE_PUBLICKEY
,
public_key
,
strlen
(
public_key
));
if
(
is_server
)
zmq_setsockopt
(
socket
,
ZMQ_CURVE_SERVER
,
&
is_server
,
sizeof
(
is_server
));
else
zmq_setsockopt
(
socket
,
ZMQ_CURVE_SERVERKEY
,
server_key
,
strlen
(
server_key
));
}
static
void
prep_server_socket
(
void
*
ctx
,
int
set_heartbeats
,
int
is_curve
,
void
**
server_out
,
void
**
mon_out
)
{
int
rc
;
// We'll be using this socket in raw mode
void
*
server
=
zmq_socket
(
ctx
,
ZMQ_ROUTER
);
assert
(
server
);
int
value
=
0
;
rc
=
zmq_setsockopt
(
server
,
ZMQ_LINGER
,
&
value
,
sizeof
(
value
));
assert
(
rc
==
0
);
if
(
set_heartbeats
)
{
value
=
50
;
rc
=
zmq_setsockopt
(
server
,
ZMQ_HEARTBEAT_IVL
,
&
value
,
sizeof
(
value
));
assert
(
rc
==
0
);
value
=
50
;
rc
=
zmq_setsockopt
(
server
,
ZMQ_HEARTBEAT_TIMEOUT
,
&
value
,
sizeof
(
value
));
assert
(
rc
==
0
);
}
if
(
is_curve
)
setup_curve
(
server
,
1
);
rc
=
zmq_bind
(
server
,
"tcp://127.0.0.1:5556"
);
assert
(
rc
==
0
);
// Create and connect a socket for collecting monitor events on dealer
void
*
server_mon
=
zmq_socket
(
ctx
,
ZMQ_PAIR
);
assert
(
server_mon
);
rc
=
zmq_socket_monitor
(
server
,
"inproc://monitor-dealer"
,
ZMQ_EVENT_CONNECTED
|
ZMQ_EVENT_DISCONNECTED
|
ZMQ_EVENT_ACCEPTED
);
assert
(
rc
==
0
);
// Connect to the inproc endpoint so we'll get events
rc
=
zmq_connect
(
server_mon
,
"inproc://monitor-dealer"
);
assert
(
rc
==
0
);
*
server_out
=
server
;
*
mon_out
=
server_mon
;
}
// This checks for a broken TCP connection (or, in this case a stuck one
// where the peer never responds to PINGS). There should be an accepted event
// then a disconnect event.
static
void
test_heartbeat_timeout
(
void
)
{
int
rc
;
// Set up our context and sockets
void
*
ctx
=
zmq_ctx_new
();
assert
(
ctx
);
void
*
server
,
*
server_mon
;
prep_server_socket
(
ctx
,
1
,
0
,
&
server
,
&
server_mon
);
struct
sockaddr_in
ip4addr
;
int
s
;
ip4addr
.
sin_family
=
AF_INET
;
ip4addr
.
sin_port
=
htons
(
5556
);
inet_pton
(
AF_INET
,
"127.0.0.1"
,
&
ip4addr
.
sin_addr
);
s
=
socket
(
AF_INET
,
SOCK_STREAM
,
IPPROTO_TCP
);
rc
=
connect
(
s
,
(
struct
sockaddr
*
)
&
ip4addr
,
sizeof
ip4addr
);
assert
(
rc
>
-
1
);
// Mock a ZMTP 3 client so we can forcibly time out a connection
mock_handshake
(
s
);
// By now everything should report as connected
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
ZMQ_EVENT_ACCEPTED
);
// We should have been disconnected
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
ZMQ_EVENT_DISCONNECTED
);
close
(
s
);
rc
=
zmq_close
(
server
);
assert
(
rc
==
0
);
rc
=
zmq_close
(
server_mon
);
assert
(
rc
==
0
);
rc
=
zmq_ctx_term
(
ctx
);
assert
(
rc
==
0
);
}
// This checks that peers respect the TTL value in ping messages
// We set up a mock ZMTP 3 client and send a ping message with a TLL
// to a server that is not doing any heartbeating. Then we sleep,
// if the server disconnects the client, then we know the TTL did
// its thing correctly.
static
void
test_heartbeat_ttl
(
void
)
{
int
rc
;
// Set up our context and sockets
void
*
ctx
=
zmq_ctx_new
();
assert
(
ctx
);
void
*
server
,
*
server_mon
;
prep_server_socket
(
ctx
,
0
,
0
,
&
server
,
&
server_mon
);
struct
sockaddr_in
ip4addr
;
int
s
;
ip4addr
.
sin_family
=
AF_INET
;
ip4addr
.
sin_port
=
htons
(
5556
);
inet_pton
(
AF_INET
,
"127.0.0.1"
,
&
ip4addr
.
sin_addr
);
s
=
socket
(
AF_INET
,
SOCK_STREAM
,
IPPROTO_TCP
);
rc
=
connect
(
s
,
(
struct
sockaddr
*
)
&
ip4addr
,
sizeof
ip4addr
);
assert
(
rc
>
-
1
);
// Mock a ZMTP 3 client so we can forcibly time out a connection
mock_handshake
(
s
);
// By now everything should report as connected
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
ZMQ_EVENT_ACCEPTED
);
// This is a ping message with a 0.5 second TTL.
uint8_t
ping_message
[]
=
{
0x4
,
// This specifies that this is a command message
0x7
,
// The total payload length is 8 bytes
0x4
,
'P'
,
'I'
,
'N'
,
'G'
,
// The command name
0
,
10
// This is a network-order 16-bit TTL value
};
rc
=
send
(
s
,
ping_message
,
sizeof
(
ping_message
),
0
);
assert
(
rc
==
sizeof
(
ping_message
));
uint8_t
pong_buffer
[
8
]
=
{
0
};
rc
=
recv
(
s
,
pong_buffer
,
7
,
0
);
assert
(
rc
==
7
&&
memcmp
(
pong_buffer
,
"
\4\5\4
PONG"
,
7
)
==
0
);
// We should have been disconnected
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
ZMQ_EVENT_DISCONNECTED
);
close
(
s
);
rc
=
zmq_close
(
server
);
assert
(
rc
==
0
);
rc
=
zmq_close
(
server_mon
);
assert
(
rc
==
0
);
rc
=
zmq_ctx_term
(
ctx
);
assert
(
rc
==
0
);
}
// This checks for normal operation - that is pings and pongs being
// exchanged normally. There should be an accepted event on the server,
// and then no event afterwards.
static
void
test_heartbeat_notimeout
(
int
is_curve
)
{
int
rc
;
// Set up our context and sockets
void
*
ctx
=
zmq_ctx_new
();
assert
(
ctx
);
void
*
server
,
*
server_mon
;
prep_server_socket
(
ctx
,
1
,
is_curve
,
&
server
,
&
server_mon
);
void
*
client
=
zmq_socket
(
ctx
,
ZMQ_DEALER
);
if
(
is_curve
)
setup_curve
(
client
,
0
);
rc
=
zmq_connect
(
client
,
"tcp://127.0.0.1:5556"
);
// Give it a sec to connect and handshake
msleep
(
100
);
// By now everything should report as connected
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
ZMQ_EVENT_ACCEPTED
);
// We should still be connected because pings and pongs are happenin'
rc
=
get_monitor_event
(
server_mon
);
assert
(
rc
==
-
1
);
rc
=
zmq_close
(
client
);
assert
(
rc
==
0
);
rc
=
zmq_close
(
server
);
assert
(
rc
==
0
);
rc
=
zmq_close
(
server_mon
);
assert
(
rc
==
0
);
rc
=
zmq_ctx_term
(
ctx
);
assert
(
rc
==
0
);
}
int
main
(
void
)
{
setup_test_environment
();
test_heartbeat_timeout
();
test_heartbeat_ttl
();
// Run this test without curve
test_heartbeat_notimeout
(
0
);
// Then rerun it with curve
test_heartbeat_notimeout
(
1
);
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment