//
// Copyright (C) 2012 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#include "shill/dns_server_tester.h"
#include <memory>
#include <string>
#include <base/bind.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include "shill/mock_connection.h"
#include "shill/mock_control.h"
#include "shill/mock_device_info.h"
#include "shill/mock_dns_client.h"
#include "shill/mock_dns_client_factory.h"
#include "shill/mock_event_dispatcher.h"
#include "shill/net/mock_time.h"
using base::Bind;
using base::Callback;
using base::Unretained;
using std::string;
using std::vector;
using testing::_;
using testing::AtLeast;
using testing::DoAll;
using testing::InSequence;
using testing::Mock;
using testing::NiceMock;
using testing::Return;
using testing::ReturnRef;
using testing::SetArgumentPointee;
using testing::StrictMock;
using testing::Test;
namespace shill {
namespace {
const char kInterfaceName[] = "int0";
const char kDNSServer0[] = "8.8.8.8";
const char kDNSServer1[] = "8.8.4.4";
const char* kDNSServers[] = { kDNSServer0, kDNSServer1 };
} // namespace
class DNSServerTesterTest : public Test {
public:
DNSServerTesterTest()
: device_info_(
new NiceMock<MockDeviceInfo>(&control_, nullptr, nullptr, nullptr)),
connection_(new StrictMock<MockConnection>(device_info_.get())),
interface_name_(kInterfaceName),
dns_servers_(kDNSServers, kDNSServers + 2) {}
virtual void SetUp() {
EXPECT_CALL(*connection_.get(), interface_name())
.WillRepeatedly(ReturnRef(interface_name_));
dns_server_tester_.reset(
new DNSServerTester(connection_.get(),
&dispatcher_,
dns_servers_,
false,
callback_target_.result_callback()));
}
protected:
class CallbackTarget {
public:
CallbackTarget()
: result_callback_(Bind(&CallbackTarget::ResultCallback,
Unretained(this))) {
}
MOCK_METHOD1(ResultCallback, void(const DNSServerTester::Status status));
Callback<void(const DNSServerTester::Status)>& result_callback() {
return result_callback_;
}
private:
Callback<void(const DNSServerTester::Status)> result_callback_;
};
DNSServerTester* dns_server_tester() { return dns_server_tester_.get(); }
MockEventDispatcher& dispatcher() { return dispatcher_; }
CallbackTarget& callback_target() { return callback_target_; }
void ExpectReset() {
EXPECT_TRUE(callback_target_.result_callback().Equals(
dns_server_tester_->dns_result_callback_));
}
private:
StrictMock<MockEventDispatcher> dispatcher_;
MockControl control_;
std::unique_ptr<MockDeviceInfo> device_info_;
scoped_refptr<MockConnection> connection_;
CallbackTarget callback_target_;
const string interface_name_;
vector<string> dns_servers_;
std::unique_ptr<DNSServerTester> dns_server_tester_;
};
TEST_F(DNSServerTesterTest, Constructor) {
ExpectReset();
}
TEST_F(DNSServerTesterTest, StartAttempt) {
// Start attempt with no delay.
EXPECT_CALL(dispatcher(), PostDelayedTask(_, 0));
dns_server_tester()->StartAttempt(0);
// Start attempt with delay.
EXPECT_CALL(dispatcher(), PostDelayedTask(_, 100));
dns_server_tester()->StartAttempt(100);
}
TEST_F(DNSServerTesterTest, StartAttemptTask) {
// Setup mock DNS test client.
MockDNSClient* dns_test_client = new MockDNSClient();
dns_server_tester()->dns_test_client_.reset(dns_test_client);
// DNS test task started successfully.
EXPECT_CALL(*dns_test_client, Start(_, _)).WillOnce(Return(true));
EXPECT_CALL(callback_target(), ResultCallback(_)).Times(0);
dns_server_tester()->StartAttemptTask();
Mock::VerifyAndClearExpectations(dns_test_client);
// DNS test task failed to start.
EXPECT_CALL(*dns_test_client, Start(_, _)).WillOnce(Return(false));
EXPECT_CALL(callback_target(),
ResultCallback(DNSServerTester::kStatusFailure)).Times(1);
dns_server_tester()->StartAttemptTask();
Mock::VerifyAndClearExpectations(dns_test_client);
}
TEST_F(DNSServerTesterTest, AttemptCompleted) {
// DNS test attempt succeed with retry_until_success_ not set.
dns_server_tester()->retry_until_success_ = false;
EXPECT_CALL(callback_target(),
ResultCallback(DNSServerTester::kStatusSuccess)).Times(1);
dns_server_tester()->CompleteAttempt(DNSServerTester::kStatusSuccess);
// DNS test attempt succeed with retry_until_success_ being set.
dns_server_tester()->retry_until_success_ = true;
EXPECT_CALL(callback_target(),
ResultCallback(DNSServerTester::kStatusSuccess)).Times(1);
dns_server_tester()->CompleteAttempt(DNSServerTester::kStatusSuccess);
// DNS test attempt failed with retry_until_success_ not set.
dns_server_tester()->retry_until_success_ = false;
EXPECT_CALL(callback_target(),
ResultCallback(DNSServerTester::kStatusFailure)).Times(1);
dns_server_tester()->CompleteAttempt(DNSServerTester::kStatusFailure);
// DNS test attempt failed with retry_until_success_ being set.
dns_server_tester()->retry_until_success_ = true;
EXPECT_CALL(callback_target(), ResultCallback(_)).Times(0);
EXPECT_CALL(dispatcher(), PostDelayedTask(_, _)).Times(1);
dns_server_tester()->CompleteAttempt(DNSServerTester::kStatusFailure);
}
TEST_F(DNSServerTesterTest, StopAttempt) {
// Setup mock DNS test client.
MockDNSClient* dns_test_client = new MockDNSClient();
dns_server_tester()->dns_test_client_.reset(dns_test_client);
// DNS test task started successfully.
EXPECT_CALL(*dns_test_client, Start(_, _)).WillOnce(Return(true));
EXPECT_CALL(callback_target(), ResultCallback(_)).Times(0);
dns_server_tester()->StartAttemptTask();
Mock::VerifyAndClearExpectations(dns_test_client);
// Stop the DNS test attempt.
EXPECT_CALL(*dns_test_client, Stop()).Times(1);
EXPECT_CALL(callback_target(), ResultCallback(_)).Times(0);
dns_server_tester()->StopAttempt();
Mock::VerifyAndClearExpectations(dns_test_client);
}
} // namespace shill