| Index: net/websockets/websocket_frame_unittest.cc
|
| diff --git a/net/websockets/websocket_frame_unittest.cc b/net/websockets/websocket_frame_unittest.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..270053c96580d2831e9b08ed4e218a742e5da242
|
| --- /dev/null
|
| +++ b/net/websockets/websocket_frame_unittest.cc
|
| @@ -0,0 +1,224 @@
|
| +// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "net/websockets/websocket_frame.h"
|
| +
|
| +#include <vector>
|
| +
|
| +#include "base/basictypes.h"
|
| +#include "testing/gtest/include/gtest/gtest.h"
|
| +
|
| +namespace net {
|
| +
|
| +TEST(WebSocketFrameHeaderTest, FrameLengths) {
|
| + struct TestCase {
|
| + const char* frame_header;
|
| + size_t frame_header_length;
|
| + uint64 frame_length;
|
| + };
|
| + static const TestCase kTests[] = {
|
| + { "\x81\x00", 2, GG_UINT64_C(0) },
|
| + { "\x81\x7D", 2, GG_UINT64_C(125) },
|
| + { "\x81\x7E\x00\x7E", 4, GG_UINT64_C(126) },
|
| + { "\x81\x7E\xFF\xFF", 4, GG_UINT64_C(0xFFFF) },
|
| + { "\x81\x7F\x00\x00\x00\x00\x00\x01\x00\x00", 10, GG_UINT64_C(0x10000) },
|
| + { "\x81\x7F\x7F\xFF\xFF\xFF\xFF\xFF\xFF\xFF", 10,
|
| + GG_UINT64_C(0x7FFFFFFFFFFFFFFF) }
|
| + };
|
| + static const int kNumTests = ARRAYSIZE_UNSAFE(kTests);
|
| +
|
| + for (int i = 0; i < kNumTests; ++i) {
|
| + WebSocketFrameHeader header;
|
| + header.final = true;
|
| + header.reserved1 = false;
|
| + header.reserved2 = false;
|
| + header.reserved3 = false;
|
| + header.opcode = WebSocketFrameHeader::kOpCodeText;
|
| + header.masked = false;
|
| + header.payload_length = kTests[i].frame_length;
|
| +
|
| + std::vector<char> expected_output(
|
| + kTests[i].frame_header,
|
| + kTests[i].frame_header + kTests[i].frame_header_length);
|
| + std::vector<char> output;
|
| + WriteWebSocketFrameHeader(header, NULL, &output);
|
| +
|
| + EXPECT_EQ(expected_output, output);
|
| + }
|
| +}
|
| +
|
| +TEST(WebSocketFrameHeaderTest, FrameLengthsWithMasking) {
|
| + static const char kMaskingKey[] = "\xDE\xAD\xBE\xEF";
|
| + COMPILE_ASSERT(ARRAYSIZE_UNSAFE(kMaskingKey) - 1 ==
|
| + WebSocketFrameHeader::kMaskingKeyLength,
|
| + incorrect_masking_key_size);
|
| +
|
| + struct TestCase {
|
| + const char* frame_header;
|
| + size_t frame_header_length;
|
| + uint64 frame_length;
|
| + };
|
| + static const TestCase kTests[] = {
|
| + { "\x81\x80\xDE\xAD\xBE\xEF", 6, GG_UINT64_C(0) },
|
| + { "\x81\xFD\xDE\xAD\xBE\xEF", 6, GG_UINT64_C(125) },
|
| + { "\x81\xFE\x00\x7E\xDE\xAD\xBE\xEF", 8, GG_UINT64_C(126) },
|
| + { "\x81\xFE\xFF\xFF\xDE\xAD\xBE\xEF", 8, GG_UINT64_C(0xFFFF) },
|
| + { "\x81\xFF\x00\x00\x00\x00\x00\x01\x00\x00\xDE\xAD\xBE\xEF", 14,
|
| + GG_UINT64_C(0x10000) },
|
| + { "\x81\xFF\x7F\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xDE\xAD\xBE\xEF", 14,
|
| + GG_UINT64_C(0x7FFFFFFFFFFFFFFF) }
|
| + };
|
| + static const int kNumTests = ARRAYSIZE_UNSAFE(kTests);
|
| +
|
| + WebSocketMaskingKey masking_key;
|
| + std::copy(kMaskingKey, kMaskingKey + WebSocketFrameHeader::kMaskingKeyLength,
|
| + masking_key.key);
|
| +
|
| + for (int i = 0; i < kNumTests; ++i) {
|
| + WebSocketFrameHeader header;
|
| + header.final = true;
|
| + header.reserved1 = false;
|
| + header.reserved2 = false;
|
| + header.reserved3 = false;
|
| + header.opcode = WebSocketFrameHeader::kOpCodeText;
|
| + header.masked = true;
|
| + header.payload_length = kTests[i].frame_length;
|
| +
|
| + std::vector<char> expected_output(
|
| + kTests[i].frame_header,
|
| + kTests[i].frame_header + kTests[i].frame_header_length);
|
| + std::vector<char> output;
|
| + WriteWebSocketFrameHeader(header, &masking_key, &output);
|
| +
|
| + EXPECT_EQ(expected_output, output);
|
| + }
|
| +}
|
| +
|
| +TEST(WebSocketFrameHeaderTest, FrameOpCodes) {
|
| + struct TestCase {
|
| + const char* frame_header;
|
| + size_t frame_header_length;
|
| + WebSocketFrameHeader::OpCode opcode;
|
| + };
|
| + static const TestCase kTests[] = {
|
| + { "\x80\x00", 2, WebSocketFrameHeader::kOpCodeContinuation },
|
| + { "\x81\x00", 2, WebSocketFrameHeader::kOpCodeText },
|
| + { "\x82\x00", 2, WebSocketFrameHeader::kOpCodeBinary },
|
| + { "\x88\x00", 2, WebSocketFrameHeader::kOpCodeClose },
|
| + { "\x89\x00", 2, WebSocketFrameHeader::kOpCodePing },
|
| + { "\x8A\x00", 2, WebSocketFrameHeader::kOpCodePong },
|
| + // These are undefined opcodes, but the builder should accept them anyway.
|
| + { "\x83\x00", 2, 0x3 },
|
| + { "\x84\x00", 2, 0x4 },
|
| + { "\x85\x00", 2, 0x5 },
|
| + { "\x86\x00", 2, 0x6 },
|
| + { "\x87\x00", 2, 0x7 },
|
| + { "\x8B\x00", 2, 0xB },
|
| + { "\x8C\x00", 2, 0xC },
|
| + { "\x8D\x00", 2, 0xD },
|
| + { "\x8E\x00", 2, 0xE },
|
| + { "\x8F\x00", 2, 0xF }
|
| + };
|
| + static const int kNumTests = ARRAYSIZE_UNSAFE(kTests);
|
| +
|
| + for (int i = 0; i < kNumTests; ++i) {
|
| + WebSocketFrameHeader header;
|
| + header.final = true;
|
| + header.reserved1 = false;
|
| + header.reserved2 = false;
|
| + header.reserved3 = false;
|
| + header.opcode = kTests[i].opcode;
|
| + header.masked = false;
|
| + header.payload_length = 0;
|
| +
|
| + std::vector<char> expected_output(
|
| + kTests[i].frame_header,
|
| + kTests[i].frame_header + kTests[i].frame_header_length);
|
| + std::vector<char> output;
|
| + WriteWebSocketFrameHeader(header, NULL, &output);
|
| +
|
| + EXPECT_EQ(expected_output, output);
|
| + }
|
| +}
|
| +
|
| +TEST(WebSocketFrameHeaderTest, FinalBitAndReservedBits) {
|
| + struct TestCase {
|
| + const char* frame_header;
|
| + size_t frame_header_length;
|
| + bool final;
|
| + bool reserved1;
|
| + bool reserved2;
|
| + bool reserved3;
|
| + };
|
| + static const TestCase kTests[] = {
|
| + { "\x81\x00", 2, true, false, false, false },
|
| + { "\x01\x00", 2, false, false, false, false },
|
| + { "\xC1\x00", 2, true, true, false, false },
|
| + { "\xA1\x00", 2, true, false, true, false },
|
| + { "\x91\x00", 2, true, false, false, true },
|
| + { "\x71\x00", 2, false, true, true, true },
|
| + { "\xF1\x00", 2, true, true, true, true }
|
| + };
|
| + static const int kNumTests = ARRAYSIZE_UNSAFE(kTests);
|
| +
|
| + for (int i = 0; i < kNumTests; ++i) {
|
| + WebSocketFrameHeader header;
|
| + header.final = kTests[i].final;
|
| + header.reserved1 = kTests[i].reserved1;
|
| + header.reserved2 = kTests[i].reserved2;
|
| + header.reserved3 = kTests[i].reserved3;
|
| + header.opcode = WebSocketFrameHeader::kOpCodeText;
|
| + header.masked = false;
|
| + header.payload_length = 0;
|
| +
|
| + std::vector<char> expected_output(
|
| + kTests[i].frame_header,
|
| + kTests[i].frame_header + kTests[i].frame_header_length);
|
| + std::vector<char> output;
|
| + WriteWebSocketFrameHeader(header, NULL, &output);
|
| +
|
| + EXPECT_EQ(expected_output, output);
|
| + }
|
| +}
|
| +
|
| +TEST(WebSocketFrameTest, MaskPayload) {
|
| + struct TestCase {
|
| + const char* masking_key;
|
| + uint64 frame_offset;
|
| + const char* input;
|
| + const char* output;
|
| + size_t data_length;
|
| + };
|
| + static const TestCase kTests[] = {
|
| + { "\xDE\xAD\xBE\xEF", 0, "FooBar", "\x98\xC2\xD1\xAD\xBF\xDF", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 1, "FooBar", "\xEB\xD1\x80\x9C\xCC\xCC", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 2, "FooBar", "\xF8\x80\xB1\xEF\xDF\x9D", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 3, "FooBar", "\xA9\xB1\xC2\xFC\x8E\xAC", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 4, "FooBar", "\x98\xC2\xD1\xAD\xBF\xDF", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 42, "FooBar", "\xF8\x80\xB1\xEF\xDF\x9D", 6 },
|
| + { "\xDE\xAD\xBE\xEF", 0, "", "", 0 },
|
| + { "\xDE\xAD\xBE\xEF", 0, "\xDE\xAD\xBE\xEF", "\x00\x00\x00\x00", 4 },
|
| + { "\xDE\xAD\xBE\xEF", 0, "\x00\x00\x00\x00", "\xDE\xAD\xBE\xEF", 4 },
|
| + { "\x00\x00\x00\x00", 0, "FooBar", "FooBar", 6 },
|
| + { "\xFF\xFF\xFF\xFF", 0, "FooBar", "\xB9\x90\x90\xBD\x9E\x8D", 6 },
|
| + };
|
| + static const int kNumTests = ARRAYSIZE_UNSAFE(kTests);
|
| +
|
| + for (int i = 0; i < kNumTests; ++i) {
|
| + WebSocketMaskingKey masking_key;
|
| + std::copy(kTests[i].masking_key,
|
| + kTests[i].masking_key + WebSocketFrameHeader::kMaskingKeyLength,
|
| + masking_key.key);
|
| + std::vector<char> frame_data(kTests[i].input,
|
| + kTests[i].input + kTests[i].data_length);
|
| + std::vector<char> expected_output(kTests[i].output,
|
| + kTests[i].output + kTests[i].data_length);
|
| + MaskWebSocketFramePayload(masking_key,
|
| + kTests[i].frame_offset,
|
| + &frame_data);
|
| + EXPECT_EQ(expected_output, frame_data);
|
| + }
|
| +}
|
| +
|
| +} // namespace net
|
|
|