blob: a2a28646a069830666d1e09c7881592f6a1eaf20 [file] [log] [blame]
// Copyright 2017 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 <string>
#include "base/command_line.h"
#include "base/files/file_path.h"
#include "base/files/scoped_temp_dir.h"
#include "base/macros.h"
#include "base/memory/ptr_util.h"
#include "base/run_loop.h"
#include "chrome/browser/chromeos/arc/arc_auth_notification.h"
#include "chrome/browser/chromeos/arc/arc_service_launcher.h"
#include "chrome/browser/chromeos/arc/arc_session_manager.h"
#include "chrome/browser/chromeos/arc/arc_util.h"
#include "chrome/browser/chromeos/arc/auth/arc_auth_context.h"
#include "chrome/browser/chromeos/arc/auth/arc_auth_service.h"
#include "chrome/browser/chromeos/arc/auth/arc_background_auth_code_fetcher.h"
#include "chrome/browser/chromeos/login/users/fake_chrome_user_manager.h"
#include "chrome/browser/chromeos/login/users/scoped_user_manager_enabler.h"
#include "chrome/browser/chromeos/profiles/profile_helper.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/signin/fake_profile_oauth2_token_service_builder.h"
#include "chrome/browser/signin/fake_signin_manager_builder.h"
#include "chrome/browser/signin/profile_oauth2_token_service_factory.h"
#include "chrome/browser/signin/signin_manager_factory.h"
#include "chrome/browser/ui/ash/multi_user/multi_user_util.h"
#include "chrome/test/base/in_process_browser_test.h"
#include "chrome/test/base/testing_profile.h"
#include "components/arc/arc_features.h"
#include "components/arc/arc_prefs.h"
#include "components/arc/arc_service_manager.h"
#include "components/arc/arc_session_runner.h"
#include "components/arc/arc_util.h"
#include "components/arc/test/fake_arc_session.h"
#include "components/prefs/pref_member.h"
#include "components/prefs/pref_service.h"
#include "components/signin/core/account_id/account_id.h"
#include "components/signin/core/browser/fake_profile_oauth2_token_service.h"
#include "components/user_manager/user_manager.h"
#include "net/url_request/test_url_fetcher_factory.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"
namespace {
constexpr char kRefreshToken[] = "fake-refresh-token";
constexpr char kFakeUserName[] = "test@example.com";
constexpr char kFakeGaiaId[] = "1234567890";
constexpr char kFakeAuthCode[] = "fake-auth-code";
} // namespace
namespace arc {
class FakeAuthInstance : public mojom::AuthInstance {
public:
// mojom::AuthInstance:
void Init(mojom::AuthHostPtr host) override { host_ = std::move(host); }
void OnAccountInfoReady(mojom::AccountInfoPtr account_info) override {
account_info_ = std::move(account_info);
base::ResetAndReturn(&done_closure_).Run();
}
void RequestAccountInfo(base::Closure done_closure) {
done_closure_ = done_closure;
host_->RequestAccountInfo();
}
mojom::AccountInfo* account_info() { return account_info_.get(); }
private:
mojom::AuthHostPtr host_;
mojom::AccountInfoPtr account_info_;
base::Closure done_closure_;
};
class ArcAuthServiceTest : public InProcessBrowserTest {
protected:
ArcAuthServiceTest() = default;
// InProcessBrowserTest:
~ArcAuthServiceTest() override = default;
void SetUpCommandLine(base::CommandLine* command_line) override {
arc::SetArcAvailableCommandLineForTesting(command_line);
}
void SetUpOnMainThread() override {
user_manager_enabler_ =
base::MakeUnique<chromeos::ScopedUserManagerEnabler>(
new chromeos::FakeChromeUserManager());
// Init ArcSessionManager for testing.
ArcServiceLauncher::Get()->ResetForTesting();
ArcSessionManager::DisableUIForTesting();
ArcAuthNotification::DisableForTesting();
ArcSessionManager::EnableCheckAndroidManagementForTesting();
ArcSessionManager::Get()->SetArcSessionRunnerForTesting(
base::MakeUnique<ArcSessionRunner>(base::Bind(FakeArcSession::Create)));
chromeos::ProfileHelper::SetAlwaysReturnPrimaryUserForTesting(true);
ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
const AccountId account_id(
AccountId::FromUserEmailGaiaId(kFakeUserName, kFakeGaiaId));
GetFakeUserManager()->AddUser(account_id);
GetFakeUserManager()->LoginUser(account_id);
GetFakeUserManager()->CreateLocalState();
// Create test profile.
TestingProfile::Builder profile_builder;
profile_builder.SetPath(temp_dir_.GetPath().AppendASCII("TestArcProfile"));
profile_builder.SetProfileName(kFakeUserName);
profile_builder.AddTestingFactory(
ProfileOAuth2TokenServiceFactory::GetInstance(),
BuildFakeProfileOAuth2TokenService);
profile_builder.AddTestingFactory(SigninManagerFactory::GetInstance(),
BuildFakeSigninManagerBase);
profile_ = profile_builder.Build();
FakeProfileOAuth2TokenService* token_service =
static_cast<FakeProfileOAuth2TokenService*>(
ProfileOAuth2TokenServiceFactory::GetForProfile(profile()));
token_service->UpdateCredentials(kFakeUserName, kRefreshToken);
token_service->set_auto_post_fetch_response_on_message_loop(true);
FakeSigninManagerBase* signin_manager = static_cast<FakeSigninManagerBase*>(
SigninManagerFactory::GetForProfile(profile()));
signin_manager->SetAuthenticatedAccountInfo(kFakeGaiaId, kFakeUserName);
profile()->GetPrefs()->SetBoolean(prefs::kArcSignedIn, true);
profile()->GetPrefs()->SetBoolean(prefs::kArcTermsAccepted, true);
ArcServiceLauncher::Get()->OnPrimaryUserProfilePrepared(profile());
// It is non-trivial to navigate through the merge session in a testing
// context; currently we just skip it.
// TODO(blundell): Figure out how to enable this flow.
ArcSessionManager::Get()->auth_context()->SkipMergeSessionForTesting();
}
void TearDownOnMainThread() override {
// Explicitly removing the user is required; otherwise ProfileHelper keeps
// a dangling pointer to the User.
// TODO(nya): Consider removing all users from ProfileHelper in the
// destructor of FakeChromeUserManager.
const AccountId account_id(
AccountId::FromUserEmailGaiaId(kFakeUserName, kFakeGaiaId));
GetFakeUserManager()->RemoveUserFromList(account_id);
// Since ArcServiceLauncher is (re-)set up with profile() in
// SetUpOnMainThread() it is necessary to Shutdown() before the profile()
// is destroyed. ArcServiceLauncher::Shutdown() will be called again on
// fixture destruction (because it is initialized with the original Profile
// instance in fixture, once), but it should be no op.
// TODO(hidehiko): Think about a way to test the code cleanly.
ArcServiceLauncher::Get()->Shutdown();
profile_.reset();
user_manager_enabler_.reset();
chromeos::ProfileHelper::SetAlwaysReturnPrimaryUserForTesting(false);
}
chromeos::FakeChromeUserManager* GetFakeUserManager() const {
return static_cast<chromeos::FakeChromeUserManager*>(
user_manager::UserManager::Get());
}
void set_profile_name(const std::string& username) {
profile_->set_profile_name(username);
}
Profile* profile() { return profile_.get(); }
private:
std::unique_ptr<chromeos::ScopedUserManagerEnabler> user_manager_enabler_;
base::ScopedTempDir temp_dir_;
std::unique_ptr<TestingProfile> profile_;
DISALLOW_COPY_AND_ASSIGN(ArcAuthServiceTest);
};
// Tests that when ARC requests account info for a non-managed account and
// the system is in silent authentication mode, Chrome supplies the info
// configured in SetUpOnMainThread() above.
IN_PROC_BROWSER_TEST_F(ArcAuthServiceTest, SuccessfulBackgroundFetch) {
net::FakeURLFetcherFactory factory(nullptr);
factory.SetFakeResponse(
GURL(arc::kAuthTokenExchangeEndPoint),
"{ \"token\" : \"" + std::string(kFakeAuthCode) + "\" }", net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
FakeAuthInstance auth_instance;
ArcAuthService* auth_service =
ArcAuthService::GetForBrowserContext(profile());
ASSERT_TRUE(auth_service);
ArcBridgeService* arc_bridge_service =
ArcServiceManager::Get()->arc_bridge_service();
ASSERT_TRUE(arc_bridge_service);
arc_bridge_service->auth()->SetInstance(&auth_instance);
base::RunLoop run_loop;
auth_instance.RequestAccountInfo(run_loop.QuitClosure());
run_loop.Run();
EXPECT_TRUE(auth_instance.account_info());
EXPECT_EQ(kFakeUserName, auth_instance.account_info()->account_name.value());
EXPECT_EQ(kFakeAuthCode, auth_instance.account_info()->auth_code.value());
EXPECT_EQ(mojom::ChromeAccountType::USER_ACCOUNT,
auth_instance.account_info()->account_type);
EXPECT_FALSE(auth_instance.account_info()->enrollment_token);
EXPECT_FALSE(auth_instance.account_info()->is_managed);
}
} // namespace arc