// Copyright 2013 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 "ppapi/tests/test_host_resolver.h" #include "ppapi/cpp/host_resolver.h" #include "ppapi/cpp/net_address.h" #include "ppapi/cpp/tcp_socket.h" #include "ppapi/cpp/var.h" #include "ppapi/tests/test_utils.h" #include "ppapi/tests/testing_instance.h" REGISTER_TEST_CASE(HostResolver); TestHostResolver::TestHostResolver(TestingInstance* instance) : TestCase(instance) { } bool TestHostResolver::Init() { bool host_resolver_is_available = pp::HostResolver::IsAvailable(); if (!host_resolver_is_available) instance_->AppendError("PPB_HostResolver interface not available"); bool tcp_socket_is_available = pp::TCPSocket::IsAvailable(); if (!tcp_socket_is_available) instance_->AppendError("PPB_TCPSocket interface not available"); bool init_host_port = GetLocalHostPort(instance_->pp_instance(), &host_, &port_); if (!init_host_port) instance_->AppendError("Can't init host and port"); return host_resolver_is_available && tcp_socket_is_available && init_host_port && CheckTestingInterface() && EnsureRunningOverHTTP(); } void TestHostResolver::RunTests(const std::string& filter) { RUN_TEST(Empty, filter); RUN_CALLBACK_TEST(TestHostResolver, Resolve, filter); RUN_CALLBACK_TEST(TestHostResolver, ResolveIPv4, filter); } std::string TestHostResolver::SyncConnect( pp::TCPSocket* socket, const pp::NetAddress& address) { TestCompletionCallback callback(instance_->pp_instance(), callback_type()); callback.WaitForResult(socket->Connect(address, callback.GetCallback())); CHECK_CALLBACK_BEHAVIOR(callback); ASSERT_EQ(PP_OK, callback.result()); PASS(); } std::string TestHostResolver::SyncRead(pp::TCPSocket* socket, char* buffer, int32_t num_bytes, int32_t* bytes_read) { TestCompletionCallback callback(instance_->pp_instance(), callback_type()); callback.WaitForResult( socket->Read(buffer, num_bytes, callback.GetCallback())); CHECK_CALLBACK_BEHAVIOR(callback); ASSERT_EQ(num_bytes, callback.result()); *bytes_read = callback.result(); PASS(); } std::string TestHostResolver::SyncWrite(pp::TCPSocket* socket, const char* buffer, int32_t num_bytes, int32_t* bytes_written) { TestCompletionCallback callback(instance_->pp_instance(), callback_type()); callback.WaitForResult( socket->Write(buffer, num_bytes, callback.GetCallback())); CHECK_CALLBACK_BEHAVIOR(callback); ASSERT_EQ(num_bytes, callback.result()); *bytes_written = callback.result(); PASS(); } std::string TestHostResolver::CheckHTTPResponse(pp::TCPSocket* socket, const std::string& request, const std::string& response) { int32_t rv = 0; ASSERT_SUBTEST_SUCCESS( SyncWrite(socket, request.c_str(), request.size(), &rv)); std::vector response_buffer(response.size()); ASSERT_SUBTEST_SUCCESS( SyncRead(socket, &response_buffer[0], response.size(), &rv)); std::string actual_response(&response_buffer[0], rv); if (response != actual_response) { return "CheckHTTPResponse failed, expected: " + response + ", actual: " + actual_response; } PASS(); } std::string TestHostResolver::SyncResolve( pp::HostResolver* host_resolver, const std::string& host, uint16_t port, const PP_HostResolver_Hint& hint) { TestCompletionCallback callback(instance_->pp_instance(), callback_type()); callback.WaitForResult( host_resolver->Resolve(host.c_str(), port, hint, callback.GetCallback())); CHECK_CALLBACK_BEHAVIOR(callback); ASSERT_EQ(PP_OK, callback.result()); PASS(); } std::string TestHostResolver::ParameterizedTestResolve( const PP_HostResolver_Hint& hint) { pp::HostResolver host_resolver(instance_); ASSERT_SUBTEST_SUCCESS(SyncResolve(&host_resolver, host_, port_, hint)); size_t size = host_resolver.GetNetAddressCount(); ASSERT_TRUE(size >= 1); pp::NetAddress address; for (size_t i = 0; i < size; ++i) { address = host_resolver.GetNetAddress(i); ASSERT_NE(0, address.pp_resource()); pp::TCPSocket socket(instance_); ASSERT_SUBTEST_SUCCESS(SyncConnect(&socket, address)); ASSERT_SUBTEST_SUCCESS(CheckHTTPResponse(&socket, "GET / HTTP/1.0\r\n\r\n", "HTTP")); socket.Close(); } address = host_resolver.GetNetAddress(size); ASSERT_EQ(0, address.pp_resource()); pp::Var canonical_name = host_resolver.GetCanonicalName(); ASSERT_TRUE(canonical_name.is_string()); ASSERT_SUBTEST_SUCCESS(SyncResolve(&host_resolver, canonical_name.AsString(), port_, hint)); size = host_resolver.GetNetAddressCount(); ASSERT_TRUE(size >= 1); PASS(); } std::string TestHostResolver::TestEmpty() { pp::HostResolver host_resolver(instance_); ASSERT_EQ(0, host_resolver.GetNetAddressCount()); pp::NetAddress address = host_resolver.GetNetAddress(0); ASSERT_EQ(0, address.pp_resource()); PASS(); } std::string TestHostResolver::TestResolve() { PP_HostResolver_Hint hint; hint.family = PP_NETADDRESS_FAMILY_UNSPECIFIED; hint.flags = PP_HOSTRESOLVER_FLAG_CANONNAME; return ParameterizedTestResolve(hint); } std::string TestHostResolver::TestResolveIPv4() { PP_HostResolver_Hint hint; hint.family = PP_NETADDRESS_FAMILY_IPV4; hint.flags = PP_HOSTRESOLVER_FLAG_CANONNAME; return ParameterizedTestResolve(hint); }