Merge branch 'master' into use-dot-env
This commit is contained in:
commit
6ea0e73b47
37 changed files with 506 additions and 308 deletions
30
BUILD.gn
30
BUILD.gn
|
@ -217,7 +217,8 @@ static_library("electron_lib") {
|
||||||
"//components/printing/common",
|
"//components/printing/common",
|
||||||
"//components/security_state/content",
|
"//components/security_state/content",
|
||||||
"//components/viz/service",
|
"//components/viz/service",
|
||||||
"//content/public/browser",
|
"//content/public/app:both",
|
||||||
|
"//content/public/child",
|
||||||
"//device/geolocation",
|
"//device/geolocation",
|
||||||
"//gin",
|
"//gin",
|
||||||
"//net:net_resources",
|
"//net:net_resources",
|
||||||
|
@ -240,6 +241,14 @@ static_library("electron_lib") {
|
||||||
"brightray",
|
"brightray",
|
||||||
"build/node",
|
"build/node",
|
||||||
]
|
]
|
||||||
|
include_dirs = [
|
||||||
|
"chromium_src",
|
||||||
|
".",
|
||||||
|
"$target_gen_dir",
|
||||||
|
# TODO(nornagon): replace usage of SchemeRegistry by an actually exported
|
||||||
|
# API of blink, then delete this include dir.
|
||||||
|
"//third_party/WebKit/Source",
|
||||||
|
]
|
||||||
if (enable_desktop_capturer) {
|
if (enable_desktop_capturer) {
|
||||||
deps += [ "//third_party/webrtc/modules/desktop_capture" ]
|
deps += [ "//third_party/webrtc/modules/desktop_capture" ]
|
||||||
}
|
}
|
||||||
|
@ -274,20 +283,15 @@ static_library("electron_lib") {
|
||||||
defines = [
|
defines = [
|
||||||
# Disable warnings for g_settings_list_schemas.
|
# Disable warnings for g_settings_list_schemas.
|
||||||
"GLIB_DISABLE_DEPRECATION_WARNINGS",
|
"GLIB_DISABLE_DEPRECATION_WARNINGS",
|
||||||
|
]
|
||||||
|
if (is_component_build) {
|
||||||
|
defines += [
|
||||||
# Import V8 symbols from shared library (node.dll / libnode.so)
|
# Import V8 symbols from shared library (node.dll / libnode.so)
|
||||||
"USING_V8_SHARED",
|
"USING_V8_SHARED",
|
||||||
"USING_V8_PLATFORM_SHARED",
|
"USING_V8_PLATFORM_SHARED",
|
||||||
"USING_V8_BASE_SHARED",
|
"USING_V8_BASE_SHARED",
|
||||||
]
|
]
|
||||||
include_dirs = [
|
}
|
||||||
"chromium_src",
|
|
||||||
".",
|
|
||||||
"$target_gen_dir",
|
|
||||||
# TODO(nornagon): replace usage of SchemeRegistry by an actually exported
|
|
||||||
# API of blink, then delete this include dir.
|
|
||||||
"//third_party/WebKit/Source",
|
|
||||||
]
|
|
||||||
if (is_linux || is_win) {
|
if (is_linux || is_win) {
|
||||||
deps += [ "//third_party/breakpad:client" ]
|
deps += [ "//third_party/breakpad:client" ]
|
||||||
include_dirs += [
|
include_dirs += [
|
||||||
|
@ -433,6 +437,7 @@ if (is_mac) {
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (is_component_build) {
|
||||||
bundle_data("electron_framework_libraries") {
|
bundle_data("electron_framework_libraries") {
|
||||||
public_deps = [ "build/node" ]
|
public_deps = [ "build/node" ]
|
||||||
sources = [
|
sources = [
|
||||||
|
@ -442,6 +447,7 @@ if (is_mac) {
|
||||||
"{{bundle_contents_dir}}/Libraries/{{source_file_part}}"
|
"{{bundle_contents_dir}}/Libraries/{{source_file_part}}"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bundle_data("electron_crashpad_helper") {
|
bundle_data("electron_crashpad_helper") {
|
||||||
sources = [
|
sources = [
|
||||||
|
@ -465,13 +471,15 @@ if (is_mac) {
|
||||||
deps = [
|
deps = [
|
||||||
"//base",
|
"//base",
|
||||||
"//base:i18n",
|
"//base:i18n",
|
||||||
":electron_framework_libraries",
|
|
||||||
":electron_framework_resources",
|
":electron_framework_resources",
|
||||||
":electron_xibs",
|
":electron_xibs",
|
||||||
]
|
]
|
||||||
if (!is_mas_build) {
|
if (!is_mas_build) {
|
||||||
deps += [ ":electron_crashpad_helper" ]
|
deps += [ ":electron_crashpad_helper" ]
|
||||||
}
|
}
|
||||||
|
if (is_component_build) {
|
||||||
|
deps += [ ":electron_framework_libraries" ]
|
||||||
|
}
|
||||||
info_plist = "atom/common/resources/mac/Info.plist"
|
info_plist = "atom/common/resources/mac/Info.plist"
|
||||||
extra_substitutions = [
|
extra_substitutions = [
|
||||||
"ATOM_BUNDLE_ID=$electron_mac_bundle_id.framework",
|
"ATOM_BUNDLE_ID=$electron_mac_bundle_id.framework",
|
||||||
|
|
2
DEPS
2
DEPS
|
@ -2,7 +2,7 @@ vars = {
|
||||||
'chromium_version':
|
'chromium_version':
|
||||||
'66.0.3359.181',
|
'66.0.3359.181',
|
||||||
'libchromiumcontent_revision':
|
'libchromiumcontent_revision':
|
||||||
'0628cd925e513bfd873524621a91834b366899f1',
|
'a55a9ce536db60702630c4b9d94dcb2145fc3b24',
|
||||||
'node_version':
|
'node_version':
|
||||||
'v10.2.0-35-g4879332def',
|
'v10.2.0-35-g4879332def',
|
||||||
|
|
||||||
|
|
|
@ -50,11 +50,6 @@ class SystemPreferences : public mate::EventEmitter<SystemPreferences>
|
||||||
#if defined(OS_WIN)
|
#if defined(OS_WIN)
|
||||||
bool IsAeroGlassEnabled();
|
bool IsAeroGlassEnabled();
|
||||||
|
|
||||||
typedef HRESULT(STDAPICALLTYPE* DwmGetColorizationColor)(DWORD*, BOOL*);
|
|
||||||
DwmGetColorizationColor dwmGetColorizationColor =
|
|
||||||
(DwmGetColorizationColor)GetProcAddress(LoadLibraryW(L"dwmapi.dll"),
|
|
||||||
"DwmGetColorizationColor");
|
|
||||||
|
|
||||||
std::string GetAccentColor();
|
std::string GetAccentColor();
|
||||||
std::string GetColor(const std::string& color, mate::Arguments* args);
|
std::string GetColor(const std::string& color, mate::Arguments* args);
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
// Use of this source code is governed by the MIT license that can be
|
// Use of this source code is governed by the MIT license that can be
|
||||||
// found in the LICENSE file.
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#include <dwmapi.h>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
|
|
||||||
#include "atom/browser/api/atom_api_system_preferences.h"
|
#include "atom/browser/api/atom_api_system_preferences.h"
|
||||||
|
@ -38,7 +39,7 @@ std::string SystemPreferences::GetAccentColor() {
|
||||||
DWORD color = 0;
|
DWORD color = 0;
|
||||||
BOOL opaque = FALSE;
|
BOOL opaque = FALSE;
|
||||||
|
|
||||||
if (FAILED(dwmGetColorizationColor(&color, &opaque))) {
|
if (FAILED(DwmGetColorizationColor(&color, &opaque))) {
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1434,6 +1434,10 @@ bool WebContents::IsAudioMuted() {
|
||||||
return web_contents()->IsAudioMuted();
|
return web_contents()->IsAudioMuted();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool WebContents::IsCurrentlyAudible() {
|
||||||
|
return web_contents()->IsCurrentlyAudible();
|
||||||
|
}
|
||||||
|
|
||||||
void WebContents::Print(mate::Arguments* args) {
|
void WebContents::Print(mate::Arguments* args) {
|
||||||
PrintSettings settings = {false, false, base::string16()};
|
PrintSettings settings = {false, false, base::string16()};
|
||||||
if (args->Length() >= 1 && !args->GetNext(&settings)) {
|
if (args->Length() >= 1 && !args->GetNext(&settings)) {
|
||||||
|
@ -2018,6 +2022,7 @@ void WebContents::BuildPrototype(v8::Isolate* isolate,
|
||||||
.SetMethod("setIgnoreMenuShortcuts", &WebContents::SetIgnoreMenuShortcuts)
|
.SetMethod("setIgnoreMenuShortcuts", &WebContents::SetIgnoreMenuShortcuts)
|
||||||
.SetMethod("setAudioMuted", &WebContents::SetAudioMuted)
|
.SetMethod("setAudioMuted", &WebContents::SetAudioMuted)
|
||||||
.SetMethod("isAudioMuted", &WebContents::IsAudioMuted)
|
.SetMethod("isAudioMuted", &WebContents::IsAudioMuted)
|
||||||
|
.SetMethod("isCurrentlyAudible", &WebContents::IsCurrentlyAudible)
|
||||||
.SetMethod("undo", &WebContents::Undo)
|
.SetMethod("undo", &WebContents::Undo)
|
||||||
.SetMethod("redo", &WebContents::Redo)
|
.SetMethod("redo", &WebContents::Redo)
|
||||||
.SetMethod("cut", &WebContents::Cut)
|
.SetMethod("cut", &WebContents::Cut)
|
||||||
|
|
|
@ -143,6 +143,7 @@ class WebContents : public mate::TrackableObject<WebContents>,
|
||||||
void SetIgnoreMenuShortcuts(bool ignore);
|
void SetIgnoreMenuShortcuts(bool ignore);
|
||||||
void SetAudioMuted(bool muted);
|
void SetAudioMuted(bool muted);
|
||||||
bool IsAudioMuted();
|
bool IsAudioMuted();
|
||||||
|
bool IsCurrentlyAudible();
|
||||||
void Print(mate::Arguments* args);
|
void Print(mate::Arguments* args);
|
||||||
std::vector<printing::PrinterBasicInfo> GetPrinterList();
|
std::vector<printing::PrinterBasicInfo> GetPrinterList();
|
||||||
void SetEmbedder(const WebContents* embedder);
|
void SetEmbedder(const WebContents* embedder);
|
||||||
|
|
|
@ -26,7 +26,8 @@ FrameSubscriber::FrameSubscriber(v8::Isolate* isolate,
|
||||||
: content::WebContentsObserver(web_contents),
|
: content::WebContentsObserver(web_contents),
|
||||||
isolate_(isolate),
|
isolate_(isolate),
|
||||||
callback_(callback),
|
callback_(callback),
|
||||||
only_dirty_(only_dirty) {}
|
only_dirty_(only_dirty),
|
||||||
|
weak_ptr_factory_(this) {}
|
||||||
|
|
||||||
FrameSubscriber::~FrameSubscriber() = default;
|
FrameSubscriber::~FrameSubscriber() = default;
|
||||||
|
|
||||||
|
@ -64,7 +65,7 @@ void FrameSubscriber::DidReceiveCompositorFrame() {
|
||||||
|
|
||||||
view->CopyFromSurface(
|
view->CopyFromSurface(
|
||||||
gfx::Rect(), view->GetViewBounds().size(),
|
gfx::Rect(), view->GetViewBounds().size(),
|
||||||
base::BindOnce(&FrameSubscriber::Done, base::Unretained(this),
|
base::BindOnce(&FrameSubscriber::Done, weak_ptr_factory_.GetWeakPtr(),
|
||||||
GetDamageRect()));
|
GetDamageRect()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "content/public/browser/web_contents.h"
|
#include "content/public/browser/web_contents.h"
|
||||||
|
|
||||||
#include "base/callback.h"
|
#include "base/callback.h"
|
||||||
|
#include "base/memory/weak_ptr.h"
|
||||||
#include "components/viz/common/frame_sinks/copy_output_result.h"
|
#include "components/viz/common/frame_sinks/copy_output_result.h"
|
||||||
#include "content/public/browser/web_contents_observer.h"
|
#include "content/public/browser/web_contents_observer.h"
|
||||||
#include "ui/gfx/image/image.h"
|
#include "ui/gfx/image/image.h"
|
||||||
|
@ -39,6 +40,8 @@ class FrameSubscriber : public content::WebContentsObserver {
|
||||||
FrameCaptureCallback callback_;
|
FrameCaptureCallback callback_;
|
||||||
bool only_dirty_;
|
bool only_dirty_;
|
||||||
|
|
||||||
|
base::WeakPtrFactory<FrameSubscriber> weak_ptr_factory_;
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(FrameSubscriber);
|
DISALLOW_COPY_AND_ASSIGN(FrameSubscriber);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -12,17 +12,27 @@
|
||||||
#include "atom/common/native_mate_converters/gurl_converter.h"
|
#include "atom/common/native_mate_converters/gurl_converter.h"
|
||||||
#include "atom/common/node_includes.h"
|
#include "atom/common/node_includes.h"
|
||||||
#include "base/hash.h"
|
#include "base/hash.h"
|
||||||
|
#include "base/process/process_handle.h"
|
||||||
|
#include "base/strings/stringprintf.h"
|
||||||
#include "native_mate/dictionary.h"
|
#include "native_mate/dictionary.h"
|
||||||
#include "url/origin.h"
|
#include "url/origin.h"
|
||||||
#include "v8/include/v8-profiler.h"
|
#include "v8/include/v8-profiler.h"
|
||||||
|
|
||||||
|
// This is defined in later versions of Chromium, remove this if you see
|
||||||
|
// compiler complaining duplicate defines.
|
||||||
|
#if defined(OS_WIN) || defined(OS_FUCHSIA)
|
||||||
|
#define CrPRIdPid "ld"
|
||||||
|
#else
|
||||||
|
#define CrPRIdPid "d"
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace std {
|
namespace std {
|
||||||
|
|
||||||
// The hash function used by DoubleIDWeakMap.
|
// The hash function used by DoubleIDWeakMap.
|
||||||
template <typename Type1, typename Type2>
|
template <typename Type1, typename Type2>
|
||||||
struct hash<std::pair<Type1, Type2>> {
|
struct hash<std::pair<Type1, Type2>> {
|
||||||
std::size_t operator()(std::pair<Type1, Type2> value) const {
|
std::size_t operator()(std::pair<Type1, Type2> value) const {
|
||||||
return base::HashInts<Type1, Type2>(value.first, value.second);
|
return base::HashInts(base::Hash(value.first), value.second);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -90,6 +100,16 @@ int32_t GetObjectHash(v8::Local<v8::Object> object) {
|
||||||
return object->GetIdentityHash();
|
return object->GetIdentityHash();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string GetContextID(v8::Isolate* isolate) {
|
||||||
|
// When a page is reloaded, V8 and blink may have optimizations that do not
|
||||||
|
// free blink::WebLocalFrame and v8::Context and reuse them for the new page,
|
||||||
|
// while we always recreate node::Environment when a page is loaded.
|
||||||
|
// So the only reliable way to return an identity for a page, is to return the
|
||||||
|
// address of the node::Environment instance.
|
||||||
|
node::Environment* env = node::Environment::GetCurrent(isolate);
|
||||||
|
return base::StringPrintf("%" CrPRIdPid "-%p", base::GetCurrentProcId(), env);
|
||||||
|
}
|
||||||
|
|
||||||
void TakeHeapSnapshot(v8::Isolate* isolate) {
|
void TakeHeapSnapshot(v8::Isolate* isolate) {
|
||||||
isolate->GetHeapProfiler()->TakeHeapSnapshot();
|
isolate->GetHeapProfiler()->TakeHeapSnapshot();
|
||||||
}
|
}
|
||||||
|
@ -112,12 +132,14 @@ void Initialize(v8::Local<v8::Object> exports,
|
||||||
dict.SetMethod("setHiddenValue", &SetHiddenValue);
|
dict.SetMethod("setHiddenValue", &SetHiddenValue);
|
||||||
dict.SetMethod("deleteHiddenValue", &DeleteHiddenValue);
|
dict.SetMethod("deleteHiddenValue", &DeleteHiddenValue);
|
||||||
dict.SetMethod("getObjectHash", &GetObjectHash);
|
dict.SetMethod("getObjectHash", &GetObjectHash);
|
||||||
|
dict.SetMethod("getContextId", &GetContextID);
|
||||||
dict.SetMethod("takeHeapSnapshot", &TakeHeapSnapshot);
|
dict.SetMethod("takeHeapSnapshot", &TakeHeapSnapshot);
|
||||||
dict.SetMethod("setRemoteCallbackFreer", &atom::RemoteCallbackFreer::BindTo);
|
dict.SetMethod("setRemoteCallbackFreer", &atom::RemoteCallbackFreer::BindTo);
|
||||||
dict.SetMethod("setRemoteObjectFreer", &atom::RemoteObjectFreer::BindTo);
|
dict.SetMethod("setRemoteObjectFreer", &atom::RemoteObjectFreer::BindTo);
|
||||||
dict.SetMethod("createIDWeakMap", &atom::api::KeyWeakMap<int32_t>::Create);
|
dict.SetMethod("createIDWeakMap", &atom::api::KeyWeakMap<int32_t>::Create);
|
||||||
dict.SetMethod("createDoubleIDWeakMap",
|
dict.SetMethod(
|
||||||
&atom::api::KeyWeakMap<std::pair<int64_t, int32_t>>::Create);
|
"createDoubleIDWeakMap",
|
||||||
|
&atom::api::KeyWeakMap<std::pair<std::string, int32_t>>::Create);
|
||||||
dict.SetMethod("requestGarbageCollectionForTesting",
|
dict.SetMethod("requestGarbageCollectionForTesting",
|
||||||
&RequestGarbageCollectionForTesting);
|
&RequestGarbageCollectionForTesting);
|
||||||
dict.SetMethod("isSameOrigin", &IsSameOrigin);
|
dict.SetMethod("isSameOrigin", &IsSameOrigin);
|
||||||
|
|
|
@ -15,17 +15,20 @@ namespace atom {
|
||||||
// static
|
// static
|
||||||
void RemoteCallbackFreer::BindTo(v8::Isolate* isolate,
|
void RemoteCallbackFreer::BindTo(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id,
|
int object_id,
|
||||||
content::WebContents* web_contents) {
|
content::WebContents* web_contents) {
|
||||||
new RemoteCallbackFreer(isolate, target, object_id, web_contents);
|
new RemoteCallbackFreer(isolate, target, context_id, object_id, web_contents);
|
||||||
}
|
}
|
||||||
|
|
||||||
RemoteCallbackFreer::RemoteCallbackFreer(v8::Isolate* isolate,
|
RemoteCallbackFreer::RemoteCallbackFreer(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id,
|
int object_id,
|
||||||
content::WebContents* web_contents)
|
content::WebContents* web_contents)
|
||||||
: ObjectLifeMonitor(isolate, target),
|
: ObjectLifeMonitor(isolate, target),
|
||||||
content::WebContentsObserver(web_contents),
|
content::WebContentsObserver(web_contents),
|
||||||
|
context_id_(context_id),
|
||||||
object_id_(object_id) {}
|
object_id_(object_id) {}
|
||||||
|
|
||||||
RemoteCallbackFreer::~RemoteCallbackFreer() {}
|
RemoteCallbackFreer::~RemoteCallbackFreer() {}
|
||||||
|
@ -34,6 +37,7 @@ void RemoteCallbackFreer::RunDestructor() {
|
||||||
base::string16 channel =
|
base::string16 channel =
|
||||||
base::ASCIIToUTF16("ELECTRON_RENDERER_RELEASE_CALLBACK");
|
base::ASCIIToUTF16("ELECTRON_RENDERER_RELEASE_CALLBACK");
|
||||||
base::ListValue args;
|
base::ListValue args;
|
||||||
|
args.AppendString(context_id_);
|
||||||
args.AppendInteger(object_id_);
|
args.AppendInteger(object_id_);
|
||||||
auto* frame_host = web_contents()->GetMainFrame();
|
auto* frame_host = web_contents()->GetMainFrame();
|
||||||
if (frame_host) {
|
if (frame_host) {
|
||||||
|
|
|
@ -4,6 +4,9 @@
|
||||||
|
|
||||||
#ifndef ATOM_COMMON_API_REMOTE_CALLBACK_FREER_H_
|
#ifndef ATOM_COMMON_API_REMOTE_CALLBACK_FREER_H_
|
||||||
#define ATOM_COMMON_API_REMOTE_CALLBACK_FREER_H_
|
#define ATOM_COMMON_API_REMOTE_CALLBACK_FREER_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
#include "atom/common/api/object_life_monitor.h"
|
#include "atom/common/api/object_life_monitor.h"
|
||||||
#include "content/public/browser/web_contents_observer.h"
|
#include "content/public/browser/web_contents_observer.h"
|
||||||
|
|
||||||
|
@ -14,12 +17,14 @@ class RemoteCallbackFreer : public ObjectLifeMonitor,
|
||||||
public:
|
public:
|
||||||
static void BindTo(v8::Isolate* isolate,
|
static void BindTo(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id,
|
int object_id,
|
||||||
content::WebContents* web_conents);
|
content::WebContents* web_conents);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
RemoteCallbackFreer(v8::Isolate* isolate,
|
RemoteCallbackFreer(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id,
|
int object_id,
|
||||||
content::WebContents* web_conents);
|
content::WebContents* web_conents);
|
||||||
~RemoteCallbackFreer() override;
|
~RemoteCallbackFreer() override;
|
||||||
|
@ -30,6 +35,7 @@ class RemoteCallbackFreer : public ObjectLifeMonitor,
|
||||||
void RenderViewDeleted(content::RenderViewHost*) override;
|
void RenderViewDeleted(content::RenderViewHost*) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
std::string context_id_;
|
||||||
int object_id_;
|
int object_id_;
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(RemoteCallbackFreer);
|
DISALLOW_COPY_AND_ASSIGN(RemoteCallbackFreer);
|
||||||
|
|
|
@ -29,14 +29,17 @@ content::RenderFrame* GetCurrentRenderFrame() {
|
||||||
// static
|
// static
|
||||||
void RemoteObjectFreer::BindTo(v8::Isolate* isolate,
|
void RemoteObjectFreer::BindTo(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id) {
|
int object_id) {
|
||||||
new RemoteObjectFreer(isolate, target, object_id);
|
new RemoteObjectFreer(isolate, target, context_id, object_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
RemoteObjectFreer::RemoteObjectFreer(v8::Isolate* isolate,
|
RemoteObjectFreer::RemoteObjectFreer(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id)
|
int object_id)
|
||||||
: ObjectLifeMonitor(isolate, target),
|
: ObjectLifeMonitor(isolate, target),
|
||||||
|
context_id_(context_id),
|
||||||
object_id_(object_id),
|
object_id_(object_id),
|
||||||
routing_id_(MSG_ROUTING_NONE) {
|
routing_id_(MSG_ROUTING_NONE) {
|
||||||
content::RenderFrame* render_frame = GetCurrentRenderFrame();
|
content::RenderFrame* render_frame = GetCurrentRenderFrame();
|
||||||
|
@ -56,6 +59,7 @@ void RemoteObjectFreer::RunDestructor() {
|
||||||
base::string16 channel = base::ASCIIToUTF16("ipc-message");
|
base::string16 channel = base::ASCIIToUTF16("ipc-message");
|
||||||
base::ListValue args;
|
base::ListValue args;
|
||||||
args.AppendString("ELECTRON_BROWSER_DEREFERENCE");
|
args.AppendString("ELECTRON_BROWSER_DEREFERENCE");
|
||||||
|
args.AppendString(context_id_);
|
||||||
args.AppendInteger(object_id_);
|
args.AppendInteger(object_id_);
|
||||||
render_frame->Send(new AtomFrameHostMsg_Message(render_frame->GetRoutingID(),
|
render_frame->Send(new AtomFrameHostMsg_Message(render_frame->GetRoutingID(),
|
||||||
channel, args));
|
channel, args));
|
||||||
|
|
|
@ -5,6 +5,8 @@
|
||||||
#ifndef ATOM_COMMON_API_REMOTE_OBJECT_FREER_H_
|
#ifndef ATOM_COMMON_API_REMOTE_OBJECT_FREER_H_
|
||||||
#define ATOM_COMMON_API_REMOTE_OBJECT_FREER_H_
|
#define ATOM_COMMON_API_REMOTE_OBJECT_FREER_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
#include "atom/common/api/object_life_monitor.h"
|
#include "atom/common/api/object_life_monitor.h"
|
||||||
|
|
||||||
namespace atom {
|
namespace atom {
|
||||||
|
@ -13,17 +15,20 @@ class RemoteObjectFreer : public ObjectLifeMonitor {
|
||||||
public:
|
public:
|
||||||
static void BindTo(v8::Isolate* isolate,
|
static void BindTo(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id);
|
int object_id);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
RemoteObjectFreer(v8::Isolate* isolate,
|
RemoteObjectFreer(v8::Isolate* isolate,
|
||||||
v8::Local<v8::Object> target,
|
v8::Local<v8::Object> target,
|
||||||
|
const std::string& context_id,
|
||||||
int object_id);
|
int object_id);
|
||||||
~RemoteObjectFreer() override;
|
~RemoteObjectFreer() override;
|
||||||
|
|
||||||
void RunDestructor() override;
|
void RunDestructor() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
std::string context_id_;
|
||||||
int object_id_;
|
int object_id_;
|
||||||
int routing_id_;
|
int routing_id_;
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,9 @@
|
||||||
|
|
||||||
// Include common headers for using node APIs.
|
// Include common headers for using node APIs.
|
||||||
|
|
||||||
|
#ifdef NODE_SHARED_MODE
|
||||||
#define BUILDING_NODE_EXTENSION
|
#define BUILDING_NODE_EXTENSION
|
||||||
|
#endif
|
||||||
|
|
||||||
// The following define makes sure that we do not include the macros
|
// The following define makes sure that we do not include the macros
|
||||||
// again. But we still need the tracing functions, so declaring them.
|
// again. But we still need the tracing functions, so declaring them.
|
||||||
|
|
|
@ -240,33 +240,6 @@ bool ShowItemInFolder(const base::FilePath& full_path) {
|
||||||
if (dir.empty())
|
if (dir.empty())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
typedef HRESULT(WINAPI * SHOpenFolderAndSelectItemsFuncPtr)(
|
|
||||||
PCIDLIST_ABSOLUTE pidl_Folder, UINT cidl, PCUITEMID_CHILD_ARRAY pidls,
|
|
||||||
DWORD flags);
|
|
||||||
|
|
||||||
static SHOpenFolderAndSelectItemsFuncPtr open_folder_and_select_itemsPtr =
|
|
||||||
NULL;
|
|
||||||
static bool initialize_open_folder_proc = true;
|
|
||||||
if (initialize_open_folder_proc) {
|
|
||||||
initialize_open_folder_proc = false;
|
|
||||||
// The SHOpenFolderAndSelectItems API is exposed by shell32 version 6
|
|
||||||
// and does not exist in Win2K. We attempt to retrieve this function export
|
|
||||||
// from shell32 and if it does not exist, we just invoke ShellExecute to
|
|
||||||
// open the folder thus losing the functionality to select the item in
|
|
||||||
// the process.
|
|
||||||
HMODULE shell32_base = GetModuleHandle(L"shell32.dll");
|
|
||||||
if (!shell32_base) {
|
|
||||||
NOTREACHED() << " " << __FUNCTION__ << "(): Can't open shell32.dll";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
open_folder_and_select_itemsPtr =
|
|
||||||
reinterpret_cast<SHOpenFolderAndSelectItemsFuncPtr>(
|
|
||||||
GetProcAddress(shell32_base, "SHOpenFolderAndSelectItems"));
|
|
||||||
}
|
|
||||||
if (!open_folder_and_select_itemsPtr) {
|
|
||||||
return ui::win::OpenFolderViaShell(dir);
|
|
||||||
}
|
|
||||||
|
|
||||||
Microsoft::WRL::ComPtr<IShellFolder> desktop;
|
Microsoft::WRL::ComPtr<IShellFolder> desktop;
|
||||||
HRESULT hr = SHGetDesktopFolder(desktop.GetAddressOf());
|
HRESULT hr = SHGetDesktopFolder(desktop.GetAddressOf());
|
||||||
if (FAILED(hr))
|
if (FAILED(hr))
|
||||||
|
@ -290,8 +263,8 @@ bool ShowItemInFolder(const base::FilePath& full_path) {
|
||||||
|
|
||||||
const ITEMIDLIST* highlight[] = {file_item};
|
const ITEMIDLIST* highlight[] = {file_item};
|
||||||
|
|
||||||
hr = (*open_folder_and_select_itemsPtr)(dir_item, arraysize(highlight),
|
hr = SHOpenFolderAndSelectItems(dir_item, arraysize(highlight), highlight,
|
||||||
highlight, NULL);
|
NULL);
|
||||||
if (!FAILED(hr))
|
if (!FAILED(hr))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
|
|
@ -81,11 +81,19 @@ base::CommandLine::StringVector GetArgv() {
|
||||||
return base::CommandLine::ForCurrentProcess()->argv();
|
return base::CommandLine::ForCurrentProcess()->argv();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
v8::Local<v8::Value> CreatePreloadScript(v8::Isolate* isolate,
|
||||||
|
v8::Local<v8::String> preloadSrc) {
|
||||||
|
auto script = v8::Script::Compile(preloadSrc);
|
||||||
|
auto func = script->Run();
|
||||||
|
return func;
|
||||||
|
}
|
||||||
|
|
||||||
void InitializeBindings(v8::Local<v8::Object> binding,
|
void InitializeBindings(v8::Local<v8::Object> binding,
|
||||||
v8::Local<v8::Context> context) {
|
v8::Local<v8::Context> context) {
|
||||||
auto* isolate = context->GetIsolate();
|
auto* isolate = context->GetIsolate();
|
||||||
mate::Dictionary b(isolate, binding);
|
mate::Dictionary b(isolate, binding);
|
||||||
b.SetMethod("get", GetBinding);
|
b.SetMethod("get", GetBinding);
|
||||||
|
b.SetMethod("createPreloadScript", CreatePreloadScript);
|
||||||
b.SetMethod("crash", AtomBindings::Crash);
|
b.SetMethod("crash", AtomBindings::Crash);
|
||||||
b.SetMethod("hang", AtomBindings::Hang);
|
b.SetMethod("hang", AtomBindings::Hang);
|
||||||
b.SetMethod("getArgv", GetArgv);
|
b.SetMethod("getArgv", GetArgv);
|
||||||
|
|
|
@ -13,7 +13,6 @@ action("configure_node") {
|
||||||
args = [
|
args = [
|
||||||
"--enable-static",
|
"--enable-static",
|
||||||
"--release-urlbase=https://atom.io/download/electron",
|
"--release-urlbase=https://atom.io/download/electron",
|
||||||
"--shared",
|
|
||||||
"--shared-openssl",
|
"--shared-openssl",
|
||||||
"--shared-openssl-includes=" + rebase_path("//third_party/boringssl/src/include"),
|
"--shared-openssl-includes=" + rebase_path("//third_party/boringssl/src/include"),
|
||||||
"--shared-openssl-libname=boringssl" + ssl_libname_suffix,
|
"--shared-openssl-libname=boringssl" + ssl_libname_suffix,
|
||||||
|
@ -29,6 +28,9 @@ action("configure_node") {
|
||||||
"--config-out-dir=" + rebase_path(target_gen_dir),
|
"--config-out-dir=" + rebase_path(target_gen_dir),
|
||||||
"--no-run-gyp",
|
"--no-run-gyp",
|
||||||
]
|
]
|
||||||
|
if (is_component_build) {
|
||||||
|
args += [ "--shared" ]
|
||||||
|
}
|
||||||
outputs = [
|
outputs = [
|
||||||
"$target_gen_dir/config.gypi",
|
"$target_gen_dir/config.gypi",
|
||||||
]
|
]
|
||||||
|
@ -75,6 +77,7 @@ action("gyp_node") {
|
||||||
"-D", "llvm_dir=" + rebase_path("//third_party/llvm-build/Release+Asserts"),
|
"-D", "llvm_dir=" + rebase_path("//third_party/llvm-build/Release+Asserts"),
|
||||||
"-D", "libcxx_dir=" + rebase_path("//buildtools/third_party/libc++"),
|
"-D", "libcxx_dir=" + rebase_path("//buildtools/third_party/libc++"),
|
||||||
"-D", "libcxxabi_dir=" + rebase_path("//buildtools/third_party/libc++abi"),
|
"-D", "libcxxabi_dir=" + rebase_path("//buildtools/third_party/libc++abi"),
|
||||||
|
"-D", "is_component_build=$is_component_build",
|
||||||
"-Goutput_dir=./$target_out_dir", # bizarrely, gyp generates from the build root instead of from cwd
|
"-Goutput_dir=./$target_out_dir", # bizarrely, gyp generates from the build root instead of from cwd
|
||||||
"-fninja",
|
"-fninja",
|
||||||
rebase_path("//third_party/electron_node/node.gyp", root_build_dir),
|
rebase_path("//third_party/electron_node/node.gyp", root_build_dir),
|
||||||
|
@ -93,16 +96,53 @@ action("build_node") {
|
||||||
script = "//electron/build/run-ninja.py"
|
script = "//electron/build/run-ninja.py"
|
||||||
args = [
|
args = [
|
||||||
"-C", rebase_path(target_out_dir, root_build_dir) + "/$node_configuration",
|
"-C", rebase_path(target_out_dir, root_build_dir) + "/$node_configuration",
|
||||||
"node_lib"
|
"node_lib",
|
||||||
|
"libuv", "nghttp2", "cares", "http_parser", "zlib"
|
||||||
]
|
]
|
||||||
if (is_mac) {
|
if (is_mac) {
|
||||||
|
if (is_component_build) {
|
||||||
outputs = [ "$target_out_dir/$node_configuration/libnode.dylib" ]
|
outputs = [ "$target_out_dir/$node_configuration/libnode.dylib" ]
|
||||||
|
} else {
|
||||||
|
outputs = [
|
||||||
|
"$target_out_dir/$node_configuration/libnode.a",
|
||||||
|
"$target_out_dir/$node_configuration/libcares.a",
|
||||||
|
"$target_out_dir/$node_configuration/libhttp_parser.a",
|
||||||
|
"$target_out_dir/$node_configuration/libnghttp2.a",
|
||||||
|
"$target_out_dir/$node_configuration/libuv.a",
|
||||||
|
"$target_out_dir/$node_configuration/libzlib.a",
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (is_linux) {
|
if (is_linux) {
|
||||||
|
if (is_component_build) {
|
||||||
outputs = [ "$target_out_dir/$node_configuration/lib/libnode.so" ]
|
outputs = [ "$target_out_dir/$node_configuration/lib/libnode.so" ]
|
||||||
|
} else {
|
||||||
|
outputs = [
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/libnode.a",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/uv/libuv.a",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/nghttp2/libnghttp2.a",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/cares/libcares.a",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/http_parser/libhttp_parser.a",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/zlib/libzlib.a",
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (is_win) {
|
if (is_win) {
|
||||||
outputs = [ "$target_out_dir/$node_configuration/node.dll.lib", "$target_out_dir/$node_configuration/node.dll" ]
|
if (is_component_build) {
|
||||||
|
outputs = [
|
||||||
|
"$target_out_dir/$node_configuration/node.dll.lib",
|
||||||
|
"$target_out_dir/$node_configuration/node.dll",
|
||||||
|
]
|
||||||
|
} else {
|
||||||
|
outputs = [
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/node.lib",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/uv/libuv.lib",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/nghttp2/nghttp2.lib",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/cares/cares.lib",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/http_parser/http_parser.lib",
|
||||||
|
"$target_out_dir/$node_configuration/obj/third_party/electron_node/deps/zlib/zlib.lib",
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -120,17 +160,25 @@ config("node_config") {
|
||||||
"//third_party/electron_node/deps/uv/include",
|
"//third_party/electron_node/deps/uv/include",
|
||||||
"//third_party/electron_node/deps/cares/include",
|
"//third_party/electron_node/deps/cares/include",
|
||||||
]
|
]
|
||||||
|
if (is_win && is_component_build) {
|
||||||
|
# Windows builds need both the .dll and the .dll.lib copied, but only the
|
||||||
|
# .dll.lib goes in the `libs` list.
|
||||||
libs = [ node_libs[0] ]
|
libs = [ node_libs[0] ]
|
||||||
|
} else {
|
||||||
|
libs = node_libs
|
||||||
|
}
|
||||||
cflags_cc = [
|
cflags_cc = [
|
||||||
"-Wno-deprecated-declarations",
|
"-Wno-deprecated-declarations",
|
||||||
]
|
]
|
||||||
defines = [
|
defines = [
|
||||||
# We need to access internal implementations of Node.
|
# We need to access internal implementations of Node.
|
||||||
"NODE_WANT_INTERNALS=1",
|
"NODE_WANT_INTERNALS=1",
|
||||||
"NODE_SHARED_MODE",
|
|
||||||
"HAVE_OPENSSL=1",
|
"HAVE_OPENSSL=1",
|
||||||
"HAVE_INSPECTOR=1",
|
"HAVE_INSPECTOR=1",
|
||||||
]
|
]
|
||||||
|
if (is_component_build) {
|
||||||
|
defines += [ "NODE_SHARED_MODE" ]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
group("node") {
|
group("node") {
|
||||||
|
|
|
@ -5,6 +5,8 @@
|
||||||
'v8_enable_inspector': 1,
|
'v8_enable_inspector': 1,
|
||||||
|
|
||||||
'shlib_suffix': '<(shlib_suffix_gn)',
|
'shlib_suffix': '<(shlib_suffix_gn)',
|
||||||
|
|
||||||
|
'is_component_build%': 'false',
|
||||||
},
|
},
|
||||||
'conditions': [
|
'conditions': [
|
||||||
['OS=="linux"', {
|
['OS=="linux"', {
|
||||||
|
@ -32,7 +34,13 @@
|
||||||
}]
|
}]
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}]
|
}],
|
||||||
|
['OS=="win"', {
|
||||||
|
'make_global_settings': [
|
||||||
|
['CC', '<(llvm_dir)/bin/clang-cl'],
|
||||||
|
# Also defining CXX doesn't appear to be necessary.
|
||||||
|
]
|
||||||
|
}],
|
||||||
],
|
],
|
||||||
'target_defaults': {
|
'target_defaults': {
|
||||||
'target_conditions': [
|
'target_conditions': [
|
||||||
|
@ -47,9 +55,12 @@
|
||||||
'EVP_CTRL_AEAD_SET_IVLEN=EVP_CTRL_GCM_SET_IVLEN',
|
'EVP_CTRL_AEAD_SET_IVLEN=EVP_CTRL_GCM_SET_IVLEN',
|
||||||
'EVP_CTRL_CCM_SET_TAG=EVP_CTRL_GCM_SET_TAG',
|
'EVP_CTRL_CCM_SET_TAG=EVP_CTRL_GCM_SET_TAG',
|
||||||
'EVP_CTRL_AEAD_GET_TAG=EVP_CTRL_GCM_GET_TAG',
|
'EVP_CTRL_AEAD_GET_TAG=EVP_CTRL_GCM_GET_TAG',
|
||||||
|
'WIN32_LEAN_AND_MEAN',
|
||||||
],
|
],
|
||||||
'conditions': [
|
'conditions': [
|
||||||
['OS=="win"', {
|
['OS=="win"', {
|
||||||
|
'conditions': [
|
||||||
|
['is_component_build=="true"', {
|
||||||
'libraries': [
|
'libraries': [
|
||||||
'-lv8.dll',
|
'-lv8.dll',
|
||||||
'-lv8_libbase.dll',
|
'-lv8_libbase.dll',
|
||||||
|
@ -70,7 +81,11 @@
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
}]
|
||||||
|
],
|
||||||
}, {
|
}, {
|
||||||
|
'conditions': [
|
||||||
|
['is_component_build=="true"', {
|
||||||
'libraries': [
|
'libraries': [
|
||||||
'-lv8',
|
'-lv8',
|
||||||
'-lv8_libbase',
|
'-lv8_libbase',
|
||||||
|
@ -79,6 +94,8 @@
|
||||||
]
|
]
|
||||||
}]
|
}]
|
||||||
]
|
]
|
||||||
|
}]
|
||||||
|
]
|
||||||
}],
|
}],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
|
@ -832,6 +832,10 @@ Mute the audio on the current web page.
|
||||||
|
|
||||||
Returns `Boolean` - Whether this page has been muted.
|
Returns `Boolean` - Whether this page has been muted.
|
||||||
|
|
||||||
|
#### `contents.isCurrentlyAudible()`
|
||||||
|
|
||||||
|
Returns `Boolean` - Whether audio is currently playing.
|
||||||
|
|
||||||
#### `contents.setZoomFactor(factor)`
|
#### `contents.setZoomFactor(factor)`
|
||||||
|
|
||||||
* `factor` Number - Zoom factor.
|
* `factor` Number - Zoom factor.
|
||||||
|
|
|
@ -450,6 +450,10 @@ Set guest page muted.
|
||||||
|
|
||||||
Returns `Boolean` - Whether guest page has been muted.
|
Returns `Boolean` - Whether guest page has been muted.
|
||||||
|
|
||||||
|
#### `<webview>.isCurrentlyAudible()`
|
||||||
|
|
||||||
|
Returns `Boolean` - Whether audio is currently playing.
|
||||||
|
|
||||||
### `<webview>.undo()`
|
### `<webview>.undo()`
|
||||||
|
|
||||||
Executes editing command `undo` in page.
|
Executes editing command `undo` in page.
|
||||||
|
|
|
@ -17,11 +17,15 @@ Create a token from https://windows-ci.electronjs.org/api-token
|
||||||
If you don't have an account, ask a team member to add you.
|
If you don't have an account, ask a team member to add you.
|
||||||
* `CIRCLE_TOKEN`:
|
* `CIRCLE_TOKEN`:
|
||||||
Create a token from "Personal API Tokens" at https://circleci.com/account/api
|
Create a token from "Personal API Tokens" at https://circleci.com/account/api
|
||||||
|
* `VSTS_TOKEN`:
|
||||||
|
Create a Personal Access Token at https://github.visualstudio.com/_usersSettings/tokens
|
||||||
|
with the scope of `Build (read and execute)`.
|
||||||
|
|
||||||
Once you've generated these tokens, put them in a `.env` file in the root directory
|
Once you've generated these tokens, put them in a `.env` file in the root directory
|
||||||
of the project. This file is gitignored, and will be loaded into the
|
of the project. This file is gitignored, and will be loaded into the
|
||||||
environment by the release scripts.
|
environment by the release scripts.
|
||||||
|
|
||||||
|
|
||||||
## Determine which branch to release from
|
## Determine which branch to release from
|
||||||
|
|
||||||
- **If releasing beta,** run the scripts below from `master`.
|
- **If releasing beta,** run the scripts below from `master`.
|
||||||
|
@ -87,8 +91,11 @@ $ ./script/bump-version.py --bump minor --dry-run
|
||||||
The `prepare-release` script will trigger the builds via API calls.
|
The `prepare-release` script will trigger the builds via API calls.
|
||||||
To monitor the build progress, see the following pages:
|
To monitor the build progress, see the following pages:
|
||||||
|
|
||||||
- [circleci.com/gh/electron/electron](https://circleci.com/gh/electron) for OS X and Linux
|
- [electron-release-mas-x64](https://github.visualstudio.com/electron/_build/index?context=allDefinitions&path=%5C&definitionId=19&_a=completed) for MAS builds.
|
||||||
- [windows-ci.electronjs.org/project/AppVeyor/electron](https://windows-ci.electronjs.org/project/AppVeyor/electron) for Windows
|
- [electron-release-osx-x64](https://github.visualstudio.com/electron/_build/index?context=allDefinitions&path=%5C&definitionId=18&_a=completed) for OSX builds.
|
||||||
|
- [circleci.com/gh/electron/electron](https://circleci.com/gh/electron) for Linux builds.
|
||||||
|
- [windows-ci.electronjs.org/project/AppVeyor/electron-39ng6](https://windows-ci.electronjs.org/project/AppVeyor/electron-39ng6) for Windows 32-bit builds.
|
||||||
|
- [windows-ci.electronjs.org/project/AppVeyor/electron](https://windows-ci.electronjs.org/project/AppVeyor/electron) for Windows 64-bit builds.
|
||||||
|
|
||||||
## Compile release notes
|
## Compile release notes
|
||||||
|
|
||||||
|
@ -195,35 +202,13 @@ under the `beta` tag and can be installed via `npm install electron@beta`.
|
||||||
1. Visit [the releases page] and you'll see a new draft release with placeholder
|
1. Visit [the releases page] and you'll see a new draft release with placeholder
|
||||||
release notes.
|
release notes.
|
||||||
2. Edit the release and add release notes.
|
2. Edit the release and add release notes.
|
||||||
3. Uncheck the `prerelease` checkbox if you're publishing a stable release;
|
3. Click 'Save draft'. **Do not click 'Publish release'!**
|
||||||
leave it checked for beta releases.
|
4. Wait for all builds to pass before proceeding.
|
||||||
4. Click 'Save draft'. **Do not click 'Publish release'!**
|
5. In the branch, verify that the release's files have been created:
|
||||||
5. Wait for all builds to pass before proceeding.
|
|
||||||
6. In the `release` branch, verify that the release's files have been created:
|
|
||||||
```sh
|
```sh
|
||||||
$ git rev-parse --abbrev-ref HEAD
|
|
||||||
release
|
|
||||||
$ npm run release -- --validateRelease
|
$ npm run release -- --validateRelease
|
||||||
```
|
```
|
||||||
|
|
||||||
## Merge temporary branch (pre-2-0-x branches only)
|
|
||||||
Once the release builds have finished, merge the `release` branch back into
|
|
||||||
the source release branch using the `merge-release` script.
|
|
||||||
If the branch cannot be successfully merged back this script will automatically
|
|
||||||
rebase the `release` branch and push the changes which will trigger the release
|
|
||||||
builds again, which means you will need to wait for the release builds to run
|
|
||||||
again before proceeding.
|
|
||||||
|
|
||||||
### Merging back into master
|
|
||||||
```sh
|
|
||||||
npm run merge-release -- master
|
|
||||||
```
|
|
||||||
|
|
||||||
### Merging back into old release branch
|
|
||||||
```sh
|
|
||||||
npm run merge-release -- 1-7-x
|
|
||||||
```
|
|
||||||
|
|
||||||
## Publish the release
|
## Publish the release
|
||||||
|
|
||||||
Once the merge has finished successfully, run the `release` script
|
Once the merge has finished successfully, run the `release` script
|
||||||
|
@ -237,7 +222,6 @@ on Windows by node-gyp to build native modules.
|
||||||
5. Validate that all of the required files are present on GitHub and S3 and have
|
5. Validate that all of the required files are present on GitHub and S3 and have
|
||||||
the correct checksums as specified in the SHASUMS files.
|
the correct checksums as specified in the SHASUMS files.
|
||||||
6. Publish the release on GitHub
|
6. Publish the release on GitHub
|
||||||
7. Delete the `release` branch.
|
|
||||||
|
|
||||||
## Publish to npm
|
## Publish to npm
|
||||||
|
|
||||||
|
@ -267,16 +251,39 @@ electron
|
||||||
$ npm run publish-to-npm
|
$ npm run publish-to-npm
|
||||||
```
|
```
|
||||||
|
|
||||||
Note: In general you should be using the latest Node during this
|
|
||||||
process; however, older versions of the `publish-to-npm` script
|
|
||||||
may have trouble with Node 7 or higher. If you have trouble with
|
|
||||||
this in an older branch, try running with an older version of Node,
|
|
||||||
e.g. a 6.x LTS.
|
|
||||||
|
|
||||||
[the releases page]: https://github.com/electron/electron/releases
|
[the releases page]: https://github.com/electron/electron/releases
|
||||||
[this bump commit]: https://github.com/electron/electron/commit/78ec1b8f89b3886b856377a1756a51617bc33f5a
|
[this bump commit]: https://github.com/electron/electron/commit/78ec1b8f89b3886b856377a1756a51617bc33f5a
|
||||||
[versioning]: /docs/tutorial/electron-versioning.md
|
[versioning]: /docs/tutorial/electron-versioning.md
|
||||||
|
|
||||||
|
# Troubleshooting
|
||||||
|
|
||||||
|
## Rerun broken builds
|
||||||
|
|
||||||
|
If a release build fails for some reason, you can use `script/ci-release-build.js` to rerun a release build:
|
||||||
|
|
||||||
|
### Rerun all linux builds:
|
||||||
|
```sh
|
||||||
|
node script/ci-release-build.js --ci=CircleCI --ghRelease TARGET_BRANCH
|
||||||
|
(TARGET_BRANCH) is the branch you are releasing from.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Rerun all macOS builds:
|
||||||
|
```sh
|
||||||
|
node script/ci-release-build.js --ci=VSTS --ghRelease TARGET_BRANCH
|
||||||
|
(TARGET_BRANCH) is the branch you are releasing from.
|
||||||
|
```
|
||||||
|
|
||||||
|
### Rerun all Windows builds:
|
||||||
|
```sh
|
||||||
|
node script/ci-release-build.js --ci=AppVeyor --ghRelease TARGET_BRANCH
|
||||||
|
(TARGET_BRANCH) is the branch you are releasing from.
|
||||||
|
```
|
||||||
|
|
||||||
|
Additionally you can pass a job name to the script to run an individual job, eg:
|
||||||
|
````sh
|
||||||
|
node script/ci-release-build.js --ci=AppVeyor --ghRelease --job=electron-x64 TARGET_BRANCH
|
||||||
|
```
|
||||||
|
|
||||||
## Fix missing binaries of a release manually
|
## Fix missing binaries of a release manually
|
||||||
|
|
||||||
In the case of a corrupted release with broken CI machines, we might have to
|
In the case of a corrupted release with broken CI machines, we might have to
|
||||||
|
|
|
@ -7,7 +7,7 @@ and then enable it in your application.
|
||||||
## Prepare a Copy of Flash Plugin
|
## Prepare a Copy of Flash Plugin
|
||||||
|
|
||||||
On macOS and Linux, the details of the Pepper Flash plugin can be found by
|
On macOS and Linux, the details of the Pepper Flash plugin can be found by
|
||||||
navigating to `chrome://plugins` in the Chrome browser. Its location and version
|
navigating to `chrome://flash` in the Chrome browser. Its location and version
|
||||||
are useful for Electron's Pepper Flash support. You can also copy it to another
|
are useful for Electron's Pepper Flash support. You can also copy it to another
|
||||||
location.
|
location.
|
||||||
|
|
||||||
|
|
|
@ -363,6 +363,7 @@
|
||||||
],
|
],
|
||||||
'link_settings': {
|
'link_settings': {
|
||||||
'libraries': [
|
'libraries': [
|
||||||
|
'-ldwmapi.lib',
|
||||||
'-limm32.lib',
|
'-limm32.lib',
|
||||||
'-lgdi32.lib',
|
'-lgdi32.lib',
|
||||||
'-loleacc.lib',
|
'-loleacc.lib',
|
||||||
|
|
|
@ -17,16 +17,15 @@ class ObjectsRegistry {
|
||||||
|
|
||||||
// Register a new object and return its assigned ID. If the object is already
|
// Register a new object and return its assigned ID. If the object is already
|
||||||
// registered then the already assigned ID would be returned.
|
// registered then the already assigned ID would be returned.
|
||||||
add (webContents, obj) {
|
add (webContents, contextId, obj) {
|
||||||
// Get or assign an ID to the object.
|
// Get or assign an ID to the object.
|
||||||
const id = this.saveToStorage(obj)
|
const id = this.saveToStorage(obj)
|
||||||
|
|
||||||
// Add object to the set of referenced objects.
|
// Add object to the set of referenced objects.
|
||||||
const webContentsId = webContents.getId()
|
let owner = this.owners[contextId]
|
||||||
let owner = this.owners[webContentsId]
|
|
||||||
if (!owner) {
|
if (!owner) {
|
||||||
owner = this.owners[webContentsId] = new Set()
|
owner = this.owners[contextId] = new Set()
|
||||||
this.registerDeleteListener(webContents, webContentsId)
|
this.registerDeleteListener(webContents, contextId)
|
||||||
}
|
}
|
||||||
if (!owner.has(id)) {
|
if (!owner.has(id)) {
|
||||||
owner.add(id)
|
owner.add(id)
|
||||||
|
@ -43,25 +42,26 @@ class ObjectsRegistry {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dereference an object according to its ID.
|
// Dereference an object according to its ID.
|
||||||
remove (webContentsId, id) {
|
// Note that an object may be double-freed (cleared when page is reloaded, and
|
||||||
|
// then garbage collected in old page).
|
||||||
|
remove (contextId, id) {
|
||||||
|
let owner = this.owners[contextId]
|
||||||
|
if (owner) {
|
||||||
|
// Remove the reference in owner.
|
||||||
|
owner.delete(id)
|
||||||
// Dereference from the storage.
|
// Dereference from the storage.
|
||||||
this.dereference(id)
|
this.dereference(id)
|
||||||
|
|
||||||
// Also remove the reference in owner.
|
|
||||||
let owner = this.owners[webContentsId]
|
|
||||||
if (owner) {
|
|
||||||
owner.delete(id)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clear all references to objects refrenced by the WebContents.
|
// Clear all references to objects refrenced by the WebContents.
|
||||||
clear (webContentsId) {
|
clear (contextId) {
|
||||||
let owner = this.owners[webContentsId]
|
let owner = this.owners[contextId]
|
||||||
if (!owner) return
|
if (!owner) return
|
||||||
|
|
||||||
for (let id of owner) this.dereference(id)
|
for (let id of owner) this.dereference(id)
|
||||||
|
|
||||||
delete this.owners[webContentsId]
|
delete this.owners[contextId]
|
||||||
}
|
}
|
||||||
|
|
||||||
// Private: Saves the object into storage and assigns an ID for it.
|
// Private: Saves the object into storage and assigns an ID for it.
|
||||||
|
@ -92,12 +92,12 @@ class ObjectsRegistry {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Private: Clear the storage when webContents is reloaded/navigated.
|
// Private: Clear the storage when webContents is reloaded/navigated.
|
||||||
registerDeleteListener (webContents, webContentsId) {
|
registerDeleteListener (webContents, contextId) {
|
||||||
const processId = webContents.getProcessId()
|
const processId = webContents.getProcessId()
|
||||||
const listener = (event, deletedProcessId) => {
|
const listener = (event, deletedProcessId) => {
|
||||||
if (deletedProcessId === processId) {
|
if (deletedProcessId === processId) {
|
||||||
webContents.removeListener('render-view-deleted', listener)
|
webContents.removeListener('render-view-deleted', listener)
|
||||||
this.clear(webContentsId)
|
this.clear(contextId)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
webContents.on('render-view-deleted', listener)
|
webContents.on('render-view-deleted', listener)
|
||||||
|
|
|
@ -56,7 +56,7 @@ let getObjectPrototype = function (object) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert a real value into meta data.
|
// Convert a real value into meta data.
|
||||||
let valueToMeta = function (sender, value, optimizeSimpleObject = false) {
|
let valueToMeta = function (sender, contextId, value, optimizeSimpleObject = false) {
|
||||||
// Determine the type of value.
|
// Determine the type of value.
|
||||||
const meta = { type: typeof value }
|
const meta = { type: typeof value }
|
||||||
if (meta.type === 'object') {
|
if (meta.type === 'object') {
|
||||||
|
@ -84,14 +84,14 @@ let valueToMeta = function (sender, value, optimizeSimpleObject = false) {
|
||||||
|
|
||||||
// Fill the meta object according to value's type.
|
// Fill the meta object according to value's type.
|
||||||
if (meta.type === 'array') {
|
if (meta.type === 'array') {
|
||||||
meta.members = value.map((el) => valueToMeta(sender, el, optimizeSimpleObject))
|
meta.members = value.map((el) => valueToMeta(sender, contextId, el, optimizeSimpleObject))
|
||||||
} else if (meta.type === 'object' || meta.type === 'function') {
|
} else if (meta.type === 'object' || meta.type === 'function') {
|
||||||
meta.name = value.constructor ? value.constructor.name : ''
|
meta.name = value.constructor ? value.constructor.name : ''
|
||||||
|
|
||||||
// Reference the original value if it's an object, because when it's
|
// Reference the original value if it's an object, because when it's
|
||||||
// passed to renderer we would assume the renderer keeps a reference of
|
// passed to renderer we would assume the renderer keeps a reference of
|
||||||
// it.
|
// it.
|
||||||
meta.id = objectsRegistry.add(sender, value)
|
meta.id = objectsRegistry.add(sender, contextId, value)
|
||||||
meta.members = getObjectMembers(value)
|
meta.members = getObjectMembers(value)
|
||||||
meta.proto = getObjectPrototype(value)
|
meta.proto = getObjectPrototype(value)
|
||||||
} else if (meta.type === 'buffer') {
|
} else if (meta.type === 'buffer') {
|
||||||
|
@ -101,7 +101,7 @@ let valueToMeta = function (sender, value, optimizeSimpleObject = false) {
|
||||||
// Instead they should appear in the renderer process
|
// Instead they should appear in the renderer process
|
||||||
value.then(function () {}, function () {})
|
value.then(function () {}, function () {})
|
||||||
|
|
||||||
meta.then = valueToMeta(sender, function (onFulfilled, onRejected) {
|
meta.then = valueToMeta(sender, contextId, function (onFulfilled, onRejected) {
|
||||||
value.then(onFulfilled, onRejected)
|
value.then(onFulfilled, onRejected)
|
||||||
})
|
})
|
||||||
} else if (meta.type === 'error') {
|
} else if (meta.type === 'error') {
|
||||||
|
@ -132,12 +132,12 @@ const plainObjectToMeta = function (obj) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert Error into meta data.
|
// Convert Error into meta data.
|
||||||
const exceptionToMeta = function (sender, error) {
|
const exceptionToMeta = function (sender, contextId, error) {
|
||||||
return {
|
return {
|
||||||
type: 'exception',
|
type: 'exception',
|
||||||
message: error.message,
|
message: error.message,
|
||||||
stack: error.stack || error,
|
stack: error.stack || error,
|
||||||
cause: valueToMeta(sender, error.cause)
|
cause: valueToMeta(sender, contextId, error.cause)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -169,7 +169,7 @@ const removeRemoteListenersAndLogWarning = (sender, meta, callIntoRenderer) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert array of meta data from renderer into array of real values.
|
// Convert array of meta data from renderer into array of real values.
|
||||||
const unwrapArgs = function (sender, args) {
|
const unwrapArgs = function (sender, contextId, args) {
|
||||||
const metaToValue = function (meta) {
|
const metaToValue = function (meta) {
|
||||||
switch (meta.type) {
|
switch (meta.type) {
|
||||||
case 'value':
|
case 'value':
|
||||||
|
@ -177,7 +177,7 @@ const unwrapArgs = function (sender, args) {
|
||||||
case 'remote-object':
|
case 'remote-object':
|
||||||
return objectsRegistry.get(meta.id)
|
return objectsRegistry.get(meta.id)
|
||||||
case 'array':
|
case 'array':
|
||||||
return unwrapArgs(sender, meta.value)
|
return unwrapArgs(sender, contextId, meta.value)
|
||||||
case 'buffer':
|
case 'buffer':
|
||||||
return bufferUtils.metaToBuffer(meta.value)
|
return bufferUtils.metaToBuffer(meta.value)
|
||||||
case 'date':
|
case 'date':
|
||||||
|
@ -201,26 +201,26 @@ const unwrapArgs = function (sender, args) {
|
||||||
return returnValue
|
return returnValue
|
||||||
}
|
}
|
||||||
case 'function': {
|
case 'function': {
|
||||||
// Merge webContentsId and meta.id, since meta.id can be the same in
|
// Merge contextId and meta.id, since meta.id can be the same in
|
||||||
// different webContents.
|
// different webContents.
|
||||||
const webContentsId = sender.getId()
|
const objectId = [contextId, meta.id]
|
||||||
const objectId = [webContentsId, meta.id]
|
|
||||||
|
|
||||||
// Cache the callbacks in renderer.
|
// Cache the callbacks in renderer.
|
||||||
if (rendererFunctions.has(objectId)) {
|
if (rendererFunctions.has(objectId)) {
|
||||||
return rendererFunctions.get(objectId)
|
return rendererFunctions.get(objectId)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const webContentsId = sender.getId()
|
||||||
let callIntoRenderer = function (...args) {
|
let callIntoRenderer = function (...args) {
|
||||||
if (!sender.isDestroyed() && webContentsId === sender.getId()) {
|
if (!sender.isDestroyed() && webContentsId === sender.getId()) {
|
||||||
sender.send('ELECTRON_RENDERER_CALLBACK', meta.id, valueToMeta(sender, args))
|
sender.send('ELECTRON_RENDERER_CALLBACK', contextId, meta.id, valueToMeta(sender, contextId, args))
|
||||||
} else {
|
} else {
|
||||||
removeRemoteListenersAndLogWarning(this, meta, callIntoRenderer)
|
removeRemoteListenersAndLogWarning(this, meta, callIntoRenderer)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Object.defineProperty(callIntoRenderer, 'length', { value: meta.length })
|
Object.defineProperty(callIntoRenderer, 'length', { value: meta.length })
|
||||||
|
|
||||||
v8Util.setRemoteCallbackFreer(callIntoRenderer, meta.id, sender)
|
v8Util.setRemoteCallbackFreer(callIntoRenderer, contextId, meta.id, sender)
|
||||||
rendererFunctions.set(objectId, callIntoRenderer)
|
rendererFunctions.set(objectId, callIntoRenderer)
|
||||||
return callIntoRenderer
|
return callIntoRenderer
|
||||||
}
|
}
|
||||||
|
@ -233,18 +233,18 @@ const unwrapArgs = function (sender, args) {
|
||||||
|
|
||||||
// Call a function and send reply asynchronously if it's a an asynchronous
|
// Call a function and send reply asynchronously if it's a an asynchronous
|
||||||
// style function and the caller didn't pass a callback.
|
// style function and the caller didn't pass a callback.
|
||||||
const callFunction = function (event, func, caller, args) {
|
const callFunction = function (event, contextId, func, caller, args) {
|
||||||
const funcMarkedAsync = v8Util.getHiddenValue(func, 'asynchronous')
|
const funcMarkedAsync = v8Util.getHiddenValue(func, 'asynchronous')
|
||||||
const funcPassedCallback = typeof args[args.length - 1] === 'function'
|
const funcPassedCallback = typeof args[args.length - 1] === 'function'
|
||||||
try {
|
try {
|
||||||
if (funcMarkedAsync && !funcPassedCallback) {
|
if (funcMarkedAsync && !funcPassedCallback) {
|
||||||
args.push(function (ret) {
|
args.push(function (ret) {
|
||||||
event.returnValue = valueToMeta(event.sender, ret, true)
|
event.returnValue = valueToMeta(event.sender, contextId, ret, true)
|
||||||
})
|
})
|
||||||
func.apply(caller, args)
|
func.apply(caller, args)
|
||||||
} else {
|
} else {
|
||||||
const ret = func.apply(caller, args)
|
const ret = func.apply(caller, args)
|
||||||
event.returnValue = valueToMeta(event.sender, ret, true)
|
event.returnValue = valueToMeta(event.sender, contextId, ret, true)
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// Catch functions thrown further down in function invocation and wrap
|
// Catch functions thrown further down in function invocation and wrap
|
||||||
|
@ -257,105 +257,105 @@ const callFunction = function (event, func, caller, args) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_REQUIRE', function (event, module) {
|
ipcMain.on('ELECTRON_BROWSER_REQUIRE', function (event, contextId, module) {
|
||||||
try {
|
try {
|
||||||
event.returnValue = valueToMeta(event.sender, process.mainModule.require(module))
|
event.returnValue = valueToMeta(event.sender, contextId, process.mainModule.require(module))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_GET_BUILTIN', function (event, module) {
|
ipcMain.on('ELECTRON_BROWSER_GET_BUILTIN', function (event, contextId, module) {
|
||||||
try {
|
try {
|
||||||
event.returnValue = valueToMeta(event.sender, electron[module])
|
event.returnValue = valueToMeta(event.sender, contextId, electron[module])
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_GLOBAL', function (event, name) {
|
ipcMain.on('ELECTRON_BROWSER_GLOBAL', function (event, contextId, name) {
|
||||||
try {
|
try {
|
||||||
event.returnValue = valueToMeta(event.sender, global[name])
|
event.returnValue = valueToMeta(event.sender, contextId, global[name])
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_CURRENT_WINDOW', function (event) {
|
ipcMain.on('ELECTRON_BROWSER_CURRENT_WINDOW', function (event, contextId) {
|
||||||
try {
|
try {
|
||||||
event.returnValue = valueToMeta(event.sender, event.sender.getOwnerBrowserWindow())
|
event.returnValue = valueToMeta(event.sender, contextId, event.sender.getOwnerBrowserWindow())
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_CURRENT_WEB_CONTENTS', function (event) {
|
ipcMain.on('ELECTRON_BROWSER_CURRENT_WEB_CONTENTS', function (event, contextId) {
|
||||||
event.returnValue = valueToMeta(event.sender, event.sender)
|
event.returnValue = valueToMeta(event.sender, contextId, event.sender)
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_CONSTRUCTOR', function (event, id, args) {
|
ipcMain.on('ELECTRON_BROWSER_CONSTRUCTOR', function (event, contextId, id, args) {
|
||||||
try {
|
try {
|
||||||
args = unwrapArgs(event.sender, args)
|
args = unwrapArgs(event.sender, contextId, args)
|
||||||
let constructor = objectsRegistry.get(id)
|
let constructor = objectsRegistry.get(id)
|
||||||
|
|
||||||
if (constructor == null) {
|
if (constructor == null) {
|
||||||
throwRPCError(`Cannot call constructor on missing remote object ${id}`)
|
throwRPCError(`Cannot call constructor on missing remote object ${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
event.returnValue = valueToMeta(event.sender, new constructor(...args))
|
event.returnValue = valueToMeta(event.sender, contextId, new constructor(...args))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_FUNCTION_CALL', function (event, id, args) {
|
ipcMain.on('ELECTRON_BROWSER_FUNCTION_CALL', function (event, contextId, id, args) {
|
||||||
try {
|
try {
|
||||||
args = unwrapArgs(event.sender, args)
|
args = unwrapArgs(event.sender, contextId, args)
|
||||||
let func = objectsRegistry.get(id)
|
let func = objectsRegistry.get(id)
|
||||||
|
|
||||||
if (func == null) {
|
if (func == null) {
|
||||||
throwRPCError(`Cannot call function on missing remote object ${id}`)
|
throwRPCError(`Cannot call function on missing remote object ${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
callFunction(event, func, global, args)
|
callFunction(event, contextId, func, global, args)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_MEMBER_CONSTRUCTOR', function (event, id, method, args) {
|
ipcMain.on('ELECTRON_BROWSER_MEMBER_CONSTRUCTOR', function (event, contextId, id, method, args) {
|
||||||
try {
|
try {
|
||||||
args = unwrapArgs(event.sender, args)
|
args = unwrapArgs(event.sender, contextId, args)
|
||||||
let object = objectsRegistry.get(id)
|
let object = objectsRegistry.get(id)
|
||||||
|
|
||||||
if (object == null) {
|
if (object == null) {
|
||||||
throwRPCError(`Cannot call constructor '${method}' on missing remote object ${id}`)
|
throwRPCError(`Cannot call constructor '${method}' on missing remote object ${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
event.returnValue = valueToMeta(event.sender, new object[method](...args))
|
event.returnValue = valueToMeta(event.sender, contextId, new object[method](...args))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_MEMBER_CALL', function (event, id, method, args) {
|
ipcMain.on('ELECTRON_BROWSER_MEMBER_CALL', function (event, contextId, id, method, args) {
|
||||||
try {
|
try {
|
||||||
args = unwrapArgs(event.sender, args)
|
args = unwrapArgs(event.sender, contextId, args)
|
||||||
let obj = objectsRegistry.get(id)
|
let obj = objectsRegistry.get(id)
|
||||||
|
|
||||||
if (obj == null) {
|
if (obj == null) {
|
||||||
throwRPCError(`Cannot call function '${method}' on missing remote object ${id}`)
|
throwRPCError(`Cannot call function '${method}' on missing remote object ${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
callFunction(event, obj[method], obj, args)
|
callFunction(event, contextId, obj[method], obj, args)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_MEMBER_SET', function (event, id, name, args) {
|
ipcMain.on('ELECTRON_BROWSER_MEMBER_SET', function (event, contextId, id, name, args) {
|
||||||
try {
|
try {
|
||||||
args = unwrapArgs(event.sender, args)
|
args = unwrapArgs(event.sender, contextId, args)
|
||||||
let obj = objectsRegistry.get(id)
|
let obj = objectsRegistry.get(id)
|
||||||
|
|
||||||
if (obj == null) {
|
if (obj == null) {
|
||||||
|
@ -365,11 +365,11 @@ ipcMain.on('ELECTRON_BROWSER_MEMBER_SET', function (event, id, name, args) {
|
||||||
obj[name] = args[0]
|
obj[name] = args[0]
|
||||||
event.returnValue = null
|
event.returnValue = null
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_MEMBER_GET', function (event, id, name) {
|
ipcMain.on('ELECTRON_BROWSER_MEMBER_GET', function (event, contextId, id, name) {
|
||||||
try {
|
try {
|
||||||
let obj = objectsRegistry.get(id)
|
let obj = objectsRegistry.get(id)
|
||||||
|
|
||||||
|
@ -377,14 +377,14 @@ ipcMain.on('ELECTRON_BROWSER_MEMBER_GET', function (event, id, name) {
|
||||||
throwRPCError(`Cannot get property '${name}' on missing remote object ${id}`)
|
throwRPCError(`Cannot get property '${name}' on missing remote object ${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
event.returnValue = valueToMeta(event.sender, obj[name])
|
event.returnValue = valueToMeta(event.sender, contextId, obj[name])
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_DEREFERENCE', function (event, id) {
|
ipcMain.on('ELECTRON_BROWSER_DEREFERENCE', function (event, contextId, id) {
|
||||||
objectsRegistry.remove(event.sender.getId(), id)
|
objectsRegistry.remove(contextId, id)
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_CONTEXT_RELEASE', (e, contextId) => {
|
ipcMain.on('ELECTRON_BROWSER_CONTEXT_RELEASE', (e, contextId) => {
|
||||||
|
@ -392,16 +392,16 @@ ipcMain.on('ELECTRON_BROWSER_CONTEXT_RELEASE', (e, contextId) => {
|
||||||
e.returnValue = null
|
e.returnValue = null
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_GUEST_WEB_CONTENTS', function (event, guestInstanceId) {
|
ipcMain.on('ELECTRON_BROWSER_GUEST_WEB_CONTENTS', function (event, contextId, guestInstanceId) {
|
||||||
try {
|
try {
|
||||||
let guestViewManager = require('./guest-view-manager')
|
let guestViewManager = require('./guest-view-manager')
|
||||||
event.returnValue = valueToMeta(event.sender, guestViewManager.getGuest(guestInstanceId))
|
event.returnValue = valueToMeta(event.sender, contextId, guestViewManager.getGuest(guestInstanceId))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
ipcMain.on('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', function (event, requestId, guestInstanceId, method, ...args) {
|
ipcMain.on('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', function (event, contextId, requestId, guestInstanceId, method, ...args) {
|
||||||
try {
|
try {
|
||||||
let guestViewManager = require('./guest-view-manager')
|
let guestViewManager = require('./guest-view-manager')
|
||||||
let guest = guestViewManager.getGuest(guestInstanceId)
|
let guest = guestViewManager.getGuest(guestInstanceId)
|
||||||
|
@ -413,7 +413,7 @@ ipcMain.on('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', function (event, request
|
||||||
}
|
}
|
||||||
guest[method].apply(guest, args)
|
guest[method].apply(guest, args)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
event.returnValue = exceptionToMeta(event.sender, error)
|
event.returnValue = exceptionToMeta(event.sender, contextId, error)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,18 @@ const bufferUtils = require('../../common/buffer-utils')
|
||||||
const callbacksRegistry = new CallbacksRegistry()
|
const callbacksRegistry = new CallbacksRegistry()
|
||||||
const remoteObjectCache = v8Util.createIDWeakMap()
|
const remoteObjectCache = v8Util.createIDWeakMap()
|
||||||
|
|
||||||
|
// An unique ID that can represent current context.
|
||||||
|
const contextId = v8Util.getContextId()
|
||||||
|
|
||||||
|
// Notify the main process when current context is going to be released.
|
||||||
|
// Note that when the renderer process is destroyed, the message may not be
|
||||||
|
// sent, we also listen to the "render-view-deleted" event in the main process
|
||||||
|
// to guard that situation.
|
||||||
|
process.on('exit', () => {
|
||||||
|
const command = 'ELECTRON_BROWSER_CONTEXT_RELEASE'
|
||||||
|
ipcRenderer.sendSync(command, contextId)
|
||||||
|
})
|
||||||
|
|
||||||
// Convert the arguments object into an array of meta data.
|
// Convert the arguments object into an array of meta data.
|
||||||
function wrapArgs (args, visited = new Set()) {
|
function wrapArgs (args, visited = new Set()) {
|
||||||
const valueToMeta = (value) => {
|
const valueToMeta = (value) => {
|
||||||
|
@ -107,7 +119,7 @@ function setObjectMembers (ref, object, metaId, members) {
|
||||||
} else {
|
} else {
|
||||||
command = 'ELECTRON_BROWSER_MEMBER_CALL'
|
command = 'ELECTRON_BROWSER_MEMBER_CALL'
|
||||||
}
|
}
|
||||||
const ret = ipcRenderer.sendSync(command, metaId, member.name, wrapArgs(args))
|
const ret = ipcRenderer.sendSync(command, contextId, metaId, member.name, wrapArgs(args))
|
||||||
return metaToValue(ret)
|
return metaToValue(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -126,7 +138,7 @@ function setObjectMembers (ref, object, metaId, members) {
|
||||||
} else if (member.type === 'get') {
|
} else if (member.type === 'get') {
|
||||||
descriptor.get = () => {
|
descriptor.get = () => {
|
||||||
const command = 'ELECTRON_BROWSER_MEMBER_GET'
|
const command = 'ELECTRON_BROWSER_MEMBER_GET'
|
||||||
const meta = ipcRenderer.sendSync(command, metaId, member.name)
|
const meta = ipcRenderer.sendSync(command, contextId, metaId, member.name)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -134,7 +146,7 @@ function setObjectMembers (ref, object, metaId, members) {
|
||||||
descriptor.set = (value) => {
|
descriptor.set = (value) => {
|
||||||
const args = wrapArgs([value])
|
const args = wrapArgs([value])
|
||||||
const command = 'ELECTRON_BROWSER_MEMBER_SET'
|
const command = 'ELECTRON_BROWSER_MEMBER_SET'
|
||||||
const meta = ipcRenderer.sendSync(command, metaId, member.name, args)
|
const meta = ipcRenderer.sendSync(command, contextId, metaId, member.name, args)
|
||||||
if (meta != null) metaToValue(meta)
|
if (meta != null) metaToValue(meta)
|
||||||
return value
|
return value
|
||||||
}
|
}
|
||||||
|
@ -164,7 +176,7 @@ function proxyFunctionProperties (remoteMemberFunction, metaId, name) {
|
||||||
if (loaded) return
|
if (loaded) return
|
||||||
loaded = true
|
loaded = true
|
||||||
const command = 'ELECTRON_BROWSER_MEMBER_GET'
|
const command = 'ELECTRON_BROWSER_MEMBER_GET'
|
||||||
const meta = ipcRenderer.sendSync(command, metaId, name)
|
const meta = ipcRenderer.sendSync(command, contextId, metaId, name)
|
||||||
setObjectMembers(remoteMemberFunction, remoteMemberFunction, meta.id, meta.members)
|
setObjectMembers(remoteMemberFunction, remoteMemberFunction, meta.id, meta.members)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -224,7 +236,7 @@ function metaToValue (meta) {
|
||||||
} else {
|
} else {
|
||||||
command = 'ELECTRON_BROWSER_FUNCTION_CALL'
|
command = 'ELECTRON_BROWSER_FUNCTION_CALL'
|
||||||
}
|
}
|
||||||
const obj = ipcRenderer.sendSync(command, meta.id, wrapArgs(args))
|
const obj = ipcRenderer.sendSync(command, contextId, meta.id, wrapArgs(args))
|
||||||
return metaToValue(obj)
|
return metaToValue(obj)
|
||||||
}
|
}
|
||||||
ret = remoteFunction
|
ret = remoteFunction
|
||||||
|
@ -237,7 +249,7 @@ function metaToValue (meta) {
|
||||||
Object.defineProperty(ret.constructor, 'name', { value: meta.name })
|
Object.defineProperty(ret.constructor, 'name', { value: meta.name })
|
||||||
|
|
||||||
// Track delegate obj's lifetime & tell browser to clean up when object is GCed.
|
// Track delegate obj's lifetime & tell browser to clean up when object is GCed.
|
||||||
v8Util.setRemoteObjectFreer(ret, meta.id)
|
v8Util.setRemoteObjectFreer(ret, contextId, meta.id)
|
||||||
v8Util.setHiddenValue(ret, 'atomId', meta.id)
|
v8Util.setHiddenValue(ret, 'atomId', meta.id)
|
||||||
remoteObjectCache.set(meta.id, ret)
|
remoteObjectCache.set(meta.id, ret)
|
||||||
return ret
|
return ret
|
||||||
|
@ -264,60 +276,51 @@ function metaToException (meta) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Browser calls a callback in renderer.
|
// Browser calls a callback in renderer.
|
||||||
ipcRenderer.on('ELECTRON_RENDERER_CALLBACK', (event, id, args) => {
|
ipcRenderer.on('ELECTRON_RENDERER_CALLBACK', (event, passedContextId, id, args) => {
|
||||||
|
if (passedContextId !== contextId) {
|
||||||
|
// The invoked callback belongs to an old page in this renderer.
|
||||||
|
return
|
||||||
|
}
|
||||||
callbacksRegistry.apply(id, metaToValue(args))
|
callbacksRegistry.apply(id, metaToValue(args))
|
||||||
})
|
})
|
||||||
|
|
||||||
// A callback in browser is released.
|
// A callback in browser is released.
|
||||||
ipcRenderer.on('ELECTRON_RENDERER_RELEASE_CALLBACK', (event, id) => {
|
ipcRenderer.on('ELECTRON_RENDERER_RELEASE_CALLBACK', (event, passedContextId, id) => {
|
||||||
|
if (passedContextId !== contextId) {
|
||||||
|
// The freed callback belongs to an old page in this renderer.
|
||||||
|
return
|
||||||
|
}
|
||||||
callbacksRegistry.remove(id)
|
callbacksRegistry.remove(id)
|
||||||
})
|
})
|
||||||
|
|
||||||
process.on('exit', () => {
|
|
||||||
const command = 'ELECTRON_BROWSER_CONTEXT_RELEASE'
|
|
||||||
ipcRenderer.sendSync(command, initialContext)
|
|
||||||
})
|
|
||||||
|
|
||||||
exports.require = (module) => {
|
exports.require = (module) => {
|
||||||
const command = 'ELECTRON_BROWSER_REQUIRE'
|
const command = 'ELECTRON_BROWSER_REQUIRE'
|
||||||
const meta = ipcRenderer.sendSync(command, module)
|
const meta = ipcRenderer.sendSync(command, contextId, module)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Alias to remote.require('electron').xxx.
|
// Alias to remote.require('electron').xxx.
|
||||||
exports.getBuiltin = (module) => {
|
exports.getBuiltin = (module) => {
|
||||||
const command = 'ELECTRON_BROWSER_GET_BUILTIN'
|
const command = 'ELECTRON_BROWSER_GET_BUILTIN'
|
||||||
const meta = ipcRenderer.sendSync(command, module)
|
const meta = ipcRenderer.sendSync(command, contextId, module)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.getCurrentWindow = () => {
|
exports.getCurrentWindow = () => {
|
||||||
const command = 'ELECTRON_BROWSER_CURRENT_WINDOW'
|
const command = 'ELECTRON_BROWSER_CURRENT_WINDOW'
|
||||||
const meta = ipcRenderer.sendSync(command)
|
const meta = ipcRenderer.sendSync(command, contextId)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get current WebContents object.
|
// Get current WebContents object.
|
||||||
exports.getCurrentWebContents = () => {
|
exports.getCurrentWebContents = () => {
|
||||||
return metaToValue(ipcRenderer.sendSync('ELECTRON_BROWSER_CURRENT_WEB_CONTENTS'))
|
return metaToValue(ipcRenderer.sendSync('ELECTRON_BROWSER_CURRENT_WEB_CONTENTS', contextId))
|
||||||
}
|
|
||||||
|
|
||||||
const CONTEXT_ARG = '--context-id='
|
|
||||||
let initialContext = process.argv.find(arg => arg.startsWith(CONTEXT_ARG))
|
|
||||||
if (process.webContentsId) {
|
|
||||||
// set by sandbox renderer init script
|
|
||||||
initialContext = process.webContentsId
|
|
||||||
} else if (initialContext) {
|
|
||||||
initialContext = parseInt(initialContext.substr(CONTEXT_ARG.length), 10)
|
|
||||||
} else {
|
|
||||||
// if not available, pull from remote
|
|
||||||
initialContext = exports.getCurrentWebContents().getId()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get a global object in browser.
|
// Get a global object in browser.
|
||||||
exports.getGlobal = (name) => {
|
exports.getGlobal = (name) => {
|
||||||
const command = 'ELECTRON_BROWSER_GLOBAL'
|
const command = 'ELECTRON_BROWSER_GLOBAL'
|
||||||
const meta = ipcRenderer.sendSync(command, name)
|
const meta = ipcRenderer.sendSync(command, contextId, name)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -334,7 +337,7 @@ exports.createFunctionWithReturnValue = (returnValue) => {
|
||||||
// Get the guest WebContents from guestInstanceId.
|
// Get the guest WebContents from guestInstanceId.
|
||||||
exports.getGuestWebContents = (guestInstanceId) => {
|
exports.getGuestWebContents = (guestInstanceId) => {
|
||||||
const command = 'ELECTRON_BROWSER_GUEST_WEB_CONTENTS'
|
const command = 'ELECTRON_BROWSER_GUEST_WEB_CONTENTS'
|
||||||
const meta = ipcRenderer.sendSync(command, guestInstanceId)
|
const meta = ipcRenderer.sendSync(command, contextId, guestInstanceId)
|
||||||
return metaToValue(meta)
|
return metaToValue(meta)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,9 @@ const webViewConstants = require('./web-view-constants')
|
||||||
|
|
||||||
const hasProp = {}.hasOwnProperty
|
const hasProp = {}.hasOwnProperty
|
||||||
|
|
||||||
|
// An unique ID that can represent current context.
|
||||||
|
const contextId = v8Util.getContextId()
|
||||||
|
|
||||||
// ID generator.
|
// ID generator.
|
||||||
let nextId = 0
|
let nextId = 0
|
||||||
|
|
||||||
|
@ -344,6 +347,7 @@ const registerWebViewElement = function () {
|
||||||
'inspectElement',
|
'inspectElement',
|
||||||
'setAudioMuted',
|
'setAudioMuted',
|
||||||
'isAudioMuted',
|
'isAudioMuted',
|
||||||
|
'isCurrentlyAudible',
|
||||||
'undo',
|
'undo',
|
||||||
'redo',
|
'redo',
|
||||||
'cut',
|
'cut',
|
||||||
|
@ -396,7 +400,7 @@ const registerWebViewElement = function () {
|
||||||
const createNonBlockHandler = function (m) {
|
const createNonBlockHandler = function (m) {
|
||||||
return function (...args) {
|
return function (...args) {
|
||||||
const internal = v8Util.getHiddenValue(this, 'internal')
|
const internal = v8Util.getHiddenValue(this, 'internal')
|
||||||
ipcRenderer.send('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', null, internal.guestInstanceId, m, ...args)
|
ipcRenderer.send('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', contextId, null, internal.guestInstanceId, m, ...args)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const method of nonblockMethods) {
|
for (const method of nonblockMethods) {
|
||||||
|
@ -410,7 +414,7 @@ const registerWebViewElement = function () {
|
||||||
hasUserGesture = false
|
hasUserGesture = false
|
||||||
}
|
}
|
||||||
const requestId = getNextId()
|
const requestId = getNextId()
|
||||||
ipcRenderer.send('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', requestId, internal.guestInstanceId, 'executeJavaScript', code, hasUserGesture)
|
ipcRenderer.send('ELECTRON_BROWSER_ASYNC_CALL_TO_GUEST_VIEW', contextId, requestId, internal.guestInstanceId, 'executeJavaScript', code, hasUserGesture)
|
||||||
ipcRenderer.once(`ELECTRON_RENDERER_ASYNC_CALL_TO_GUEST_VIEW_RESPONSE_${requestId}`, function (event, result) {
|
ipcRenderer.once(`ELECTRON_RENDERER_ASYNC_CALL_TO_GUEST_VIEW_RESPONSE_${requestId}`, function (event, result) {
|
||||||
if (callback) callback(result)
|
if (callback) callback(result)
|
||||||
})
|
})
|
||||||
|
|
|
@ -112,10 +112,8 @@ if (preloadSrc) {
|
||||||
${preloadSrc}
|
${preloadSrc}
|
||||||
})`
|
})`
|
||||||
|
|
||||||
// eval in window scope:
|
// eval in window scope
|
||||||
// http://www.ecma-international.org/ecma-262/5.1/#sec-10.4.2
|
const preloadFn = binding.createPreloadScript(preloadWrapperSrc)
|
||||||
const geval = eval
|
|
||||||
const preloadFn = geval(preloadWrapperSrc)
|
|
||||||
const {setImmediate, clearImmediate} = require('timers')
|
const {setImmediate, clearImmediate} = require('timers')
|
||||||
preloadFn(preloadRequire, preloadProcess, Buffer, global, setImmediate, clearImmediate)
|
preloadFn(preloadRequire, preloadProcess, Buffer, global, setImmediate, clearImmediate)
|
||||||
} else if (preloadError) {
|
} else if (preloadError) {
|
||||||
|
|
|
@ -120,7 +120,7 @@ async function callAppVeyor (targetBranch, job, options) {
|
||||||
let appVeyorResponse = await makeRequest(requestOpts, true).catch(err => {
|
let appVeyorResponse = await makeRequest(requestOpts, true).catch(err => {
|
||||||
console.log('Error calling AppVeyor:', err)
|
console.log('Error calling AppVeyor:', err)
|
||||||
})
|
})
|
||||||
const buildUrl = `https://windows-ci.electronjs.org/project/AppVeyor/electron/build/${appVeyorResponse.version}`
|
const buildUrl = `https://windows-ci.electronjs.org/project/AppVeyor/${appVeyorJobs[job]}/build/${appVeyorResponse.version}`
|
||||||
console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`)
|
console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
'use strict'
|
'use strict'
|
||||||
|
|
||||||
/* eslint-disable no-unused-expressions */
|
const chai = require('chai')
|
||||||
|
const dirtyChai = require('dirty-chai')
|
||||||
const {expect} = require('chai')
|
|
||||||
const {nativeImage} = require('electron')
|
const {nativeImage} = require('electron')
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
|
|
||||||
|
const {expect} = chai
|
||||||
|
chai.use(dirtyChai)
|
||||||
|
|
||||||
describe('nativeImage module', () => {
|
describe('nativeImage module', () => {
|
||||||
const ImageFormat = {
|
const ImageFormat = {
|
||||||
PNG: 'png',
|
PNG: 'png',
|
||||||
|
@ -111,16 +113,16 @@ describe('nativeImage module', () => {
|
||||||
expect(empty.toDataURL()).to.equal('data:image/png;base64,')
|
expect(empty.toDataURL()).to.equal('data:image/png;base64,')
|
||||||
expect(empty.toDataURL({scaleFactor: 2.0})).to.equal('data:image/png;base64,')
|
expect(empty.toDataURL({scaleFactor: 2.0})).to.equal('data:image/png;base64,')
|
||||||
expect(empty.getSize()).to.deep.equal({width: 0, height: 0})
|
expect(empty.getSize()).to.deep.equal({width: 0, height: 0})
|
||||||
expect(empty.getBitmap()).to.be.empty
|
expect(empty.getBitmap()).to.be.empty()
|
||||||
expect(empty.getBitmap({scaleFactor: 2.0})).to.be.empty
|
expect(empty.getBitmap({scaleFactor: 2.0})).to.be.empty()
|
||||||
expect(empty.toBitmap()).to.be.empty
|
expect(empty.toBitmap()).to.be.empty()
|
||||||
expect(empty.toBitmap({scaleFactor: 2.0})).to.be.empty
|
expect(empty.toBitmap({scaleFactor: 2.0})).to.be.empty()
|
||||||
expect(empty.toJPEG(100)).to.be.empty
|
expect(empty.toJPEG(100)).to.be.empty()
|
||||||
expect(empty.toPNG()).to.be.empty
|
expect(empty.toPNG()).to.be.empty()
|
||||||
expect(empty.toPNG({scaleFactor: 2.0})).to.be.empty
|
expect(empty.toPNG({scaleFactor: 2.0})).to.be.empty()
|
||||||
|
|
||||||
if (process.platform === 'darwin') {
|
if (process.platform === 'darwin') {
|
||||||
expect(empty.getNativeHandle()).to.be.empty
|
expect(empty.getNativeHandle()).to.be.empty()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -135,7 +137,7 @@ describe('nativeImage module', () => {
|
||||||
|
|
||||||
const imageB = nativeImage.createFromBuffer(imageA.toPNG())
|
const imageB = nativeImage.createFromBuffer(imageA.toPNG())
|
||||||
expect(imageB.getSize()).to.deep.equal({width: 538, height: 190})
|
expect(imageB.getSize()).to.deep.equal({width: 538, height: 190})
|
||||||
expect(imageA.toBitmap().equals(imageB.toBitmap())).to.be.true
|
expect(imageA.toBitmap().equals(imageB.toBitmap())).to.be.true()
|
||||||
|
|
||||||
const imageC = nativeImage.createFromBuffer(imageA.toJPEG(100))
|
const imageC = nativeImage.createFromBuffer(imageA.toJPEG(100))
|
||||||
expect(imageC.getSize()).to.deep.equal({width: 538, height: 190})
|
expect(imageC.getSize()).to.deep.equal({width: 538, height: 190})
|
||||||
|
@ -214,7 +216,7 @@ describe('nativeImage module', () => {
|
||||||
expect(imageTwo.getSize()).to.deep.equal(
|
expect(imageTwo.getSize()).to.deep.equal(
|
||||||
{width: imageData.width, height: imageData.height})
|
{width: imageData.width, height: imageData.height})
|
||||||
|
|
||||||
expect(imageOne.toBitmap().equals(imageTwo.toBitmap())).to.be.true
|
expect(imageOne.toBitmap().equals(imageTwo.toBitmap())).to.be.true()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('supports a scale factor', () => {
|
it('supports a scale factor', () => {
|
||||||
|
@ -249,7 +251,7 @@ describe('nativeImage module', () => {
|
||||||
expect(imageC.getSize()).to.deep.equal(
|
expect(imageC.getSize()).to.deep.equal(
|
||||||
{width: imageData.width, height: imageData.height})
|
{width: imageData.width, height: imageData.height})
|
||||||
|
|
||||||
expect(imageB.toBitmap().equals(imageC.toBitmap())).to.be.true
|
expect(imageB.toBitmap().equals(imageC.toBitmap())).to.be.true()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('supports a scale factor', () => {
|
it('supports a scale factor', () => {
|
||||||
|
@ -280,21 +282,21 @@ describe('nativeImage module', () => {
|
||||||
it('loads images from paths relative to the current working directory', () => {
|
it('loads images from paths relative to the current working directory', () => {
|
||||||
const imagePath = `.${path.sep}${path.join('spec', 'fixtures', 'assets', 'logo.png')}`
|
const imagePath = `.${path.sep}${path.join('spec', 'fixtures', 'assets', 'logo.png')}`
|
||||||
const image = nativeImage.createFromPath(imagePath)
|
const image = nativeImage.createFromPath(imagePath)
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('loads images from paths with `.` segments', () => {
|
it('loads images from paths with `.` segments', () => {
|
||||||
const imagePath = `${path.join(__dirname, 'fixtures')}${path.sep}.${path.sep}${path.join('assets', 'logo.png')}`
|
const imagePath = `${path.join(__dirname, 'fixtures')}${path.sep}.${path.sep}${path.join('assets', 'logo.png')}`
|
||||||
const image = nativeImage.createFromPath(imagePath)
|
const image = nativeImage.createFromPath(imagePath)
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('loads images from paths with `..` segments', () => {
|
it('loads images from paths with `..` segments', () => {
|
||||||
const imagePath = `${path.join(__dirname, 'fixtures', 'api')}${path.sep}..${path.sep}${path.join('assets', 'logo.png')}`
|
const imagePath = `${path.join(__dirname, 'fixtures', 'api')}${path.sep}..${path.sep}${path.join('assets', 'logo.png')}`
|
||||||
const image = nativeImage.createFromPath(imagePath)
|
const image = nativeImage.createFromPath(imagePath)
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
expect(image.getSize()).to.deep.equal({width: 538, height: 190})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -325,7 +327,7 @@ describe('nativeImage module', () => {
|
||||||
|
|
||||||
const imagePath = path.join(__dirname, 'fixtures', 'assets', 'icon.ico')
|
const imagePath = path.join(__dirname, 'fixtures', 'assets', 'icon.ico')
|
||||||
const image = nativeImage.createFromPath(imagePath)
|
const image = nativeImage.createFromPath(imagePath)
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 256, height: 256})
|
expect(image.getSize()).to.deep.equal({width: 256, height: 256})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -355,7 +357,7 @@ describe('nativeImage module', () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
const image = nativeImage.createFromNamedImage('NSActionTemplate')
|
const image = nativeImage.createFromNamedImage('NSActionTemplate')
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('returns allows an HSL shift for a valid image on darwin', function () {
|
it('returns allows an HSL shift for a valid image on darwin', function () {
|
||||||
|
@ -366,7 +368,7 @@ describe('nativeImage module', () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
const image = nativeImage.createFromNamedImage('NSActionTemplate', [0.5, 0.2, 0.8])
|
const image = nativeImage.createFromNamedImage('NSActionTemplate', [0.5, 0.2, 0.8])
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -425,7 +427,7 @@ describe('nativeImage module', () => {
|
||||||
const cropB = image.crop({width: 25, height: 64, x: 30, y: 40})
|
const cropB = image.crop({width: 25, height: 64, x: 30, y: 40})
|
||||||
expect(cropA.getSize()).to.deep.equal({width: 25, height: 64})
|
expect(cropA.getSize()).to.deep.equal({width: 25, height: 64})
|
||||||
expect(cropB.getSize()).to.deep.equal({width: 25, height: 64})
|
expect(cropB.getSize()).to.deep.equal({width: 25, height: 64})
|
||||||
expect(cropA.toPNG().equals(cropB.toPNG())).to.be.false
|
expect(cropA.toPNG().equals(cropB.toPNG())).to.be.false()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -471,7 +473,7 @@ describe('nativeImage module', () => {
|
||||||
buffer: 'invalid'
|
buffer: 'invalid'
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 1, height: 1})
|
expect(image.getSize()).to.deep.equal({width: 1, height: 1})
|
||||||
|
|
||||||
expect(image.toDataURL({scaleFactor: 1.0})).to.equal(imageDataOne.dataUrl)
|
expect(image.toDataURL({scaleFactor: 1.0})).to.equal(imageDataOne.dataUrl)
|
||||||
|
@ -506,7 +508,7 @@ describe('nativeImage module', () => {
|
||||||
dataURL: 'invalid'
|
dataURL: 'invalid'
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(image.isEmpty()).to.be.false
|
expect(image.isEmpty()).to.be.false()
|
||||||
expect(image.getSize()).to.deep.equal({width: 1, height: 1})
|
expect(image.getSize()).to.deep.equal({width: 1, height: 1})
|
||||||
|
|
||||||
expect(image.toDataURL({scaleFactor: 1.0})).to.equal(imageDataOne.dataUrl)
|
expect(image.toDataURL({scaleFactor: 1.0})).to.equal(imageDataOne.dataUrl)
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
const assert = require('assert')
|
const chai = require('chai')
|
||||||
|
const dirtyChai = require('dirty-chai')
|
||||||
const http = require('http')
|
const http = require('http')
|
||||||
const fs = require('fs')
|
const fs = require('fs')
|
||||||
const os = require('os')
|
const os = require('os')
|
||||||
|
@ -10,13 +11,15 @@ const appPath = path.join(__dirname, 'fixtures', 'api', 'net-log')
|
||||||
const dumpFile = path.join(os.tmpdir(), 'net_log.json')
|
const dumpFile = path.join(os.tmpdir(), 'net_log.json')
|
||||||
const dumpFileDynamic = path.join(os.tmpdir(), 'net_log_dynamic.json')
|
const dumpFileDynamic = path.join(os.tmpdir(), 'net_log_dynamic.json')
|
||||||
|
|
||||||
|
const {expect} = chai
|
||||||
|
chai.use(dirtyChai)
|
||||||
const isCI = remote.getGlobal('isCi')
|
const isCI = remote.getGlobal('isCi')
|
||||||
|
|
||||||
describe('netLog module', () => {
|
describe('netLog module', () => {
|
||||||
let server
|
let server
|
||||||
const connections = new Set()
|
const connections = new Set()
|
||||||
|
|
||||||
before((done) => {
|
before(done => {
|
||||||
server = http.createServer()
|
server = http.createServer()
|
||||||
server.listen(0, '127.0.0.1', () => {
|
server.listen(0, '127.0.0.1', () => {
|
||||||
server.url = `http://127.0.0.1:${server.address().port}`
|
server.url = `http://127.0.0.1:${server.address().port}`
|
||||||
|
@ -33,7 +36,7 @@ describe('netLog module', () => {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
after((done) => {
|
after(done => {
|
||||||
for (const connection of connections) {
|
for (const connection of connections) {
|
||||||
connection.destroy()
|
connection.destroy()
|
||||||
}
|
}
|
||||||
|
@ -52,36 +55,35 @@ describe('netLog module', () => {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should begin and end logging to file when .startLogging() and .stopLogging() is called', (done) => {
|
it('should begin and end logging to file when .startLogging() and .stopLogging() is called', done => {
|
||||||
assert(!netLog.currentlyLogging)
|
expect(netLog.currentlyLogging).to.be.false()
|
||||||
assert.equal(netLog.currentlyLoggingPath, '')
|
expect(netLog.currentlyLoggingPath).to.equal('')
|
||||||
|
|
||||||
netLog.startLogging(dumpFileDynamic)
|
netLog.startLogging(dumpFileDynamic)
|
||||||
|
|
||||||
assert(netLog.currentlyLogging)
|
expect(netLog.currentlyLogging).to.be.true()
|
||||||
assert.equal(netLog.currentlyLoggingPath, dumpFileDynamic)
|
expect(netLog.currentlyLoggingPath).to.equal(dumpFileDynamic)
|
||||||
|
|
||||||
netLog.stopLogging((path) => {
|
netLog.stopLogging((path) => {
|
||||||
assert(!netLog.currentlyLogging)
|
expect(netLog.currentlyLogging).to.be.false()
|
||||||
assert.equal(netLog.currentlyLoggingPath, '')
|
expect(netLog.currentlyLoggingPath).to.equal('')
|
||||||
|
|
||||||
assert.equal(path, dumpFileDynamic)
|
expect(path).to.equal(dumpFileDynamic)
|
||||||
|
expect(fs.existsSync(dumpFileDynamic)).to.be.true()
|
||||||
assert(fs.existsSync(dumpFileDynamic))
|
|
||||||
|
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should silence when .stopLogging() is called without calling .startLogging()', (done) => {
|
it('should silence when .stopLogging() is called without calling .startLogging()', done => {
|
||||||
assert(!netLog.currentlyLogging)
|
expect(netLog.currentlyLogging).to.be.false()
|
||||||
assert.equal(netLog.currentlyLoggingPath, '')
|
expect(netLog.currentlyLoggingPath).to.equal('')
|
||||||
|
|
||||||
netLog.stopLogging((path) => {
|
netLog.stopLogging(path => {
|
||||||
assert(!netLog.currentlyLogging)
|
expect(netLog.currentlyLogging).to.be.false()
|
||||||
assert.equal(netLog.currentlyLoggingPath, '')
|
expect(netLog.currentlyLoggingPath).to.equal('')
|
||||||
|
|
||||||
assert.equal(path, '')
|
expect(path).to.equal('')
|
||||||
|
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
|
@ -89,7 +91,7 @@ describe('netLog module', () => {
|
||||||
|
|
||||||
// The following tests are skipped on Linux CI
|
// The following tests are skipped on Linux CI
|
||||||
|
|
||||||
it('should begin and end logging automatically when --log-net-log is passed', (done) => {
|
it('should begin and end logging automatically when --log-net-log is passed', done => {
|
||||||
if (isCI && process.platform === 'linux') {
|
if (isCI && process.platform === 'linux') {
|
||||||
done()
|
done()
|
||||||
return
|
return
|
||||||
|
@ -103,12 +105,12 @@ describe('netLog module', () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
appProcess.once('exit', () => {
|
appProcess.once('exit', () => {
|
||||||
assert(fs.existsSync(dumpFile))
|
expect(fs.existsSync(dumpFile)).to.be.true()
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should begin and end logging automtically when --log-net-log is passed, and behave correctly when .startLogging() and .stopLogging() is called', (done) => {
|
it('should begin and end logging automtically when --log-net-log is passed, and behave correctly when .startLogging() and .stopLogging() is called', done => {
|
||||||
if (isCI && process.platform === 'linux') {
|
if (isCI && process.platform === 'linux') {
|
||||||
done()
|
done()
|
||||||
return
|
return
|
||||||
|
@ -123,18 +125,18 @@ describe('netLog module', () => {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
appProcess.stdout.on('data', (data) => {
|
appProcess.stdout.on('data', data => {
|
||||||
console.log(data.toString())
|
console.log(data.toString())
|
||||||
})
|
})
|
||||||
|
|
||||||
appProcess.once('exit', () => {
|
appProcess.once('exit', () => {
|
||||||
assert(fs.existsSync(dumpFile))
|
expect(fs.existsSync(dumpFile)).to.be.true()
|
||||||
assert(fs.existsSync(dumpFileDynamic))
|
expect(fs.existsSync(dumpFileDynamic)).to.be.true()
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should end logging automatically when only .startLogging() is called', (done) => {
|
it('should end logging automatically when only .startLogging() is called', done => {
|
||||||
if (isCI && process.platform === 'linux') {
|
if (isCI && process.platform === 'linux') {
|
||||||
done()
|
done()
|
||||||
return
|
return
|
||||||
|
@ -149,7 +151,7 @@ describe('netLog module', () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
appProcess.once('exit', () => {
|
appProcess.once('exit', () => {
|
||||||
assert(fs.existsSync(dumpFileDynamic))
|
expect(fs.existsSync(dumpFileDynamic)).to.be.true()
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
//
|
//
|
||||||
// See https://pypi.python.org/pypi/python-dbusmock to read about dbusmock.
|
// See https://pypi.python.org/pypi/python-dbusmock to read about dbusmock.
|
||||||
|
|
||||||
const assert = require('assert')
|
const {expect} = require('chai')
|
||||||
const dbus = require('dbus-native')
|
const dbus = require('dbus-native')
|
||||||
const Promise = require('bluebird')
|
const Promise = require('bluebird')
|
||||||
|
|
||||||
|
@ -27,12 +27,12 @@ const skip = process.platform !== 'linux' ||
|
||||||
before(async () => {
|
before(async () => {
|
||||||
// init app
|
// init app
|
||||||
app.setName(appName)
|
app.setName(appName)
|
||||||
app.setDesktopName(appName + '.desktop')
|
app.setDesktopName(`${appName}.desktop`)
|
||||||
// init dbus
|
// init dbus
|
||||||
const path = '/org/freedesktop/Notifications'
|
const path = '/org/freedesktop/Notifications'
|
||||||
const iface = 'org.freedesktop.DBus.Mock'
|
const iface = 'org.freedesktop.DBus.Mock'
|
||||||
const bus = dbus.sessionBus()
|
const bus = dbus.sessionBus()
|
||||||
console.log('session bus: ' + process.env.DBUS_SESSION_BUS_ADDRESS)
|
console.log(`session bus: ${process.env.DBUS_SESSION_BUS_ADDRESS}`)
|
||||||
const service = bus.getService(serviceName)
|
const service = bus.getService(serviceName)
|
||||||
const getInterface = Promise.promisify(service.getInterface, {context: service})
|
const getInterface = Promise.promisify(service.getInterface, {context: service})
|
||||||
mock = await getInterface(path, iface)
|
mock = await getInterface(path, iface)
|
||||||
|
@ -48,10 +48,10 @@ const skip = process.platform !== 'linux' ||
|
||||||
app.setVersion(realAppVersion)
|
app.setVersion(realAppVersion)
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('Notification module using ' + serviceName, () => {
|
describe(`Notification module using ${serviceName}`, () => {
|
||||||
function onMethodCalled (done) {
|
function onMethodCalled (done) {
|
||||||
function cb (name) {
|
function cb (name) {
|
||||||
console.log('onMethodCalled: ' + name)
|
console.log(`onMethodCalled: ${name}`)
|
||||||
if (name === 'Notify') {
|
if (name === 'Notify') {
|
||||||
mock.removeListener('MethodCalled', cb)
|
mock.removeListener('MethodCalled', cb)
|
||||||
console.log('done')
|
console.log('done')
|
||||||
|
@ -83,7 +83,7 @@ const skip = process.platform !== 'linux' ||
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
before((done) => {
|
before(done => {
|
||||||
mock.on('MethodCalled', onMethodCalled(done))
|
mock.on('MethodCalled', onMethodCalled(done))
|
||||||
// lazy load Notification after we listen to MethodCalled mock signal
|
// lazy load Notification after we listen to MethodCalled mock signal
|
||||||
Notification = require('electron').remote.Notification
|
Notification = require('electron').remote.Notification
|
||||||
|
@ -98,14 +98,16 @@ const skip = process.platform !== 'linux' ||
|
||||||
n.show()
|
n.show()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should call ' + serviceName + ' to show notifications', async () => {
|
it(`should call ${serviceName} to show notifications`, async () => {
|
||||||
const calls = await getCalls()
|
const calls = await getCalls()
|
||||||
assert(calls.length >= 1)
|
expect(calls).to.be.an('array').of.lengthOf.at.least(1)
|
||||||
|
|
||||||
let lastCall = calls[calls.length - 1]
|
let lastCall = calls[calls.length - 1]
|
||||||
let methodName = lastCall[1]
|
let methodName = lastCall[1]
|
||||||
assert.equal(methodName, 'Notify')
|
expect(methodName).to.equal('Notify')
|
||||||
|
|
||||||
let args = unmarshalDBusNotifyArgs(lastCall[2])
|
let args = unmarshalDBusNotifyArgs(lastCall[2])
|
||||||
assert.deepEqual(args, {
|
expect(args).to.deep.equal({
|
||||||
app_name: appName,
|
app_name: appName,
|
||||||
replaces_id: 0,
|
replaces_id: 0,
|
||||||
app_icon: '',
|
app_icon: '',
|
||||||
|
|
|
@ -4,12 +4,18 @@ const assert = require('assert')
|
||||||
const http = require('http')
|
const http = require('http')
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const {closeWindow} = require('./window-helpers')
|
const {closeWindow} = require('./window-helpers')
|
||||||
|
const {emittedOnce} = require('./events-helpers')
|
||||||
|
const chai = require('chai')
|
||||||
|
const dirtyChai = require('dirty-chai')
|
||||||
|
|
||||||
const {ipcRenderer, remote} = require('electron')
|
const {ipcRenderer, remote} = require('electron')
|
||||||
const {BrowserWindow, webContents, ipcMain, session} = remote
|
const {BrowserWindow, webContents, ipcMain, session} = remote
|
||||||
|
const {expect} = chai
|
||||||
|
|
||||||
const isCi = remote.getGlobal('isCi')
|
const isCi = remote.getGlobal('isCi')
|
||||||
|
|
||||||
|
chai.use(dirtyChai)
|
||||||
|
|
||||||
/* The whole webContents API doesn't use standard callbacks */
|
/* The whole webContents API doesn't use standard callbacks */
|
||||||
/* eslint-disable standard/no-callback-literal */
|
/* eslint-disable standard/no-callback-literal */
|
||||||
|
|
||||||
|
@ -116,6 +122,21 @@ describe('webContents module', () => {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('isCurrentlyAudible() API', () => {
|
||||||
|
it('returns whether audio is playing', async () => {
|
||||||
|
w.loadURL(`file://${path.join(__dirname, 'fixtures', 'api', 'is-currently-audible.html')}`)
|
||||||
|
w.show()
|
||||||
|
await emittedOnce(w.webContents, 'did-finish-load')
|
||||||
|
|
||||||
|
expect(w.webContents.isCurrentlyAudible()).to.be.false()
|
||||||
|
|
||||||
|
w.webContents.send('play')
|
||||||
|
await emittedOnce(ipcMain, 'playing')
|
||||||
|
|
||||||
|
expect(w.webContents.isCurrentlyAudible()).to.be.true()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('getWebPreferences() API', () => {
|
describe('getWebPreferences() API', () => {
|
||||||
it('should not crash when called for devTools webContents', (done) => {
|
it('should not crash when called for devTools webContents', (done) => {
|
||||||
w.webContents.openDevTools()
|
w.webContents.openDevTools()
|
||||||
|
|
21
spec/fixtures/api/is-currently-audible.html
vendored
Normal file
21
spec/fixtures/api/is-currently-audible.html
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
<html>
|
||||||
|
<body>
|
||||||
|
<div id="video"></div>
|
||||||
|
<script type="text/javascript" charset="utf-8">
|
||||||
|
const {ipcRenderer} = window.top != null ? window.top.require('electron') : require('electron')
|
||||||
|
ipcRenderer.on('play', (event) => {
|
||||||
|
const context = new window.AudioContext();
|
||||||
|
const oscillator = context.createOscillator();
|
||||||
|
|
||||||
|
// A beep
|
||||||
|
oscillator.type = 'sine';
|
||||||
|
oscillator.frequency.value = 440
|
||||||
|
oscillator.connect(context.destination)
|
||||||
|
oscillator.start()
|
||||||
|
|
||||||
|
// It'll take a few ms before the beep shows up
|
||||||
|
setTimeout(() => event.sender.send('playing'), 100)
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
10
spec/fixtures/module/preload-context.js
vendored
Normal file
10
spec/fixtures/module/preload-context.js
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
var test = 'test' // eslint-disable-line
|
||||||
|
|
||||||
|
const types = {
|
||||||
|
require: typeof require,
|
||||||
|
electron: typeof electron,
|
||||||
|
window: typeof window,
|
||||||
|
localVar: typeof window.test
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(JSON.stringify(types))
|
|
@ -249,6 +249,22 @@ describe('<webview> tag', function () {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('runs in the correct scope when sandboxed', async () => {
|
||||||
|
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
||||||
|
preload: `${fixtures}/module/preload-context.js`,
|
||||||
|
src: `file://${fixtures}/api/blank.html`,
|
||||||
|
webpreferences: 'sandbox=yes'
|
||||||
|
})
|
||||||
|
|
||||||
|
const types = JSON.parse(message)
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function', // arguments passed to it should be availale
|
||||||
|
electron: 'undefined', // objects from the scope it is called from should not be available
|
||||||
|
window: 'object', // the window object should be available
|
||||||
|
localVar: 'undefined' // but local variables should not be exposed to the window
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
it('preload script can require modules that still use "process" and "Buffer" when nodeintegration is off', async () => {
|
it('preload script can require modules that still use "process" and "Buffer" when nodeintegration is off', async () => {
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
||||||
preload: `${fixtures}/module/preload-node-off-wrapper.js`,
|
preload: `${fixtures}/module/preload-node-off-wrapper.js`,
|
||||||
|
|
2
vendor/libchromiumcontent
vendored
2
vendor/libchromiumcontent
vendored
|
@ -1 +1 @@
|
||||||
Subproject commit 5ad14b556198ba9a5422154aba2f9b7914dc528c
|
Subproject commit a55a9ce536db60702630c4b9d94dcb2145fc3b24
|
Loading…
Reference in a new issue