| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719 | 
							- /*
 
-  * Wslay - The WebSocket Library
 
-  *
 
-  * Copyright (c) 2011, 2012 Tatsuhiro Tsujikawa
 
-  *
 
-  * Permission is hereby granted, free of charge, to any person obtaining
 
-  * a copy of this software and associated documentation files (the
 
-  * "Software"), to deal in the Software without restriction, including
 
-  * without limitation the rights to use, copy, modify, merge, publish,
 
-  * distribute, sublicense, and/or sell copies of the Software, and to
 
-  * permit persons to whom the Software is furnished to do so, subject to
 
-  * the following conditions:
 
-  *
 
-  * The above copyright notice and this permission notice shall be
 
-  * included in all copies or substantial portions of the Software.
 
-  *
 
-  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 
-  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 
-  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 
-  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 
-  * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 
-  * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 
-  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 
-  */
 
- #include "wslay_frame_test.h"
 
- #include <assert.h>
 
- #include <CUnit/CUnit.h>
 
- #include "wslay_frame.h"
 
- void test_wslay_frame_context_init(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks;
 
-   int user_data;
 
-   CU_ASSERT_FALSE(wslay_frame_context_init(&ctx, &callbacks, &user_data));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- struct scripted_data_feed {
 
-   uint8_t data[8192];
 
-   uint8_t *datamark;
 
-   uint8_t *datalimit;
 
-   size_t feedseq[8192];
 
-   size_t seqidx;
 
- };
 
- static void scripted_data_feed_init(struct scripted_data_feed *df,
 
-                                     uint8_t *data, size_t data_length) {
 
-   memset(df, 0, sizeof(struct scripted_data_feed));
 
-   if (data_length) {
 
-     memcpy(df->data, data, data_length);
 
-   }
 
-   df->datamark = df->data;
 
-   df->datalimit = df->data + data_length;
 
-   df->feedseq[0] = data_length;
 
- }
 
- static ssize_t scripted_recv_callback(uint8_t *data, size_t len, int flags,
 
-                                       void *user_data) {
 
-   struct scripted_data_feed *df = (struct scripted_data_feed *)user_data;
 
-   size_t wlen = df->feedseq[df->seqidx] > len ? len : df->feedseq[df->seqidx];
 
-   memcpy(data, df->datamark, wlen);
 
-   df->datamark += wlen;
 
-   if (wlen <= len) {
 
-     ++df->seqidx;
 
-   } else {
 
-     df->feedseq[df->seqidx] -= wlen;
 
-   }
 
-   return wlen;
 
- }
 
- static ssize_t scripted_send_callback(const uint8_t *data, size_t len,
 
-                                       int flags, void *user_data) {
 
-   struct scripted_data_feed *df = (struct scripted_data_feed *)user_data;
 
-   size_t wlen = df->feedseq[df->seqidx] > len ? len : df->feedseq[df->seqidx];
 
-   memcpy(df->datamark, data, wlen);
 
-   df->datamark += wlen;
 
-   if (wlen <= len) {
 
-     ++df->seqidx;
 
-   } else {
 
-     df->feedseq[df->seqidx] -= wlen;
 
-   }
 
-   return wlen;
 
- }
 
- void test_wslay_frame_recv(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT(5 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(1, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(0x1, iocb.opcode);
 
-   CU_ASSERT_EQUAL(5, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(1, iocb.mask);
 
-   CU_ASSERT_EQUAL(5, iocb.data_length);
 
-   CU_ASSERT(memcmp("Hello", iocb.data, iocb.data_length) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_1byte(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   size_t i;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   for (i = 0; i < sizeof(msg); ++i) {
 
-     df.feedseq[i] = 1;
 
-   }
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   for (i = 0; i < 4; ++i) {
 
-     CU_ASSERT(WSLAY_ERR_WANT_READ == wslay_frame_recv(ctx, &iocb));
 
-   }
 
-   for (i = 0; i < 5; ++i) {
 
-     CU_ASSERT(1 == wslay_frame_recv(ctx, &iocb));
 
-     CU_ASSERT_EQUAL(1, iocb.fin);
 
-     CU_ASSERT_EQUAL(0, iocb.rsv);
 
-     CU_ASSERT_EQUAL(0x1, iocb.opcode);
 
-     CU_ASSERT_EQUAL(5, iocb.payload_length);
 
-     CU_ASSERT_EQUAL(1, iocb.mask);
 
-     CU_ASSERT_EQUAL(1, iocb.data_length);
 
-     CU_ASSERT_EQUAL(msg[6 + i] ^ msg[2 + i % 4], iocb.data[0]);
 
-   }
 
-   CU_ASSERT(WSLAY_ERR_WANT_READ == wslay_frame_recv(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_fragmented(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg[] = {0x01, 0x03, 0x48, 0x65, 0x6c, /* "Hel" */
 
-                    0x80, 0x02, 0x6c, 0x6f};      /* "lo" */
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = 5;
 
-   df.feedseq[1] = 4;
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT(3 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(0, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_TEXT_FRAME, iocb.opcode);
 
-   CU_ASSERT_EQUAL(3, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(3, iocb.data_length);
 
-   CU_ASSERT(memcmp("Hel", iocb.data, iocb.data_length) == 0);
 
-   CU_ASSERT(2 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(1, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_CONTINUATION_FRAME, iocb.opcode);
 
-   CU_ASSERT_EQUAL(2, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(2, iocb.data_length);
 
-   CU_ASSERT(memcmp("lo", iocb.data, iocb.data_length) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_interleaved_ctrl_frame(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg[] = {0x01, 0x03, 0x48, 0x65, 0x6c, /* "Hel" */
 
-                    /* ping with "Hello" */
 
-                    0x89, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f, 0x80, 0x02, 0x6c,
 
-                    0x6f}; /* "lo" */
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = 5;
 
-   df.feedseq[1] = 7, df.feedseq[2] = 4;
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT(3 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(0, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_TEXT_FRAME, iocb.opcode);
 
-   CU_ASSERT_EQUAL(3, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(3, iocb.data_length);
 
-   CU_ASSERT(memcmp("Hel", iocb.data, iocb.data_length) == 0);
 
-   CU_ASSERT(5 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(1, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_PING, iocb.opcode);
 
-   CU_ASSERT_EQUAL(5, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(5, iocb.data_length);
 
-   CU_ASSERT(memcmp("Hello", iocb.data, iocb.data_length) == 0);
 
-   CU_ASSERT(2 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(1, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_CONTINUATION_FRAME, iocb.opcode);
 
-   CU_ASSERT_EQUAL(2, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(2, iocb.data_length);
 
-   CU_ASSERT(memcmp("lo", iocb.data, iocb.data_length) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_zero_payloadlen(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg[] = {0x81, 0x00}; /* "" */
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = 2;
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT(0 == wslay_frame_recv(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(1, iocb.fin);
 
-   CU_ASSERT_EQUAL(0, iocb.rsv);
 
-   CU_ASSERT_EQUAL(WSLAY_TEXT_FRAME, iocb.opcode);
 
-   CU_ASSERT_EQUAL(0, iocb.payload_length);
 
-   CU_ASSERT_EQUAL(0, iocb.mask);
 
-   CU_ASSERT_EQUAL(0, iocb.data_length);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t msg[] = {0x81, 0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = sizeof(msg);
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_PROTO, wslay_frame_recv(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_ctrl_frame_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t msg[] = {0x88, 0x7e};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = sizeof(msg);
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_PROTO, wslay_frame_recv(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_minimum_ext_payload16(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t msg[] = {0x81, 0x7e, 0x00, 0x7d};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = sizeof(msg);
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_PROTO, wslay_frame_recv(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_recv_minimum_ext_payload64(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, scripted_recv_callback, NULL};
 
-   struct scripted_data_feed df;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t msg[] = {0x81, 0x7f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff};
 
-   scripted_data_feed_init(&df, msg, sizeof(msg));
 
-   df.feedseq[0] = sizeof(msg);
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_PROTO, wslay_frame_recv(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- struct accumulator {
 
-   uint8_t buf[4096];
 
-   size_t length;
 
- };
 
- static ssize_t accumulator_send_callback(const uint8_t *buf, size_t len,
 
-                                          int flags, void *user_data) {
 
-   struct accumulator *acc = (struct accumulator *)user_data;
 
-   assert(acc->length + len < sizeof(acc->buf));
 
-   memcpy(acc->buf + acc->length, buf, len);
 
-   acc->length += len;
 
-   return len;
 
- }
 
- static int static_genmask_callback(uint8_t *buf, size_t len, void *user_data) {
 
-   static const uint8_t makskey[] = {0x37u, 0xfau, 0x21u, 0x3du};
 
-   memcpy(buf, makskey, 4);
 
-   return 0;
 
- }
 
- void test_wslay_frame_send(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {accumulator_send_callback, NULL,
 
-                                             static_genmask_callback};
 
-   struct accumulator acc;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   wslay_frame_context_init(&ctx, &callbacks, &acc);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   acc.length = 0;
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 1;
 
-   iocb.payload_length = 5;
 
-   iocb.data = (const uint8_t *)"Hello";
 
-   iocb.data_length = 5;
 
-   CU_ASSERT(5 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg), acc.length);
 
-   CU_ASSERT(memcmp(msg, acc.buf, sizeof(msg)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_fragmented(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {accumulator_send_callback, NULL,
 
-                                             static_genmask_callback};
 
-   struct accumulator acc;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg1[] = {0x01, 0x03, 0x48, 0x65, 0x6c}; /* "Hel" */
 
-   uint8_t msg2[] = {0x80, 0x02, 0x6c, 0x6f};       /* "lo" */
 
-   wslay_frame_context_init(&ctx, &callbacks, &acc);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   acc.length = 0;
 
-   iocb.fin = 0;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 3;
 
-   iocb.data = (const uint8_t *)"Hel";
 
-   iocb.data_length = 3;
 
-   CU_ASSERT(3 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg1), acc.length);
 
-   CU_ASSERT(memcmp(msg1, acc.buf, sizeof(msg1)) == 0);
 
-   acc.length = 0;
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_CONTINUATION_FRAME;
 
-   iocb.payload_length = 2;
 
-   iocb.data = (const uint8_t *)"lo";
 
-   iocb.data_length = 2;
 
-   CU_ASSERT(2 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg2), acc.length);
 
-   CU_ASSERT(memcmp(msg2, acc.buf, sizeof(msg2)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_interleaved_ctrl_frame(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {accumulator_send_callback, NULL,
 
-                                             static_genmask_callback};
 
-   struct accumulator acc;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   /* text with "Hel", with fin = 0 */
 
-   uint8_t msg1[] = {0x01, 0x03, 0x48, 0x65, 0x6c};
 
-   /* ping with "Hello" */
 
-   uint8_t msg2[] = {0x89, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f};
 
-   /* text with "lo", continuation frame for msg1, with fin = 1 */
 
-   uint8_t msg3[] = {0x80, 0x02, 0x6c, 0x6f};
 
-   wslay_frame_context_init(&ctx, &callbacks, &acc);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   acc.length = 0;
 
-   iocb.fin = 0;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 3;
 
-   iocb.data = (const uint8_t *)"Hel";
 
-   iocb.data_length = 3;
 
-   CU_ASSERT(3 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg1), acc.length);
 
-   CU_ASSERT(memcmp(msg1, acc.buf, sizeof(msg1)) == 0);
 
-   acc.length = 0;
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_PING;
 
-   iocb.payload_length = 5;
 
-   iocb.data = (const uint8_t *)"Hello";
 
-   iocb.data_length = 5;
 
-   CU_ASSERT(5 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg2), acc.length);
 
-   CU_ASSERT(memcmp(msg2, acc.buf, sizeof(msg2)) == 0);
 
-   acc.length = 0;
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_CONTINUATION_FRAME;
 
-   iocb.payload_length = 2;
 
-   iocb.data = (const uint8_t *)"lo";
 
-   iocb.data_length = 2;
 
-   CU_ASSERT(2 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg3), acc.length);
 
-   CU_ASSERT(memcmp(msg3, acc.buf, sizeof(msg3)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_1byte_masked(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {scripted_send_callback, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   uint8_t hello[] = "Hello";
 
-   struct scripted_data_feed df;
 
-   size_t i;
 
-   scripted_data_feed_init(&df, NULL, 0);
 
-   for (i = 0; i < sizeof(msg); ++i) {
 
-     df.feedseq[i] = 1;
 
-   }
 
-   wslay_frame_context_init(&ctx, &callbacks, &df);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 1;
 
-   iocb.payload_length = 5;
 
-   iocb.data = hello;
 
-   iocb.data_length = sizeof(hello) - 1;
 
-   for (i = 0; i < 5; ++i) {
 
-     CU_ASSERT_EQUAL(WSLAY_ERR_WANT_WRITE, wslay_frame_send(ctx, &iocb));
 
-   }
 
-   CU_ASSERT_EQUAL(5, wslay_frame_send(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_zero_payloadlen(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {accumulator_send_callback, NULL,
 
-                                             static_genmask_callback};
 
-   struct accumulator acc;
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg[] = {0x81, 0x00}; /* "" */
 
-   acc.length = 0;
 
-   wslay_frame_context_init(&ctx, &callbacks, &acc);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 0;
 
-   iocb.data_length = 0;
 
-   CU_ASSERT(0 == wslay_frame_send(ctx, &iocb));
 
-   CU_ASSERT_EQUAL(sizeof(msg), acc.length);
 
-   CU_ASSERT(memcmp(msg, acc.buf, sizeof(msg)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks;
 
-   struct wslay_frame_iocb iocb;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = UINT64_MAX;
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_INVALID_ARGUMENT, wslay_frame_send(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_send_ctrl_frame_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks;
 
-   struct wslay_frame_iocb iocb;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_PING;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 1024;
 
-   CU_ASSERT_EQUAL(WSLAY_ERR_INVALID_ARGUMENT, wslay_frame_send(ctx, &iocb));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 1;
 
-   iocb.payload_length = 5;
 
-   iocb.data = (const uint8_t *)"Hello";
 
-   iocb.data_length = 5;
 
-   CU_ASSERT(11 ==
 
-             wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(iocb.data_length == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg, buf, sizeof(msg)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_fragmented(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg1[] = {0x01, 0x03, 0x48, 0x65, 0x6c}; /* "Hel" */
 
-   uint8_t msg2[] = {0x80, 0x02, 0x6c, 0x6f};       /* "lo" */
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 0;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 3;
 
-   iocb.data = (const uint8_t *)"Hel";
 
-   iocb.data_length = 3;
 
-   CU_ASSERT(5 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(3 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg1, buf, sizeof(msg1)) == 0);
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_CONTINUATION_FRAME;
 
-   iocb.payload_length = 2;
 
-   iocb.data = (const uint8_t *)"lo";
 
-   iocb.data_length = 2;
 
-   CU_ASSERT(4 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(2 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg2, buf, sizeof(msg2)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_interleaved_ctrl_frame(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   /* text with "Hel", with fin = 0 */
 
-   uint8_t msg1[] = {0x01, 0x03, 0x48, 0x65, 0x6c};
 
-   /* ping with "Hello" */
 
-   uint8_t msg2[] = {0x89, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f};
 
-   /* text with "lo", continuation frame for msg1, with fin = 1 */
 
-   uint8_t msg3[] = {0x80, 0x02, 0x6c, 0x6f};
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 0;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 3;
 
-   iocb.data = (const uint8_t *)"Hel";
 
-   iocb.data_length = 3;
 
-   CU_ASSERT(5 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(3 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg1, buf, sizeof(msg1)) == 0);
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_PING;
 
-   iocb.payload_length = 5;
 
-   iocb.data = (const uint8_t *)"Hello";
 
-   iocb.data_length = 5;
 
-   CU_ASSERT(7 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(5 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg2, buf, sizeof(msg2)) == 0);
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_CONTINUATION_FRAME;
 
-   iocb.payload_length = 2;
 
-   iocb.data = (const uint8_t *)"lo";
 
-   iocb.data_length = 2;
 
-   CU_ASSERT(4 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(2 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg3, buf, sizeof(msg3)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_1byte_masked(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Masked text frame containing "Hello" */
 
-   uint8_t msg[] = {0x81u, 0x85u, 0x37u, 0xfau, 0x21u, 0x3du,
 
-                    0x7fu, 0x9fu, 0x4du, 0x51u, 0x58u};
 
-   uint8_t hello[] = "Hello";
 
-   size_t i;
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 1;
 
-   iocb.payload_length = 5;
 
-   iocb.data = hello;
 
-   iocb.data_length = sizeof(hello) - 1;
 
-   CU_ASSERT_EQUAL(6, wslay_frame_write(ctx, &iocb, buf, 6, &wpayloadlen));
 
-   CU_ASSERT(0 == wpayloadlen);
 
-   for (i = 0; i < 5; ++i) {
 
-     CU_ASSERT_EQUAL(
 
-         1, wslay_frame_write(ctx, &iocb, buf + 6 + i, 1, &wpayloadlen));
 
-     CU_ASSERT(1 == wpayloadlen);
 
-     ++iocb.data;
 
-     --iocb.data_length;
 
-   }
 
-   CU_ASSERT(memcmp(msg, buf, sizeof(msg)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_zero_payloadlen(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks = {NULL, NULL,
 
-                                             static_genmask_callback};
 
-   struct wslay_frame_iocb iocb;
 
-   /* Unmasked message */
 
-   uint8_t msg[] = {0x81, 0x00}; /* "" */
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 0;
 
-   iocb.data_length = 0;
 
-   CU_ASSERT(2 == wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   CU_ASSERT(0 == wpayloadlen);
 
-   CU_ASSERT(memcmp(msg, buf, sizeof(msg)) == 0);
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_TEXT_FRAME;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = UINT64_MAX;
 
-   CU_ASSERT_EQUAL(
 
-       WSLAY_ERR_INVALID_ARGUMENT,
 
-       wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   wslay_frame_context_free(ctx);
 
- }
 
- void test_wslay_frame_write_ctrl_frame_too_large_payload(void) {
 
-   wslay_frame_context_ptr ctx;
 
-   struct wslay_frame_callbacks callbacks;
 
-   struct wslay_frame_iocb iocb;
 
-   uint8_t buf[1024];
 
-   size_t wpayloadlen;
 
-   wslay_frame_context_init(&ctx, &callbacks, NULL);
 
-   memset(&iocb, 0, sizeof(iocb));
 
-   iocb.fin = 1;
 
-   iocb.opcode = WSLAY_PING;
 
-   iocb.mask = 0;
 
-   iocb.payload_length = 1024;
 
-   CU_ASSERT_EQUAL(
 
-       WSLAY_ERR_INVALID_ARGUMENT,
 
-       wslay_frame_write(ctx, &iocb, buf, sizeof(buf), &wpayloadlen));
 
-   wslay_frame_context_free(ctx);
 
- }
 
 
  |