OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/extensions/api/socket/udp_socket.h" | 5 #include "chrome/browser/extensions/api/socket/udp_socket.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 | 8 |
9 #include "base/memory/scoped_ptr.h" | 9 #include "base/memory/scoped_ptr.h" |
10 #include "base/message_loop.h" | 10 #include "base/message_loop.h" |
11 #include "chrome/test/base/browser_with_test_window_test.h" | 11 #include "chrome/test/base/browser_with_test_window_test.h" |
12 #include "net/base/io_buffer.h" | 12 #include "net/base/io_buffer.h" |
13 #include "testing/gtest/include/gtest/gtest.h" | 13 #include "testing/gtest/include/gtest/gtest.h" |
14 | 14 |
15 namespace extensions { | 15 namespace extensions { |
16 | 16 |
17 // UDPSocketUnitTest exists solely to make it easier to pass a specific | 17 // UDPSocketUnitTest exists solely to make it easier to pass a specific |
18 // gtest_filter argument during development. | 18 // gtest_filter argument during development. |
19 class UDPSocketUnitTest : public BrowserWithTestWindowTest { | 19 class UDPSocketUnitTest : public BrowserWithTestWindowTest { |
20 }; | 20 }; |
21 | 21 |
22 static void OnConnected(int result) { | 22 static void OnConnected(int result) { |
23 DCHECK(result == 0); | 23 EXPECT_EQ(0, result); |
24 } | 24 } |
25 | 25 |
26 static void OnCompleted(int bytes_read, | 26 static void OnCompleted(int bytes_read, |
27 scoped_refptr<net::IOBuffer> io_buffer, | 27 scoped_refptr<net::IOBuffer> io_buffer, |
28 const std::string& address, | 28 const std::string& address, |
29 int port) { | 29 int port) { |
30 // Do nothing; don't care. | 30 // Do nothing; don't care. |
31 } | 31 } |
32 | 32 |
33 static const char test_message[] = "$$TESTMESSAGETESTMESSAGETESTMESSAGETEST$$"; | |
34 static const int test_message_length = ARRAYSIZE_UNSAFE(test_message); | |
35 | |
36 static void OnSendCompleted(int result) { | |
37 EXPECT_EQ(test_message_length, result); | |
38 } | |
39 | |
33 TEST(UDPSocketUnitTest, TestUDPSocketRecvFrom) { | 40 TEST(UDPSocketUnitTest, TestUDPSocketRecvFrom) { |
34 MessageLoopForIO io_loop; // for RecvFrom to do its threaded work. | 41 MessageLoopForIO io_loop; // For RecvFrom to do its threaded work. |
35 UDPSocket socket("abcdefghijklmnopqrst"); | 42 UDPSocket socket("abcdefghijklmnopqrst"); |
36 | 43 |
37 // Confirm that we can call two RecvFroms in quick succession without | 44 // Confirm that we can call two RecvFroms in quick succession without |
38 // triggering crbug.com/146606. | 45 // triggering crbug.com/146606. |
39 socket.Connect("127.0.0.1", 40000, base::Bind(&OnConnected)); | 46 socket.Connect("127.0.0.1", 40000, base::Bind(&OnConnected)); |
40 socket.RecvFrom(4096, base::Bind(&OnCompleted)); | 47 socket.RecvFrom(4096, base::Bind(&OnCompleted)); |
41 socket.RecvFrom(4096, base::Bind(&OnCompleted)); | 48 socket.RecvFrom(4096, base::Bind(&OnCompleted)); |
42 } | 49 } |
43 | 50 |
51 TEST(UDPSocketUnitTest, TestUDPMulticastJoinGroup) { | |
52 const char* kGroup = "237.132.100.17"; | |
53 UDPSocket src("abcdefghijklmnopqrst"); | |
54 UDPSocket dest("abcdefghijklmnopqrst"); | |
55 | |
56 EXPECT_EQ(0, dest.Bind("0.0.0.0", 13333)); | |
57 EXPECT_EQ(0, dest.JoinGroup(kGroup)); | |
58 base::hash_set<std::string> groups; | |
59 dest.GetJoinedGroups(&groups); | |
60 EXPECT_EQ(static_cast<size_t>(1), groups.size()); | |
61 EXPECT_EQ(kGroup, *groups.begin()); | |
62 EXPECT_NE(0, dest.LeaveGroup("237.132.100.13")); | |
63 EXPECT_EQ(0, dest.LeaveGroup(kGroup)); | |
64 dest.GetJoinedGroups(&groups); | |
65 EXPECT_EQ(static_cast<size_t>(0), groups.size()); | |
66 } | |
67 | |
68 TEST(UDPSocketUnitTest, TestUDPMulticastTimeToLive) { | |
69 const char* kGroup = "237.132.100.17"; | |
70 UDPSocket socket("abcdefghijklmnopqrst"); | |
71 EXPECT_NE(0, socket.SetMulticastTimeToLive(-1)); // Negative TTL shall fail. | |
72 EXPECT_EQ(0, socket.SetMulticastTimeToLive(3)); | |
73 socket.Connect(kGroup, 13333, base::Bind(&OnConnected)); | |
74 } | |
75 | |
76 TEST(UDPSocketUnitTest, TestUDPMulticastLoopbackMode) { | |
77 const char* kGroup = "237.132.100.17"; | |
78 UDPSocket socket("abcdefghijklmnopqrst"); | |
79 EXPECT_EQ(0, socket.SetMulticastLoopbackMode(false)); | |
80 socket.Connect(kGroup, 13333, base::Bind(&OnConnected)); | |
81 } | |
82 | |
83 static void QuitMessageLoop() { | |
84 MessageLoopForIO::current()->QuitNow(); | |
85 } | |
86 | |
87 static void SendMulticastPacket(UDPSocket* src, | |
wtc
2013/04/23 18:01:07
This method should be documented. It is not obviou
Bei Zhang
2013/04/23 20:36:39
Done.
| |
88 const char* group, | |
wtc
2013/04/23 18:01:07
The |group| argument is not used. Can we remove it
Bei Zhang
2013/04/23 20:36:39
Done.
| |
89 int result) { | |
90 if (result == 0) { | |
91 scoped_refptr<net::IOBuffer> data = new net::WrappedIOBuffer(test_message); | |
92 src->Write(data, test_message_length, base::Bind(&OnSendCompleted)); | |
93 MessageLoopForIO::current()->PostDelayedTask(FROM_HERE, | |
94 base::Bind(&SendMulticastPacket, src, group, result), | |
95 base::TimeDelta::FromSeconds(1)); | |
wtc
2013/04/23 18:01:07
Do we need to cancel this delayed task if it is no
Bei Zhang
2013/04/23 20:36:39
We do not need to cancel the task.
It will not be
| |
96 } else { | |
97 QuitMessageLoop(); | |
98 FAIL() << "Failed to connect to multicast address. Error code: " << result; | |
99 } | |
100 } | |
101 | |
102 static void OnMulticastReadCompleted(bool *packet_received, | |
wtc
2013/04/23 18:01:07
|packet_received| is an output argument. Our Style
Bei Zhang
2013/04/23 20:36:39
Unfortunately we cannot bind argument to the end o
wtc
2013/04/23 22:15:26
Thank you for the explanation. (I am not familiar
| |
103 int count, | |
104 scoped_refptr<net::IOBuffer> io_buffer) { | |
105 EXPECT_EQ(test_message_length, count); | |
106 EXPECT_EQ(0, strncmp(io_buffer->data(), test_message, test_message_length)); | |
107 *packet_received = true; | |
108 QuitMessageLoop(); | |
109 } | |
110 | |
111 TEST(UDPSocketUnitTest, TestUDPMulticastRecv) { | |
112 const int kPort = 9999; | |
113 const char* const kGroup = "237.132.100.17"; | |
114 bool packet_received = false; | |
115 MessageLoopForIO io_loop; // For Read to do its threaded work. | |
116 UDPSocket dest("abcdefghijklmnopqrst"); | |
117 UDPSocket src("abcdefghijklmnopqrst"); | |
118 | |
119 // Receiver | |
120 EXPECT_EQ(0, dest.Bind("0.0.0.0", kPort)); | |
121 EXPECT_EQ(0, dest.JoinGroup(kGroup)); | |
122 dest.Read(1024, base::Bind(&OnMulticastReadCompleted, &packet_received)); | |
123 | |
124 // Sender | |
125 EXPECT_EQ(0, src.SetMulticastTimeToLive(0)); | |
126 src.Connect(kGroup, kPort, base::Bind(&SendMulticastPacket, &src, kGroup)); | |
127 | |
128 // If not received in 4 seconds quit the message loop. | |
wtc
2013/04/23 18:01:07
This should be increased to a very large number, s
Bei Zhang
2013/04/23 20:36:39
Done. I found out that there is no per tests case
| |
129 io_loop.PostDelayedTask(FROM_HERE, | |
130 base::Bind(&QuitMessageLoop), | |
131 base::TimeDelta::FromSeconds(4)); | |
132 | |
133 io_loop.Run(); | |
134 | |
135 EXPECT_TRUE(packet_received) << "Failed to receive from multicast address"; | |
136 } | |
137 | |
44 } // namespace extensions | 138 } // namespace extensions |
OLD | NEW |