henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2004 The WebRTC Project Authors. All rights reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
Jonas Olsson | a4d8737 | 2019-07-05 17:08:33 | [diff] [blame] | 11 | #include "p2p/base/stun_server.h" |
| 12 | |
Yves Gerey | 3e70781 | 2018-11-28 15:47:49 | [diff] [blame] | 13 | #include <string.h> |
Jonas Olsson | a4d8737 | 2019-07-05 17:08:33 | [diff] [blame] | 14 | |
kwiberg | 3ec4679 | 2016-04-27 14:22:53 | [diff] [blame] | 15 | #include <memory> |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 16 | #include <string> |
| 17 | |
Karl Wiberg | 918f50c | 2018-07-05 09:40:33 | [diff] [blame] | 18 | #include "absl/memory/memory.h" |
Steve Anton | 10542f2 | 2019-01-11 17:11:00 | [diff] [blame] | 19 | #include "rtc_base/byte_buffer.h" |
| 20 | #include "rtc_base/ip_address.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 04:47:31 | [diff] [blame] | 21 | #include "rtc_base/logging.h" |
Steve Anton | 10542f2 | 2019-01-11 17:11:00 | [diff] [blame] | 22 | #include "rtc_base/test_client.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 04:47:31 | [diff] [blame] | 23 | #include "rtc_base/thread.h" |
Steve Anton | 10542f2 | 2019-01-11 17:11:00 | [diff] [blame] | 24 | #include "rtc_base/virtual_socket_server.h" |
Yves Gerey | 3e70781 | 2018-11-28 15:47:49 | [diff] [blame] | 25 | #include "test/gtest.h" |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 26 | |
Steve Anton | 6c38cc7 | 2017-11-29 18:25:58 | [diff] [blame] | 27 | namespace cricket { |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 28 | |
Steve Anton | 6c38cc7 | 2017-11-29 18:25:58 | [diff] [blame] | 29 | namespace { |
| 30 | const rtc::SocketAddress server_addr("99.99.99.1", 3478); |
| 31 | const rtc::SocketAddress client_addr("1.2.3.4", 1234); |
| 32 | } // namespace |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 33 | |
Mirko Bonadei | 6a489f2 | 2019-04-09 13:11:12 | [diff] [blame] | 34 | class StunServerTest : public ::testing::Test { |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 35 | public: |
deadbeef | 98e186c | 2017-05-17 01:00:06 | [diff] [blame] | 36 | StunServerTest() : ss_(new rtc::VirtualSocketServer()), network_(ss_.get()) {} |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 37 | virtual void SetUp() { |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 38 | server_.reset( |
| 39 | new StunServer(rtc::AsyncUDPSocket::Create(ss_.get(), server_addr))); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 40 | client_.reset(new rtc::TestClient( |
Karl Wiberg | 918f50c | 2018-07-05 09:40:33 | [diff] [blame] | 41 | absl::WrapUnique(rtc::AsyncUDPSocket::Create(ss_.get(), client_addr)))); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 42 | |
johan | 27c3d5b | 2016-10-17 07:54:57 | [diff] [blame] | 43 | network_.Start(); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 44 | } |
| 45 | void Send(const StunMessage& msg) { |
jbauch | f1f8720 | 2016-03-30 13:43:37 | [diff] [blame] | 46 | rtc::ByteBufferWriter buf; |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 47 | msg.Write(&buf); |
| 48 | Send(buf.Data(), static_cast<int>(buf.Length())); |
| 49 | } |
| 50 | void Send(const char* buf, int len) { |
| 51 | client_->SendTo(buf, len, server_addr); |
| 52 | } |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 53 | bool ReceiveFails() { return (client_->CheckNoPacket()); } |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 54 | StunMessage* Receive() { |
| 55 | StunMessage* msg = NULL; |
nisse | 32f2505 | 2017-05-08 08:57:18 | [diff] [blame] | 56 | std::unique_ptr<rtc::TestClient::Packet> packet = |
jlmiller@webrtc.org | ec499be | 2015-02-07 22:37:59 | [diff] [blame] | 57 | client_->NextPacket(rtc::TestClient::kTimeoutMs); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 58 | if (packet) { |
jbauch | f1f8720 | 2016-03-30 13:43:37 | [diff] [blame] | 59 | rtc::ByteBufferReader buf(packet->buf, packet->size); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 60 | msg = new StunMessage(); |
| 61 | msg->Read(&buf); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 62 | } |
| 63 | return msg; |
| 64 | } |
Steve Anton | 6c38cc7 | 2017-11-29 18:25:58 | [diff] [blame] | 65 | |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 66 | private: |
kwiberg | 3ec4679 | 2016-04-27 14:22:53 | [diff] [blame] | 67 | std::unique_ptr<rtc::VirtualSocketServer> ss_; |
johan | 27c3d5b | 2016-10-17 07:54:57 | [diff] [blame] | 68 | rtc::Thread network_; |
kwiberg | 3ec4679 | 2016-04-27 14:22:53 | [diff] [blame] | 69 | std::unique_ptr<StunServer> server_; |
| 70 | std::unique_ptr<rtc::TestClient> client_; |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 71 | }; |
| 72 | |
| 73 | // Disable for TSan v2, see |
| 74 | // https://code.google.com/p/webrtc/issues/detail?id=2517 for details. |
| 75 | #if !defined(THREAD_SANITIZER) |
| 76 | |
| 77 | TEST_F(StunServerTest, TestGood) { |
| 78 | StunMessage req; |
Min Wang | 1e00dbc | 2019-06-26 18:08:29 | [diff] [blame] | 79 | // kStunLegacyTransactionIdLength = 16 for legacy RFC 3489 request |
| 80 | std::string transaction_id = "0123456789abcdef"; |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 81 | req.SetType(STUN_BINDING_REQUEST); |
| 82 | req.SetTransactionID(transaction_id); |
| 83 | Send(req); |
| 84 | |
| 85 | StunMessage* msg = Receive(); |
| 86 | ASSERT_TRUE(msg != NULL); |
| 87 | EXPECT_EQ(STUN_BINDING_RESPONSE, msg->type()); |
| 88 | EXPECT_EQ(req.transaction_id(), msg->transaction_id()); |
| 89 | |
| 90 | const StunAddressAttribute* mapped_addr = |
| 91 | msg->GetAddress(STUN_ATTR_MAPPED_ADDRESS); |
| 92 | EXPECT_TRUE(mapped_addr != NULL); |
| 93 | EXPECT_EQ(1, mapped_addr->family()); |
| 94 | EXPECT_EQ(client_addr.port(), mapped_addr->port()); |
Min Wang | 1e00dbc | 2019-06-26 18:08:29 | [diff] [blame] | 95 | |
| 96 | delete msg; |
| 97 | } |
| 98 | |
| 99 | TEST_F(StunServerTest, TestGoodXorMappedAddr) { |
| 100 | StunMessage req; |
| 101 | // kStunTransactionIdLength = 12 for RFC 5389 request |
| 102 | // StunMessage::Write will automatically insert magic cookie (0x2112A442) |
| 103 | std::string transaction_id = "0123456789ab"; |
| 104 | req.SetType(STUN_BINDING_REQUEST); |
| 105 | req.SetTransactionID(transaction_id); |
| 106 | Send(req); |
| 107 | |
| 108 | StunMessage* msg = Receive(); |
| 109 | ASSERT_TRUE(msg != NULL); |
| 110 | EXPECT_EQ(STUN_BINDING_RESPONSE, msg->type()); |
| 111 | EXPECT_EQ(req.transaction_id(), msg->transaction_id()); |
| 112 | |
| 113 | const StunAddressAttribute* mapped_addr = |
| 114 | msg->GetAddress(STUN_ATTR_XOR_MAPPED_ADDRESS); |
| 115 | EXPECT_TRUE(mapped_addr != NULL); |
| 116 | EXPECT_EQ(1, mapped_addr->family()); |
| 117 | EXPECT_EQ(client_addr.port(), mapped_addr->port()); |
| 118 | |
| 119 | delete msg; |
| 120 | } |
| 121 | |
| 122 | // Send legacy RFC 3489 request, should not get xor mapped addr |
| 123 | TEST_F(StunServerTest, TestNoXorMappedAddr) { |
| 124 | StunMessage req; |
| 125 | // kStunLegacyTransactionIdLength = 16 for legacy RFC 3489 request |
| 126 | std::string transaction_id = "0123456789abcdef"; |
| 127 | req.SetType(STUN_BINDING_REQUEST); |
| 128 | req.SetTransactionID(transaction_id); |
| 129 | Send(req); |
| 130 | |
| 131 | StunMessage* msg = Receive(); |
| 132 | ASSERT_TRUE(msg != NULL); |
| 133 | EXPECT_EQ(STUN_BINDING_RESPONSE, msg->type()); |
| 134 | EXPECT_EQ(req.transaction_id(), msg->transaction_id()); |
| 135 | |
| 136 | const StunAddressAttribute* mapped_addr = |
| 137 | msg->GetAddress(STUN_ATTR_XOR_MAPPED_ADDRESS); |
| 138 | EXPECT_TRUE(mapped_addr == NULL); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 139 | |
| 140 | delete msg; |
| 141 | } |
| 142 | |
Steve Anton | 6c38cc7 | 2017-11-29 18:25:58 | [diff] [blame] | 143 | #endif // if !defined(THREAD_SANITIZER) |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 144 | |
| 145 | TEST_F(StunServerTest, TestBad) { |
Yves Gerey | 665174f | 2018-06-19 13:03:05 | [diff] [blame] | 146 | const char* bad = |
| 147 | "this is a completely nonsensical message whose only " |
| 148 | "purpose is to make the parser go 'ack'. it doesn't " |
| 149 | "look anything like a normal stun message"; |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 150 | Send(bad, static_cast<int>(strlen(bad))); |
| 151 | |
jlmiller@webrtc.org | ec499be | 2015-02-07 22:37:59 | [diff] [blame] | 152 | ASSERT_TRUE(ReceiveFails()); |
henrike@webrtc.org | 269fb4b | 2014-10-28 22:20:11 | [diff] [blame] | 153 | } |
Steve Anton | 6c38cc7 | 2017-11-29 18:25:58 | [diff] [blame] | 154 | |
| 155 | } // namespace cricket |