Exemplo n.º 1
0
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);
}
Exemplo n.º 2
0
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);
}
Exemplo n.º 3
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);
}
Exemplo n.º 4
0
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);
}
Exemplo n.º 5
0
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);
}
Exemplo n.º 6
0
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);
}
Exemplo n.º 7
0
int wslay_event_send(wslay_event_context_ptr ctx)
{
	struct wslay_frame_iocb iocb;
	ssize_t r;

	while (ctx->write_enabled && (!wslay_queue_empty(ctx->send_queue) || !wslay_queue_empty(ctx->send_ctrl_queue) || ctx->omsg)) {
		if (!ctx->omsg) {
			if (wslay_queue_empty(ctx->send_ctrl_queue)) {
				ctx->omsg = wslay_queue_top(ctx->send_queue);
				wslay_queue_pop(ctx->send_queue);
			} else {
				ctx->omsg = wslay_event_send_ctrl_queue_pop(ctx);
				if (ctx->omsg == NULL) {
					break;
				}
			}
			if (ctx->omsg->type == WSLAY_NON_FRAGMENTED) {
				wslay_event_on_non_fragmented_msg_popped(ctx);
			}
		} else if (!wslay_is_ctrl_frame(ctx->omsg->opcode) && ctx->frame_ctx->ostate == PREP_HEADER && !wslay_queue_empty(ctx->send_ctrl_queue)) {
			if ((r = wslay_queue_push_front(ctx->send_queue, ctx->omsg)) != 0) {
				ctx->write_enabled = 0;
				return r;
			}
			ctx->omsg = wslay_event_send_ctrl_queue_pop(ctx);
			if (ctx->omsg == NULL) {
				break;
			}
			/* ctrl message has WSLAY_NON_FRAGMENTED */
			wslay_event_on_non_fragmented_msg_popped(ctx);
		}
		if (ctx->omsg->type == WSLAY_NON_FRAGMENTED) {
			memset(&iocb, 0, sizeof(iocb));
			iocb.fin = 1;
			iocb.opcode = ctx->omsg->opcode;
			iocb.rsv = ctx->omsg->rsv;
			iocb.mask = ctx->server ^ 1;
			iocb.data = ctx->omsg->data + ctx->opayloadoff;
			iocb.data_length = ctx->opayloadlen - ctx->opayloadoff;
			iocb.payload_length = ctx->opayloadlen;
			r = wslay_frame_send(ctx->frame_ctx, &iocb);
			if (r >= 0) {
				ctx->opayloadoff += r;
				if (ctx->opayloadoff == ctx->opayloadlen) {
					--ctx->queued_msg_count;
					ctx->queued_msg_length -= ctx->omsg->data_length;
					if (ctx->omsg->opcode == WSLAY_CONNECTION_CLOSE) {
						uint16_t status_code = 0;
						ctx->write_enabled = 0;
						ctx->close_status |= WSLAY_CLOSE_SENT;
						if (ctx->omsg->data_length >= 2) {
							memcpy(&status_code, ctx->omsg->data, 2);
							status_code = ntohs(status_code);
						}
						ctx->status_code_sent = status_code == 0 ? WSLAY_CODE_NO_STATUS_RCVD : status_code;
					}
					wslay_event_omsg_free(ctx->omsg);
					ctx->omsg = NULL;
				} else {
					break;
				}
			} else {
				if (r != WSLAY_ERR_WANT_WRITE || (ctx->error != WSLAY_ERR_WOULDBLOCK && ctx->error != 0)) {
					ctx->write_enabled = 0;
					return WSLAY_ERR_CALLBACK_FAILURE;
				}
				break;
			}
		} else {
			if (ctx->omsg->fin == 0 && ctx->obuflimit == ctx->obufmark) {
				int eof = 0;
				r = ctx->omsg->read_callback(ctx, ctx->obuf, sizeof(ctx->obuf), &ctx->omsg->source, &eof, ctx->user_data);
				if (r == 0) {
					break;
				} else if (r < 0) {
					ctx->write_enabled = 0;
					return WSLAY_ERR_CALLBACK_FAILURE;
				}
				ctx->obuflimit = ctx->obuf + r;
				if (eof) {
					ctx->omsg->fin = 1;
				}
				ctx->opayloadlen = (uint64_t)r;
				ctx->opayloadoff = 0;
			}
			memset(&iocb, 0, sizeof(iocb));
			iocb.fin = ctx->omsg->fin;
			iocb.opcode = ctx->omsg->opcode;
			iocb.rsv = ctx->omsg->rsv;
			iocb.mask = ctx->server ? 0 : 1;
			iocb.data = ctx->obufmark;
			iocb.data_length = ctx->obuflimit - ctx->obufmark;
			iocb.payload_length = ctx->opayloadlen;
			r = wslay_frame_send(ctx->frame_ctx, &iocb);
			if (r >= 0) {
				ctx->obufmark += r;
				if (ctx->obufmark == ctx->obuflimit) {
					ctx->obufmark = ctx->obuflimit = ctx->obuf;
					if (ctx->omsg->fin) {
						--ctx->queued_msg_count;
						wslay_event_omsg_free(ctx->omsg);
						ctx->omsg = NULL;
						break;
					} else {
						ctx->omsg->opcode = WSLAY_CONTINUATION_FRAME;
						/* RSV1 is not set on continuation frames */
						ctx->omsg->rsv = ctx->omsg->rsv & ~WSLAY_RSV1_BIT;
					}
				} else {
					break;
				}
			} else {
				if (r != WSLAY_ERR_WANT_WRITE || (ctx->error != WSLAY_ERR_WOULDBLOCK && ctx->error != 0)) {
					ctx->write_enabled = 0;
					return WSLAY_ERR_CALLBACK_FAILURE;
				}
				break;
			}
		}
	}
	return 0;
}