// // Copyright 2017 Garrett D'Amore // Copyright 2017 Capitar IT Group BV // // This software is supplied under the terms of the MIT License, a // copy of which should be located in the distribution where this // file was obtained (LICENSE.txt). A copy of the license may also be // found online at https://opensource.org/licenses/MIT. // #include "convey.h" #include "core/nng_impl.h" #include "nng.h" #include "trantest.h" #include #define APPENDSTR(m, s) nng_msg_append(m, s, strlen(s)) #define CHECKSTR(m, s) \ So(nng_msg_len(m) == strlen(s)); \ So(memcmp(nng_msg_body(m), s, strlen(s)) == 0) TestMain("PAIRv1 protocol", { const char *templ = "inproc://pairv1/%u"; char addr[NNG_MAXADDRLEN + 1]; nng_socket s1 = 0; nng_socket c1 = 0; nng_socket c2 = 0; uint64_t tmo; uint32_t v; size_t sz; Reset({ nng_close(s1); nng_close(c1); nng_close(c2); nni_fini(); }); Convey("Given a few sockets", { trantest_next_address(addr, templ); So(nng_pair1_open(&s1) == 0); So(nng_pair1_open(&c1) == 0); So(nng_pair1_open(&c2) == 0); tmo = 300000; So(nng_setopt(s1, NNG_OPT_RCVTIMEO, &tmo, sizeof(tmo)) == 0); So(nng_setopt(c1, NNG_OPT_RCVTIMEO, &tmo, sizeof(tmo)) == 0); So(nng_setopt(c2, NNG_OPT_RCVTIMEO, &tmo, sizeof(tmo)) == 0); Convey("Monogamous cooked mode works", { nng_msg *msg; So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "ALPHA"); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "ALPHA"); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "BETA"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c1, &msg, 0) == 0); CHECKSTR(msg, "BETA"); nng_msg_free(msg); }); Convey("Monogamous mode ignores new conns", { int rv; nng_msg *msg; So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); nng_usleep(100000); So(nng_dial(c2, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "ONE"); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "ONE"); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "TWO"); So(nng_sendmsg(c2, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == NNG_ETIMEDOUT); }); Convey("Cannot set raw mode after connect", { int r = 1; So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); nng_usleep(100000); So(nng_setopt(s1, NNG_OPT_RAW, &r, sizeof(r)) == NNG_ESTATE); So(nng_setopt(c1, NNG_OPT_RAW, &r, sizeof(r)) == NNG_ESTATE); }); Convey("Polyamorous mode is best effort", { int r; int rv; int i; size_t sz; nng_msg *msg; sz = sizeof(r); r = 1; So(nng_setopt( s1, NNG_OPT_POLYAMOROUS, &r, sizeof(r)) == 0); r = 1; So(nng_setopt(s1, NNG_OPT_RCVBUF, &r, sizeof(r)) == 0); So(nng_setopt(s1, NNG_OPT_SNDBUF, &r, sizeof(r)) == 0); So(nng_setopt(c1, NNG_OPT_RCVBUF, &r, sizeof(r)) == 0); tmo = 10000; So(nng_setopt( s1, NNG_OPT_SNDTIMEO, &tmo, sizeof(tmo)) == 0); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); for (i = 0, rv = 0; i < 10; i++) { So(nng_msg_alloc(&msg, 0) == 0); if ((rv = nng_sendmsg(s1, msg, 0)) != 0) { break; } } So(rv == 0); So(i == 10); }); Convey("Monogamous mode exerts backpressure", { int r; int i; int rv; nng_msg *msg; r = 1; So(nng_setopt(s1, NNG_OPT_RCVBUF, &r, sizeof(r)) == 0); So(nng_setopt(s1, NNG_OPT_SNDBUF, &r, sizeof(r)) == 0); So(nng_setopt(c1, NNG_OPT_RCVBUF, &r, sizeof(r)) == 0); tmo = 10000; So(nng_setopt( s1, NNG_OPT_SNDTIMEO, &tmo, sizeof(tmo)) == 0); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); // We choose to allow some buffering. In reality the // buffer size is just 1, and we will fail after 2. for (i = 0, rv = 0; i < 10; i++) { So(nng_msg_alloc(&msg, 0) == 0); if ((rv = nng_sendmsg(s1, msg, 0)) != 0) { nng_msg_free(msg); break; } } So(rv == NNG_ETIMEDOUT); So(i < 10); }); Convey("Cannot set polyamorous mode after connect", { int r = 1; So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); nng_usleep(100000); So(nng_setopt(s1, NNG_OPT_POLYAMOROUS, &r, sizeof(r)) == NNG_ESTATE); So(nng_setopt(c1, NNG_OPT_POLYAMOROUS, &r, sizeof(r)) == NNG_ESTATE); }); Convey("Monogamous raw mode works", { nng_msg *msg; int r = 1; uint32_t hops; So(nng_setopt(s1, NNG_OPT_RAW, &r, sizeof(r)) == 0); So(nng_setopt(c1, NNG_OPT_RAW, &r, sizeof(r)) == 0); So(nng_setopt(c2, NNG_OPT_RAW, &r, sizeof(r)) == 0); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); Convey("Send/recv work", { So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "GAMMA"); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_msg_header_len(msg) == sizeof(uint32_t)); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); So(nng_msg_get_pipe(msg) != 0); CHECKSTR(msg, "GAMMA"); So(nng_msg_header_len(msg) == sizeof(uint32_t)); So(nng_msg_header_trim_u32(msg, &hops) == 0); So(hops == 2); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "EPSILON"); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c1, &msg, 0) == 0); CHECKSTR(msg, "EPSILON"); So(nng_msg_header_len(msg) == sizeof(uint32_t)); So(nng_msg_header_trim_u32(msg, &hops) == 0); So(nng_msg_get_pipe(msg) != 0); So(hops == 2); nng_msg_free(msg); }); Convey("Missing raw header fails", { So(nng_msg_alloc(&msg, 0) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == NNG_ETIMEDOUT); So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_append_u32(msg, 0xFEEDFACE) == 0); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); So(nng_msg_trim_u32(msg, &v) == 0); So(v == 0xFEEDFACE); }); Convey("Reserved bits in raw header", { Convey("Nonzero bits fail", { So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_header_append_u32( msg, 0xDEAD0000) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == NNG_ETIMEDOUT); }); Convey("Zero bits pass", { So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_append_u32( msg, 0xFEEDFACE) == 0); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); So(nng_msg_trim_u32(msg, &v) == 0); So(v == 0xFEEDFACE); }); }); Convey("TTL is honored", { int ttl = 4; sz = sizeof(ttl); So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == 0); ttl = 0; So(nng_getopt(s1, NNG_OPT_MAXTTL, &ttl, &sz) == 0); So(ttl == 4); Convey("Bad TTL bounces", { So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_header_append_u32(msg, 4) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == NNG_ETIMEDOUT); }); Convey("Good TTL passes", { So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_append_u32( msg, 0xFEEDFACE) == 0); So(nng_msg_header_append_u32(msg, 3) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); So(nng_msg_trim_u32(msg, &v) == 0); So(v == 0xFEEDFACE); So(nng_msg_header_trim_u32(msg, &v) == 0); So(v == 4); }); Convey("Large TTL passes", { ttl = 0xff; So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == 0); So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_append_u32(msg, 1234) == 0); So(nng_msg_header_append_u32( msg, 0xfe) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); So(nng_msg_trim_u32(msg, &v) == 0); So(v == 1234); So(nng_msg_header_trim_u32(msg, &v) == 0); So(v == 0xff); }); Convey("Max TTL fails", { ttl = 0xff; So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == 0); So(nng_msg_alloc(&msg, 0) == 0); So(nng_msg_header_append_u32( msg, 0xff) == 0); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == NNG_ETIMEDOUT); }); }); }); Convey("We cannot set insane TTLs", { int ttl; ttl = 0; sz = sizeof(ttl); So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == NNG_EINVAL); ttl = 1000; sz = sizeof(ttl); So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == NNG_EINVAL); sz = 1; ttl = 8; So(nng_setopt(s1, NNG_OPT_MAXTTL, &ttl, sz) == NNG_EINVAL); }); Convey("Polyamorous cooked mode works", { nng_msg *msg; int poly; nng_pipe p1; nng_pipe p2; size_t sz; sz = sizeof(poly); So(nng_getopt(s1, NNG_OPT_POLYAMOROUS, &poly, &sz) == 0); So(poly == 0); poly = 1; So(nng_setopt(s1, NNG_OPT_POLYAMOROUS, &poly, sizeof(poly)) == 0); So(nng_getopt(s1, NNG_OPT_POLYAMOROUS, &poly, &sz) == 0); So(poly == 1); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c2, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "ONE"); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "ONE"); p1 = nng_msg_get_pipe(msg); So(p1 != 0); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "TWO"); So(nng_sendmsg(c2, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "TWO"); p2 = nng_msg_get_pipe(msg); So(p2 != 0); nng_msg_free(msg); So(p1 != p2); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p1); APPENDSTR(msg, "UNO"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c1, &msg, 0) == 0); CHECKSTR(msg, "UNO"); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p2); APPENDSTR(msg, "DOS"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c2, &msg, 0) == 0); CHECKSTR(msg, "DOS"); nng_msg_free(msg); nng_close(c1); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p1); APPENDSTR(msg, "EIN"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c2, &msg, 0) == NNG_ETIMEDOUT); }); Convey("Polyamorous default works", { nng_msg *msg; int poly; nng_pipe p1; size_t sz; poly = 1; So(nng_setopt(s1, NNG_OPT_POLYAMOROUS, &poly, sizeof(poly)) == 0); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); nng_usleep(100000); So(nng_dial(c2, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "YES"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c1, &msg, 0) == 0); CHECKSTR(msg, "YES"); nng_msg_free(msg); nng_close(c1); nng_usleep(10000); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "AGAIN"); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c2, &msg, 0) == 0); CHECKSTR(msg, "AGAIN"); nng_msg_free(msg); }); Convey("Polyamorous raw mode works", { nng_msg *msg; int poly; int raw; uint32_t hops; nng_pipe p1; nng_pipe p2; size_t sz; sz = sizeof(poly); So(nng_getopt(s1, NNG_OPT_POLYAMOROUS, &poly, &sz) == 0); So(poly == 0); poly = 1; So(nng_setopt(s1, NNG_OPT_POLYAMOROUS, &poly, sizeof(poly)) == 0); So(nng_getopt(s1, NNG_OPT_POLYAMOROUS, &poly, &sz) == 0); So(poly == 1); raw = 1; So(nng_setopt(s1, NNG_OPT_RAW, &raw, sizeof(poly)) == 0); raw = 0; sz = sizeof(raw); So(nng_getopt(s1, NNG_OPT_RAW, &raw, &sz) == 0); So(raw == 1); So(nng_listen(s1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c1, addr, NULL, NNG_FLAG_SYNCH) == 0); So(nng_dial(c2, addr, NULL, NNG_FLAG_SYNCH) == 0); Convey("Send/recv works", { So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "ONE"); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "ONE"); p1 = nng_msg_get_pipe(msg); So(p1 != 0); So(nng_msg_header_trim_u32(msg, &hops) == 0); So(hops == 1); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "TWO"); So(nng_sendmsg(c2, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "TWO"); p2 = nng_msg_get_pipe(msg); So(p2 != 0); So(nng_msg_header_trim_u32(msg, &hops) == 0); So(hops == 1); nng_msg_free(msg); So(p1 != p2); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p1); APPENDSTR(msg, "UNO"); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c1, &msg, 0) == 0); CHECKSTR(msg, "UNO"); nng_msg_free(msg); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p2); APPENDSTR(msg, "DOS"); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c2, &msg, 0) == 0); CHECKSTR(msg, "DOS"); nng_msg_free(msg); }); Convey("Closed pipes don't work", { So(nng_msg_alloc(&msg, 0) == 0); APPENDSTR(msg, "ONE"); So(nng_sendmsg(c1, msg, 0) == 0); So(nng_recvmsg(s1, &msg, 0) == 0); CHECKSTR(msg, "ONE"); p1 = nng_msg_get_pipe(msg); So(p1 != 0); nng_msg_free(msg); nng_close(c1); So(nng_msg_alloc(&msg, 0) == 0); nng_msg_set_pipe(msg, p1); APPENDSTR(msg, "EIN"); So(nng_msg_header_append_u32(msg, 1) == 0); So(nng_sendmsg(s1, msg, 0) == 0); So(nng_recvmsg(c2, &msg, 0) == NNG_ETIMEDOUT); }); }); }); })