From 707b9a58cc991a7a7d120f719701f06181753a0e Mon Sep 17 00:00:00 2001 From: Shelley Vohr Date: Fri, 22 Mar 2024 14:00:21 +0100 Subject: [PATCH] fix: `app.setLoginItemSettings` error getting swallowed by gin conversion (#41647) * fix: errors getting swallowed by gin conversion * fix: Windows build * test: re-enable disabled test --- filenames.gni | 2 + shell/browser/api/electron_api_app.cc | 79 +---------------- shell/browser/api/electron_api_app.h | 2 +- shell/browser/browser.cc | 21 +++-- shell/browser/browser.h | 87 ++++++++++--------- shell/browser/browser_linux.cc | 7 +- shell/browser/browser_mac.mm | 17 ++-- shell/browser/browser_win.cc | 22 ++--- .../login_item_settings_converter.cc | 86 ++++++++++++++++++ .../login_item_settings_converter.h | 39 +++++++++ spec/api-app-spec.ts | 3 +- 11 files changed, 211 insertions(+), 154 deletions(-) create mode 100644 shell/common/gin_converters/login_item_settings_converter.cc create mode 100644 shell/common/gin_converters/login_item_settings_converter.h diff --git a/filenames.gni b/filenames.gni index 740b27375a3..c7bb2f8389d 100644 --- a/filenames.gni +++ b/filenames.gni @@ -583,6 +583,8 @@ filenames = { "shell/common/gin_converters/hid_device_info_converter.h", "shell/common/gin_converters/image_converter.cc", "shell/common/gin_converters/image_converter.h", + "shell/common/gin_converters/login_item_settings_converter.cc", + "shell/common/gin_converters/login_item_settings_converter.h", "shell/common/gin_converters/media_converter.cc", "shell/common/gin_converters/media_converter.h", "shell/common/gin_converters/message_box_converter.cc", diff --git a/shell/browser/api/electron_api_app.cc b/shell/browser/api/electron_api_app.cc index e7e2dfd0a6c..cad66ef5e81 100644 --- a/shell/browser/api/electron_api_app.cc +++ b/shell/browser/api/electron_api_app.cc @@ -89,7 +89,6 @@ #if BUILDFLAG(IS_MAC) #include -#include "base/mac/mac_util.h" #include "shell/browser/ui/cocoa/electron_bundle_mover.h" #endif @@ -324,80 +323,6 @@ struct Converter { }; #endif -#if BUILDFLAG(IS_WIN) -template <> -struct Converter { - static bool FromV8(v8::Isolate* isolate, - v8::Local val, - Browser::LaunchItem* out) { - gin_helper::Dictionary dict; - if (!ConvertFromV8(isolate, val, &dict)) - return false; - - dict.Get("name", &(out->name)); - dict.Get("path", &(out->path)); - dict.Get("args", &(out->args)); - dict.Get("scope", &(out->scope)); - dict.Get("enabled", &(out->enabled)); - return true; - } - - static v8::Local ToV8(v8::Isolate* isolate, - Browser::LaunchItem val) { - auto dict = gin_helper::Dictionary::CreateEmpty(isolate); - dict.Set("name", val.name); - dict.Set("path", val.path); - dict.Set("args", val.args); - dict.Set("scope", val.scope); - dict.Set("enabled", val.enabled); - return dict.GetHandle(); - } -}; -#endif - -template <> -struct Converter { - static bool FromV8(v8::Isolate* isolate, - v8::Local val, - Browser::LoginItemSettings* out) { - gin_helper::Dictionary dict; - if (!ConvertFromV8(isolate, val, &dict)) - return false; - - dict.Get("openAtLogin", &(out->open_at_login)); - dict.Get("openAsHidden", &(out->open_as_hidden)); - dict.Get("path", &(out->path)); - dict.Get("args", &(out->args)); -#if BUILDFLAG(IS_WIN) - dict.Get("name", &(out->name)); - dict.Get("enabled", &(out->enabled)); -#elif BUILDFLAG(IS_MAC) - dict.Get("serviceName", &(out->service_name)); - dict.Get("type", &(out->type)); -#endif - return true; - } - - static v8::Local ToV8(v8::Isolate* isolate, - Browser::LoginItemSettings val) { - auto dict = gin_helper::Dictionary::CreateEmpty(isolate); -#if BUILDFLAG(IS_WIN) - dict.Set("launchItems", val.launch_items); - dict.Set("executableWillLaunchAtLogin", - val.executable_will_launch_at_login); -#elif BUILDFLAG(IS_MAC) - if (base::mac::MacOSMajorVersion() >= 13) - dict.Set("status", val.status); -#endif - dict.Set("openAtLogin", val.open_at_login); - dict.Set("openAsHidden", val.open_as_hidden); - dict.Set("restoreState", val.restore_state); - dict.Set("wasOpenedAtLogin", val.opened_at_login); - dict.Set("wasOpenedAsHidden", val.opened_as_hidden); - return dict.GetHandle(); - } -}; - template <> struct Converter { static bool FromV8(v8::Isolate* isolate, @@ -1216,8 +1141,8 @@ void App::SetAccessibilitySupportEnabled(gin_helper::ErrorThrower thrower, Browser::Get()->OnAccessibilitySupportChanged(); } -Browser::LoginItemSettings App::GetLoginItemSettings(gin::Arguments* args) { - Browser::LoginItemSettings options; +v8::Local App::GetLoginItemSettings(gin::Arguments* args) { + LoginItemSettings options; args->GetNext(&options); return Browser::Get()->GetLoginItemSettings(options); } diff --git a/shell/browser/api/electron_api_app.h b/shell/browser/api/electron_api_app.h index 124427cbae1..be9b7af97b2 100644 --- a/shell/browser/api/electron_api_app.h +++ b/shell/browser/api/electron_api_app.h @@ -206,7 +206,7 @@ class App : public ElectronBrowserClient::Delegate, bool IsAccessibilitySupportEnabled(); void SetAccessibilitySupportEnabled(gin_helper::ErrorThrower thrower, bool enabled); - Browser::LoginItemSettings GetLoginItemSettings(gin::Arguments* args); + v8::Local GetLoginItemSettings(gin::Arguments* args); #if BUILDFLAG(USE_NSS_CERTS) void ImportCertificate(gin_helper::ErrorThrower thrower, base::Value options, diff --git a/shell/browser/browser.cc b/shell/browser/browser.cc index 2d8a8317656..be2fe442392 100644 --- a/shell/browser/browser.cc +++ b/shell/browser/browser.cc @@ -27,6 +27,16 @@ namespace electron { +LoginItemSettings::LoginItemSettings() = default; +LoginItemSettings::~LoginItemSettings() = default; +LoginItemSettings::LoginItemSettings(const LoginItemSettings& other) = default; + +#if BUILDFLAG(IS_WIN) +LaunchItem::LaunchItem() = default; +LaunchItem::~LaunchItem() = default; +LaunchItem::LaunchItem(const LaunchItem& other) = default; +#endif + namespace { // Call |quit| after Chromium is fully started. @@ -43,17 +53,6 @@ void RunQuitClosure(base::OnceClosure quit) { } // namespace -#if BUILDFLAG(IS_WIN) -Browser::LaunchItem::LaunchItem() = default; -Browser::LaunchItem::~LaunchItem() = default; -Browser::LaunchItem::LaunchItem(const LaunchItem& other) = default; -#endif - -Browser::LoginItemSettings::LoginItemSettings() = default; -Browser::LoginItemSettings::~LoginItemSettings() = default; -Browser::LoginItemSettings::LoginItemSettings(const LoginItemSettings& other) = - default; - Browser::Browser() { WindowList::AddObserver(this); } diff --git a/shell/browser/browser.h b/shell/browser/browser.h index cb82f9a061a..ea08eec4a1f 100644 --- a/shell/browser/browser.h +++ b/shell/browser/browser.h @@ -17,6 +17,7 @@ #include "gin/dictionary.h" #include "shell/browser/browser_observer.h" #include "shell/browser/window_list_observer.h" +#include "shell/common/gin_converters/login_item_settings_converter.h" #include "shell/common/gin_helper/promise.h" #if BUILDFLAG(IS_WIN) @@ -41,6 +42,48 @@ namespace electron { class ElectronMenuModel; +#if BUILDFLAG(IS_WIN) +struct LaunchItem { + std::wstring name; + std::wstring path; + std::wstring scope; + std::vector args; + bool enabled = true; + + LaunchItem(); + ~LaunchItem(); + LaunchItem(const LaunchItem&); +}; +#endif + +struct LoginItemSettings { + bool open_at_login = false; + bool open_as_hidden = false; + bool restore_state = false; + bool opened_at_login = false; + bool opened_as_hidden = false; + std::u16string path; + std::vector args; + +#if BUILDFLAG(IS_MAC) + std::string type = "mainAppService"; + std::string service_name; + std::string status; +#elif BUILDFLAG(IS_WIN) + // used in browser::setLoginItemSettings + bool enabled = true; + std::wstring name; + + // used in browser::getLoginItemSettings + bool executable_will_launch_at_login = false; + std::vector launch_items; +#endif + + LoginItemSettings(); + ~LoginItemSettings(); + LoginItemSettings(const LoginItemSettings&); +}; + // This class is used for control application-wide operations. class Browser : public WindowListObserver { public: @@ -112,50 +155,8 @@ class Browser : public WindowListObserver { bool SetBadgeCount(std::optional count); [[nodiscard]] int badge_count() const { return badge_count_; } -#if BUILDFLAG(IS_WIN) - struct LaunchItem { - std::wstring name; - std::wstring path; - std::wstring scope; - std::vector args; - bool enabled = true; - - LaunchItem(); - ~LaunchItem(); - LaunchItem(const LaunchItem&); - }; -#endif - - // Set/Get the login item settings of the app - struct LoginItemSettings { - bool open_at_login = false; - bool open_as_hidden = false; - bool restore_state = false; - bool opened_at_login = false; - bool opened_as_hidden = false; - std::u16string path; - std::vector args; - -#if BUILDFLAG(IS_MAC) - std::string type = "mainAppService"; - std::string service_name; - std::string status; -#elif BUILDFLAG(IS_WIN) - // used in browser::setLoginItemSettings - bool enabled = true; - std::wstring name; - - // used in browser::getLoginItemSettings - bool executable_will_launch_at_login = false; - std::vector launch_items; -#endif - - LoginItemSettings(); - ~LoginItemSettings(); - LoginItemSettings(const LoginItemSettings&); - }; void SetLoginItemSettings(LoginItemSettings settings); - LoginItemSettings GetLoginItemSettings(const LoginItemSettings& options); + v8::Local GetLoginItemSettings(const LoginItemSettings& options); #if BUILDFLAG(IS_MAC) // Set the handler which decides whether to shutdown. diff --git a/shell/browser/browser_linux.cc b/shell/browser/browser_linux.cc index fc297546c47..69c6d1c835a 100644 --- a/shell/browser/browser_linux.cc +++ b/shell/browser/browser_linux.cc @@ -11,9 +11,11 @@ #include "base/environment.h" #include "base/process/launch.h" #include "electron/electron_version.h" +#include "shell/browser/javascript_environment.h" #include "shell/browser/native_window.h" #include "shell/browser/window_list.h" #include "shell/common/application_info.h" +#include "shell/common/gin_converters/login_item_settings_converter.h" #include "shell/common/thread_restrictions.h" #if BUILDFLAG(IS_LINUX) @@ -138,9 +140,10 @@ bool Browser::SetBadgeCount(std::optional count) { void Browser::SetLoginItemSettings(LoginItemSettings settings) {} -Browser::LoginItemSettings Browser::GetLoginItemSettings( +v8::Local Browser::GetLoginItemSettings( const LoginItemSettings& options) { - return LoginItemSettings(); + LoginItemSettings settings; + return gin::ConvertToV8(JavascriptEnvironment::GetIsolate(), settings); } std::string Browser::GetExecutableFileVersion() const { diff --git a/shell/browser/browser_mac.mm b/shell/browser/browser_mac.mm index 2a7f6dc96db..c42d2c6cabd 100644 --- a/shell/browser/browser_mac.mm +++ b/shell/browser/browser_mac.mm @@ -29,6 +29,7 @@ #include "shell/common/api/electron_api_native_image.h" #include "shell/common/application_info.h" #include "shell/common/gin_converters/image_converter.h" +#include "shell/common/gin_converters/login_item_settings_converter.h" #include "shell/common/gin_helper/arguments.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/gin_helper/error_thrower.h" @@ -88,8 +89,8 @@ bool CheckLoginItemStatus(bool* is_hidden) { return true; } -Browser::LoginItemSettings GetLoginItemSettingsDeprecated() { - Browser::LoginItemSettings settings; +LoginItemSettings GetLoginItemSettingsDeprecated() { + LoginItemSettings settings; settings.open_at_login = CheckLoginItemStatus(&settings.open_as_hidden); settings.restore_state = base::mac::WasLaunchedAsLoginItemRestoreState(); settings.opened_at_login = base::mac::WasLaunchedAsLoginOrResumeItem(); @@ -375,13 +376,15 @@ void Browser::ApplyForcedRTL() { } } -Browser::LoginItemSettings Browser::GetLoginItemSettings( +v8::Local Browser::GetLoginItemSettings( const LoginItemSettings& options) { LoginItemSettings settings; + v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); + if (options.type != "mainAppService" && options.service_name.empty()) { - gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate()) - .ThrowTypeError("'name' is required when type is not mainAppService"); - return settings; + gin_helper::ErrorThrower(isolate).ThrowTypeError( + "'name' is required when type is not mainAppService"); + return v8::Local(); } #if IS_MAS_BUILD() @@ -408,7 +411,7 @@ Browser::LoginItemSettings Browser::GetLoginItemSettings( settings = settings_deprecated; } #endif - return settings; + return gin::ConvertToV8(isolate, settings); } void Browser::SetLoginItemSettings(LoginItemSettings settings) { diff --git a/shell/browser/browser_win.cc b/shell/browser/browser_win.cc index e6ac9b7fddf..6f8315c32f2 100644 --- a/shell/browser/browser_win.cc +++ b/shell/browser/browser_win.cc @@ -30,12 +30,14 @@ #include "shell/browser/api/electron_api_app.h" #include "shell/browser/badging/badge_manager.h" #include "shell/browser/electron_browser_main_parts.h" +#include "shell/browser/javascript_environment.h" #include "shell/browser/ui/message_box.h" #include "shell/browser/ui/win/jump_list.h" #include "shell/browser/window_list.h" #include "shell/common/application_info.h" #include "shell/common/gin_converters/file_path_converter.h" #include "shell/common/gin_converters/image_converter.h" +#include "shell/common/gin_converters/login_item_settings_converter.h" #include "shell/common/gin_helper/arguments.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/skia_util.h" @@ -154,12 +156,12 @@ bool FormatCommandLineString(std::wstring* exe, // a list of launchItem with matching paths to our application. // if a launchItem with a matching path also has a matching entry within the // startup_approved_key_path, set executable_will_launch_at_login to be `true` -std::vector GetLoginItemSettingsHelper( +std::vector GetLoginItemSettingsHelper( base::win::RegistryValueIterator* it, boolean* executable_will_launch_at_login, std::wstring scope, - const Browser::LoginItemSettings& options) { - std::vector launch_items; + const LoginItemSettings& options) { + std::vector launch_items; base::FilePath lookup_exe_path; if (options.path.empty()) { @@ -183,7 +185,7 @@ std::vector GetLoginItemSettingsHelper( // add launch item to vector if it has a matching path (case-insensitive) if (exe_match) { - Browser::LaunchItem launch_item; + LaunchItem launch_item; launch_item.name = it->Name(); launch_item.path = registry_launch_path.value(); launch_item.args = registry_launch_cmd.GetArgs(); @@ -656,7 +658,7 @@ void Browser::SetLoginItemSettings(LoginItemSettings settings) { } } -Browser::LoginItemSettings Browser::GetLoginItemSettings( +v8::Local Browser::GetLoginItemSettings( const LoginItemSettings& options) { LoginItemSettings settings; std::wstring keyPath = L"Software\\Microsoft\\Windows\\CurrentVersion\\Run"; @@ -675,7 +677,7 @@ Browser::LoginItemSettings Browser::GetLoginItemSettings( // if there exists a launch entry with property enabled=='true', // set executable_will_launch_at_login to 'true'. boolean executable_will_launch_at_login = false; - std::vector launch_items; + std::vector launch_items; base::win::RegistryValueIterator hkcu_iterator(HKEY_CURRENT_USER, keyPath.c_str()); base::win::RegistryValueIterator hklm_iterator(HKEY_LOCAL_MACHINE, @@ -683,16 +685,14 @@ Browser::LoginItemSettings Browser::GetLoginItemSettings( launch_items = GetLoginItemSettingsHelper( &hkcu_iterator, &executable_will_launch_at_login, L"user", options); - std::vector launch_items_hklm = - GetLoginItemSettingsHelper(&hklm_iterator, - &executable_will_launch_at_login, L"machine", - options); + std::vector launch_items_hklm = GetLoginItemSettingsHelper( + &hklm_iterator, &executable_will_launch_at_login, L"machine", options); launch_items.insert(launch_items.end(), launch_items_hklm.begin(), launch_items_hklm.end()); settings.executable_will_launch_at_login = executable_will_launch_at_login; settings.launch_items = launch_items; - return settings; + return gin::ConvertToV8(JavascriptEnvironment::GetIsolate(), settings); } PCWSTR Browser::GetAppUserModelID() { diff --git a/shell/common/gin_converters/login_item_settings_converter.cc b/shell/common/gin_converters/login_item_settings_converter.cc new file mode 100644 index 00000000000..d0be574ff9e --- /dev/null +++ b/shell/common/gin_converters/login_item_settings_converter.cc @@ -0,0 +1,86 @@ +// Copyright (c) 2024 Microsoft, GmbH. +// Use of this source code is governed by the MIT license that can be +// found in the LICENSE file. + +#include "shell/common/gin_converters/login_item_settings_converter.h" + +#if BUILDFLAG(IS_MAC) +#include "base/mac/mac_util.h" +#endif + +#include "shell/browser/browser.h" +#include "shell/common/gin_helper/dictionary.h" + +namespace gin { + +#if BUILDFLAG(IS_WIN) +bool Converter::FromV8(v8::Isolate* isolate, + v8::Local val, + electron::LaunchItem* out) { + gin_helper::Dictionary dict; + if (!ConvertFromV8(isolate, val, &dict)) + return false; + + dict.Get("name", &(out->name)); + dict.Get("path", &(out->path)); + dict.Get("args", &(out->args)); + dict.Get("scope", &(out->scope)); + dict.Get("enabled", &(out->enabled)); + return true; +} + +v8::Local Converter::ToV8( + v8::Isolate* isolate, + electron::LaunchItem val) { + auto dict = gin_helper::Dictionary::CreateEmpty(isolate); + dict.Set("name", val.name); + dict.Set("path", val.path); + dict.Set("args", val.args); + dict.Set("scope", val.scope); + dict.Set("enabled", val.enabled); + return dict.GetHandle(); +} +#endif + +bool Converter::FromV8( + v8::Isolate* isolate, + v8::Local val, + electron::LoginItemSettings* out) { + gin_helper::Dictionary dict; + if (!ConvertFromV8(isolate, val, &dict)) + return false; + + dict.Get("openAtLogin", &(out->open_at_login)); + dict.Get("openAsHidden", &(out->open_as_hidden)); + dict.Get("path", &(out->path)); + dict.Get("args", &(out->args)); +#if BUILDFLAG(IS_WIN) + dict.Get("name", &(out->name)); + dict.Get("enabled", &(out->enabled)); +#elif BUILDFLAG(IS_MAC) + dict.Get("serviceName", &(out->service_name)); + dict.Get("type", &(out->type)); +#endif + return true; +} + +v8::Local Converter::ToV8( + v8::Isolate* isolate, + electron::LoginItemSettings val) { + auto dict = gin_helper::Dictionary::CreateEmpty(isolate); +#if BUILDFLAG(IS_WIN) + dict.Set("launchItems", val.launch_items); + dict.Set("executableWillLaunchAtLogin", val.executable_will_launch_at_login); +#elif BUILDFLAG(IS_MAC) + if (base::mac::MacOSMajorVersion() >= 13) + dict.Set("status", val.status); +#endif + dict.Set("openAtLogin", val.open_at_login); + dict.Set("openAsHidden", val.open_as_hidden); + dict.Set("restoreState", val.restore_state); + dict.Set("wasOpenedAtLogin", val.opened_at_login); + dict.Set("wasOpenedAsHidden", val.opened_as_hidden); + return dict.GetHandle(); +} + +} // namespace gin diff --git a/shell/common/gin_converters/login_item_settings_converter.h b/shell/common/gin_converters/login_item_settings_converter.h new file mode 100644 index 00000000000..6170b9e2be9 --- /dev/null +++ b/shell/common/gin_converters/login_item_settings_converter.h @@ -0,0 +1,39 @@ +// Copyright (c) 2024 Microsoft, GmbH. +// Use of this source code is governed by the MIT license that can be +// found in the LICENSE file. + +#ifndef ELECTRON_SHELL_COMMON_GIN_CONVERTERS_LOGIN_ITEM_SETTINGS_CONVERTER_H_ +#define ELECTRON_SHELL_COMMON_GIN_CONVERTERS_LOGIN_ITEM_SETTINGS_CONVERTER_H_ + +#include "gin/converter.h" + +namespace electron { +struct LoginItemSettings; +struct LaunchItem; +} // namespace electron + +namespace gin { + +#if BUILDFLAG(IS_WIN) +template <> +struct Converter { + static v8::Local ToV8(v8::Isolate* isolate, + electron::LaunchItem val); + static bool FromV8(v8::Isolate* isolate, + v8::Local val, + electron::LaunchItem* out); +}; +#endif + +template <> +struct Converter { + static v8::Local ToV8(v8::Isolate* isolate, + electron::LoginItemSettings val); + static bool FromV8(v8::Isolate* isolate, + v8::Local val, + electron::LoginItemSettings* out); +}; + +} // namespace gin + +#endif // ELECTRON_SHELL_COMMON_GIN_CONVERTERS_LOGIN_ITEM_SETTINGS_CONVERTER_H_ diff --git a/spec/api-app-spec.ts b/spec/api-app-spec.ts index c0b01fd41ab..4b3f9d185c1 100644 --- a/spec/api-app-spec.ts +++ b/spec/api-app-spec.ts @@ -760,8 +760,7 @@ describe('app module', () => { }).to.throw(/'name' is required when type is not mainAppService/); }); - // TODO this test does not work on CircleCI arm64 macs - ifit(isVenturaOrHigher && process.arch !== 'arm64')('throws when getting non-default type with no name', () => { + ifit(isVenturaOrHigher)('throws when getting non-default type with no name', () => { expect(() => { app.getLoginItemSettings({ type: 'daemonService'