blob: a87262cd0e451d70fe275014bd863f3d3e9f5a8f [file] [log] [blame]
// Copyright (c) 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 "chrome/browser/chrome_process_singleton.h"
#include "base/bind.h"
#include "base/command_line.h"
#include "base/compiler_specific.h"
#include "base/files/file_path.h"
#include "base/files/scoped_temp_dir.h"
#include "build/build_config.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace {
bool ServerCallback(int* callback_count,
const base::CommandLine& command_line,
const base::FilePath& current_directory) {
++(*callback_count);
return true;
}
bool ClientCallback(const base::CommandLine& command_line,
const base::FilePath& current_directory) {
ADD_FAILURE();
return false;
}
} // namespace
TEST(ChromeProcessSingletonTest, Basic) {
base::ScopedTempDir profile_dir;
ASSERT_TRUE(profile_dir.CreateUniqueTempDir());
int callback_count = 0;
ChromeProcessSingleton ps1(
profile_dir.GetPath(),
base::Bind(&ServerCallback, base::Unretained(&callback_count)));
ps1.Unlock();
ChromeProcessSingleton ps2(profile_dir.GetPath(),
base::Bind(&ClientCallback));
ps2.Unlock();
ProcessSingleton::NotifyResult result = ps1.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NONE, result);
ASSERT_EQ(0, callback_count);
result = ps2.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NOTIFIED, result);
ASSERT_EQ(1, callback_count);
}
TEST(ChromeProcessSingletonTest, Lock) {
base::ScopedTempDir profile_dir;
ASSERT_TRUE(profile_dir.CreateUniqueTempDir());
int callback_count = 0;
ChromeProcessSingleton ps1(
profile_dir.GetPath(),
base::Bind(&ServerCallback, base::Unretained(&callback_count)));
ChromeProcessSingleton ps2(profile_dir.GetPath(),
base::Bind(&ClientCallback));
ps2.Unlock();
ProcessSingleton::NotifyResult result = ps1.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NONE, result);
ASSERT_EQ(0, callback_count);
result = ps2.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NOTIFIED, result);
ASSERT_EQ(0, callback_count);
ps1.Unlock();
ASSERT_EQ(1, callback_count);
}
#if defined(OS_WIN) && !defined(USE_AURA)
namespace {
void ModalNotificationHandler(bool* flag) {
*flag = true;
}
} // namespace
TEST(ChromeProcessSingletonTest, LockWithModalDialog) {
base::ScopedTempDir profile_dir;
ASSERT_TRUE(profile_dir.CreateUniqueTempDir());
int callback_count = 0;
bool called_modal_notification_handler = false;
ChromeProcessSingleton ps1(
profile_dir.GetPath(),
base::Bind(&ServerCallback, base::Unretained(&callback_count)));
ps1.SetModalDialogNotificationHandler(
base::Bind(&ModalNotificationHandler,
base::Unretained(&called_modal_notification_handler)));
ChromeProcessSingleton ps2(profile_dir.GetPath(),
base::Bind(&ClientCallback));
ps2.Unlock();
ProcessSingleton::NotifyResult result = ps1.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NONE, result);
ASSERT_EQ(0, callback_count);
ASSERT_FALSE(called_modal_notification_handler);
result = ps2.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NOTIFIED, result);
ASSERT_TRUE(called_modal_notification_handler);
ASSERT_EQ(0, callback_count);
ps1.SetModalDialogNotificationHandler(base::Closure());
ps1.Unlock();
// The notifications sent while a modal dialog was open were processed after
// unlock.
ASSERT_EQ(2, callback_count);
// And now that the handler was cleared notifications will still be handled.
result = ps2.NotifyOtherProcessOrCreate();
ASSERT_EQ(ProcessSingleton::PROCESS_NOTIFIED, result);
ASSERT_EQ(3, callback_count);
}
#endif // defined(OS_WIN) && !defined(USE_AURA)