Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1114)

Unified Diff: mojo/services/network/udp_socket_unittest.cc

Issue 596383002: Mojo UDP API implementation. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@udp_interface
Patch Set: Created 6 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: mojo/services/network/udp_socket_unittest.cc
diff --git a/mojo/services/network/udp_socket_unittest.cc b/mojo/services/network/udp_socket_unittest.cc
new file mode 100644
index 0000000000000000000000000000000000000000..9d8e670d11bc717955f5095e5ea6ded9613976be
--- /dev/null
+++ b/mojo/services/network/udp_socket_unittest.cc
@@ -0,0 +1,386 @@
+// Copyright 2014 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 "base/at_exit.h"
+#include "base/macros.h"
+#include "base/memory/scoped_ptr.h"
+#include "mojo/public/cpp/bindings/callback.h"
+#include "mojo/services/public/interfaces/network/network_service.mojom.h"
+#include "mojo/services/public/interfaces/network/udp_socket.mojom.h"
+#include "mojo/shell/shell_test_helper.h"
+#include "net/base/net_errors.h"
+#include "testing/gtest/include/gtest/gtest.h"
+#include "url/gurl.h"
+
+namespace mojo {
+namespace service {
+namespace {
+
+NetAddressPtr GetLocalHostWithAnyPort() {
+ NetAddressPtr addr(NetAddress::New());
+ addr->family = NET_ADDRESS_FAMILY_IPV4;
+ addr->ipv4 = NetAddressIPv4::New();
+ addr->ipv4->port = 0;
+ addr->ipv4->addr.resize(4);
+ addr->ipv4->addr[0] = 127;
+ addr->ipv4->addr[1] = 0;
+ addr->ipv4->addr[2] = 0;
+ addr->ipv4->addr[3] = 1;
+
+ return addr.Pass();
+}
+
+Array<uint8_t> CreateTestMessage(uint8_t initial, size_t size) {
+ Array<uint8_t> array(size);
+ for (size_t i = 0; i < size; ++i)
+ array[i] = static_cast<uint8_t>((i + initial) % 256);
+ return array.Pass();
+}
+
+bool AreEqualArrays(const Array<uint8_t>& array_1,
+ const Array<uint8_t>& array_2) {
+ if (array_1.is_null() != array_2.is_null())
+ return false;
+ else if (array_1.is_null())
+ return true;
+
+ if (array_1.size() != array_2.size())
+ return false;
+
+ for (size_t i = 0; i < array_1.size(); ++i) {
+ if (array_1[i] != array_2[i])
+ return false;
+ }
+
+ return true;
+}
+
+template <typename CallbackType>
+class TestCallbackBase {
+ public:
+ TestCallbackBase() : state_(nullptr), run_loop_(nullptr), ran_(false) {}
+
+ ~TestCallbackBase() {
+ state_->set_test_callback(nullptr);
+ }
+
+ CallbackType callback() const { return callback_; }
+
+ void WaitForResult() {
+ if (ran_)
+ return;
+
+ base::RunLoop run_loop;
+ run_loop_ = &run_loop;
+ run_loop.Run();
+ run_loop_ = nullptr;
+ }
+
+ protected:
+ struct StateBase : public CallbackType::Runnable {
+ StateBase() : test_callback_(nullptr) {}
+ virtual ~StateBase() {}
+
+ void set_test_callback(TestCallbackBase* test_callback) {
+ test_callback_ = test_callback;
+ }
+
+ protected:
+ void NotifyRun() const {
+ if (test_callback_) {
+ test_callback_->ran_ = true;
+ if (test_callback_->run_loop_)
+ test_callback_->run_loop_->Quit();
+ }
+ }
+
+ TestCallbackBase* test_callback_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(StateBase);
+ };
+
+ // Takes ownership of |state|, and guarantees that it lives at least as long
+ // as this object.
+ void Initialize(StateBase* state) {
+ state_ = state;
+ state_->set_test_callback(this);
+ callback_ = CallbackType(
+ static_cast<typename CallbackType::Runnable*>(state_));
+ }
+
+ private:
+ // The lifespan is managed by |callback_| (and its copies).
+ StateBase* state_;
+ CallbackType callback_;
+ base::RunLoop* run_loop_;
+ bool ran_;
+
+ DISALLOW_COPY_AND_ASSIGN(TestCallbackBase);
+};
+
+class TestCallback : public TestCallbackBase<Callback<void(NetworkErrorPtr)>> {
+ public:
+ TestCallback() {
+ Initialize(new State());
+ }
+ ~TestCallback() {}
+
+ const NetworkErrorPtr& result() const { return result_; }
+
+ private:
+ struct State: public StateBase {
+ virtual ~State() {}
+
+ virtual void Run(NetworkErrorPtr result) const override {
+ if (test_callback_) {
+ TestCallback* callback = static_cast<TestCallback*>(test_callback_);
+ callback->result_ = result.Pass();
+ }
+ NotifyRun();
+ }
+ };
+
+ NetworkErrorPtr result_;
+};
+
+class TestCallbackWithAddress
+ : public TestCallbackBase<Callback<void(NetworkErrorPtr, NetAddressPtr)>> {
+ public:
+ TestCallbackWithAddress() {
+ Initialize(new State());
+ }
+ ~TestCallbackWithAddress() {}
+
+ const NetworkErrorPtr& result() const { return result_; }
+ const NetAddressPtr& net_address() const { return net_address_; }
+
+ private:
+ struct State : public StateBase {
+ virtual ~State() {}
+
+ virtual void Run(NetworkErrorPtr result,
+ NetAddressPtr net_address) const override {
+ if (test_callback_) {
+ TestCallbackWithAddress* callback =
+ static_cast<TestCallbackWithAddress*>(test_callback_);
+ callback->result_ = result.Pass();
+ callback->net_address_ = net_address.Pass();
+ }
+ NotifyRun();
+ }
+ };
+
+ NetworkErrorPtr result_;
+ NetAddressPtr net_address_;
+};
+
+class TestCallbackWithUint32
+ : public TestCallbackBase<Callback<void(uint32_t)>> {
+ public:
+ TestCallbackWithUint32() : result_(0) {
+ Initialize(new State());
+ }
+ ~TestCallbackWithUint32() {}
+
+ uint32_t result() const { return result_; }
+
+ private:
+ struct State : public StateBase {
+ virtual ~State() {}
+
+ virtual void Run(uint32_t result) const override {
+ if (test_callback_) {
+ TestCallbackWithUint32* callback =
+ static_cast<TestCallbackWithUint32*>(test_callback_);
+ callback->result_ = result;
+ }
+ NotifyRun();
+ }
+ };
+
+ uint32_t result_;
+};
+
+class UDPSocketTest : public testing::Test {
+ public:
+ UDPSocketTest() {}
+ virtual ~UDPSocketTest() {}
+
+ virtual void SetUp() OVERRIDE {
+ test_helper_.Init();
+
+ test_helper_.application_manager()->ConnectToService(
+ GURL("mojo:mojo_network_service"), &network_service_);
+
+ network_service_->CreateUDPSocket(GetProxy(&udp_socket_));
+ udp_socket_.set_client(&udp_socket_client_);
+ }
+
+ protected:
+ struct ReceiveResult {
+ NetworkErrorPtr result;
+ NetAddressPtr addr;
+ Array<uint8_t> data;
+ };
+
+ class UDPSocketClientImpl : public UDPSocketClient {
+ public:
+
+ UDPSocketClientImpl() : run_loop_(nullptr), expected_receive_count_(0) {}
+
+ virtual ~UDPSocketClientImpl() {
+ while (!results_.empty()) {
+ delete results_.front();
+ results_.pop();
+ }
+ }
+
+ virtual void OnReceived(NetworkErrorPtr result,
+ NetAddressPtr src_addr,
+ Array<uint8_t> data) OVERRIDE {
+ ReceiveResult* entry = new ReceiveResult();
+ entry->result = result.Pass();
+ entry->addr = src_addr.Pass();
+ entry->data = data.Pass();
+
+ results_.push(entry);
+
+ if (results_.size() == expected_receive_count_ && run_loop_) {
+ expected_receive_count_ = 0;
+ run_loop_->Quit();
+ }
+ }
+
+ base::RunLoop* run_loop_;
+ std::queue<ReceiveResult*> results_;
+ size_t expected_receive_count_;
+
+ DISALLOW_COPY_AND_ASSIGN(UDPSocketClientImpl);
+ };
+
+ std::queue<ReceiveResult*>* GetReceiveResults() {
+ return &udp_socket_client_.results_;
+ }
+
+ void WaitForReceiveResults(size_t count) {
+ if (GetReceiveResults()->size() == count)
+ return;
+
+ udp_socket_client_.expected_receive_count_ = count;
+ base::RunLoop run_loop;
+ udp_socket_client_.run_loop_ = &run_loop;
+ run_loop.Run();
+ udp_socket_client_.run_loop_ = nullptr;
+ }
+
+ base::ShadowingAtExitManager at_exit_;
+ shell::ShellTestHelper test_helper_;
+
+ NetworkServicePtr network_service_;
+ UDPSocketPtr udp_socket_;
+ UDPSocketClientImpl udp_socket_client_;
+
+ DISALLOW_COPY_AND_ASSIGN(UDPSocketTest);
+};
+
+} // namespace
+
+TEST_F(UDPSocketTest, Settings) {
+ TestCallback callback1;
+ udp_socket_->AllowAddressReuse(callback1.callback());
+ callback1.WaitForResult();
+ EXPECT_EQ(net::OK, callback1.result()->code);
+
+ // Should fail because the socket hasn't been bound.
+ TestCallback callback2;
+ udp_socket_->SetSendBufferSize(1024, callback2.callback());
+ callback2.WaitForResult();
+ EXPECT_NE(net::OK, callback2.result()->code);
+
+ // Should fail because the socket hasn't been bound.
+ TestCallback callback3;
+ udp_socket_->SetReceiveBufferSize(2048, callback3.callback());
+ callback3.WaitForResult();
+ EXPECT_NE(net::OK, callback3.result()->code);
+
+ TestCallbackWithAddress callback4;
+ udp_socket_->Bind(GetLocalHostWithAnyPort(), callback4.callback());
+ callback4.WaitForResult();
+ EXPECT_EQ(net::OK, callback4.result()->code);
+ EXPECT_NE(0u, callback4.net_address()->ipv4->port);
+
+ // Should fail because the socket has been bound.
+ TestCallback callback5;
+ udp_socket_->AllowAddressReuse(callback5.callback());
+ callback5.WaitForResult();
+ EXPECT_NE(net::OK, callback5.result()->code);
+
+ TestCallback callback6;
+ udp_socket_->SetSendBufferSize(1024, callback6.callback());
+ callback6.WaitForResult();
+ EXPECT_EQ(net::OK, callback6.result()->code);
+
+ TestCallback callback7;
+ udp_socket_->SetReceiveBufferSize(2048, callback7.callback());
+ callback7.WaitForResult();
+ EXPECT_EQ(net::OK, callback7.result()->code);
+
+ TestCallbackWithUint32 callback8;
+ udp_socket_->NegotiateMaxPendingSendRequests(0, callback8.callback());
+ callback8.WaitForResult();
+ EXPECT_GT(callback8.result(), 0u);
+
+ TestCallbackWithUint32 callback9;
+ udp_socket_->NegotiateMaxPendingSendRequests(16, callback9.callback());
+ callback9.WaitForResult();
+ EXPECT_GT(callback9.result(), 0u);
+}
+
+TEST_F(UDPSocketTest, TestReadWrite) {
+ TestCallbackWithAddress callback1;
+ udp_socket_->Bind(GetLocalHostWithAnyPort(), callback1.callback());
+ callback1.WaitForResult();
+ ASSERT_EQ(net::OK, callback1.result()->code);
+ ASSERT_NE(0u, callback1.net_address()->ipv4->port);
+
+ NetAddressPtr server_addr = callback1.net_address().Clone();
+
+ UDPSocketPtr client_socket;
+ network_service_->CreateUDPSocket(GetProxy(&client_socket));
+
+ TestCallbackWithAddress callback2;
+ client_socket->Bind(GetLocalHostWithAnyPort(), callback2.callback());
+ callback2.WaitForResult();
+ ASSERT_EQ(net::OK, callback2.result()->code);
+ ASSERT_NE(0u, callback2.net_address()->ipv4->port);
+
+ const size_t kDatagramCount = 6;
+ const size_t kDatagramSize = 255;
+ udp_socket_->ReceiveMore(kDatagramCount);
+
+ for (size_t i = 0; i < kDatagramCount; ++i) {
+ TestCallback callback;
+ client_socket->SendTo(
+ server_addr.Clone(),
+ CreateTestMessage(static_cast<uint8_t>(i), kDatagramSize),
+ callback.callback());
+ callback.WaitForResult();
+ EXPECT_EQ(255, callback.result()->code);
+ }
+
+ WaitForReceiveResults(kDatagramCount);
+ for (size_t i = 0; i < kDatagramCount; ++i) {
+ scoped_ptr<ReceiveResult> result(GetReceiveResults()->front());
+ GetReceiveResults()->pop();
+
+ EXPECT_EQ(static_cast<int>(kDatagramSize), result->result->code);
+ EXPECT_TRUE(AreEqualArrays(
+ CreateTestMessage(static_cast<uint8_t>(i), kDatagramSize),
+ result->data));
+ }
+}
+
+} // namespace service
+} // namespace mojo
« no previous file with comments | « mojo/services/network/udp_socket_impl.cc ('k') | mojo/services/public/interfaces/network/network_service.mojom » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698