feat: implement allowFileAccess loadExtension option (#25198)

Co-authored-by: Samuel Maddock <samuel.maddock@gmail.com>
Co-authored-by: Jeremy Rose <jeremya@chromium.org>
This commit is contained in:
Сковорода Никита Андреевич 2021-02-02 01:41:08 +03:00 committed by GitHub
parent a75cd89d2a
commit a5e9af330f
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 38 additions and 11 deletions

View file

@ -15,7 +15,7 @@ extension capabilities.
Electron only supports loading unpacked extensions (i.e., `.crx` files do not Electron only supports loading unpacked extensions (i.e., `.crx` files do not
work). Extensions are installed per-`session`. To load an extension, call work). Extensions are installed per-`session`. To load an extension, call
[`ses.loadExtension`](session.md#sesloadextensionpath): [`ses.loadExtension`](session.md#sesloadextensionpath-options):
```js ```js
const { session } = require('electron') const { session } = require('electron')

View file

@ -750,9 +750,13 @@ will not work on non-persistent (in-memory) sessions.
**Note:** On macOS and Windows 10 this word will be removed from the OS custom dictionary as well **Note:** On macOS and Windows 10 this word will be removed from the OS custom dictionary as well
#### `ses.loadExtension(path)` #### `ses.loadExtension(path[, options])`
* `path` String - Path to a directory containing an unpacked Chrome extension * `path` String - Path to a directory containing an unpacked Chrome extension
* `options` Object (optional)
* `allowFileAccess` Boolean - Whether to allow the extension to read local files over `file://`
protocol and inject content scripts into `file://` pages. This is required e.g. for loading
devtools extensions on `file://` URLs. Defaults to false.
Returns `Promise<Extension>` - resolves when the extension is loaded. Returns `Promise<Extension>` - resolves when the extension is loaded.
@ -775,7 +779,11 @@ const { app, session } = require('electron')
const path = require('path') const path = require('path')
app.on('ready', async () => { app.on('ready', async () => {
await session.defaultSession.loadExtension(path.join(__dirname, 'react-devtools')) await session.defaultSession.loadExtension(
path.join(__dirname, 'react-devtools'),
// allowFileAccess is required to load the devtools extension on file:// URLs.
{ allowFileAccess: true }
)
// Note that in order to use the React DevTools extension, you'll need to // Note that in order to use the React DevTools extension, you'll need to
// download and unzip a copy of the extension. // download and unzip a copy of the extension.
}) })

View file

@ -795,7 +795,8 @@ std::vector<base::FilePath> Session::GetPreloads() const {
#if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS) #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
v8::Local<v8::Promise> Session::LoadExtension( v8::Local<v8::Promise> Session::LoadExtension(
const base::FilePath& extension_path) { const base::FilePath& extension_path,
gin::Arguments* args) {
v8::Isolate* isolate = JavascriptEnvironment::GetIsolate(); v8::Isolate* isolate = JavascriptEnvironment::GetIsolate();
gin_helper::Promise<const extensions::Extension*> promise(isolate); gin_helper::Promise<const extensions::Extension*> promise(isolate);
v8::Local<v8::Promise> handle = promise.GetHandle(); v8::Local<v8::Promise> handle = promise.GetHandle();
@ -812,10 +813,19 @@ v8::Local<v8::Promise> Session::LoadExtension(
return handle; return handle;
} }
int load_flags = extensions::Extension::FOLLOW_SYMLINKS_ANYWHERE;
gin_helper::Dictionary options;
if (args->GetNext(&options)) {
bool allowFileAccess = false;
options.Get("allowFileAccess", &allowFileAccess);
if (allowFileAccess)
load_flags |= extensions::Extension::ALLOW_FILE_ACCESS;
}
auto* extension_system = static_cast<extensions::ElectronExtensionSystem*>( auto* extension_system = static_cast<extensions::ElectronExtensionSystem*>(
extensions::ExtensionSystem::Get(browser_context())); extensions::ExtensionSystem::Get(browser_context()));
extension_system->LoadExtension( extension_system->LoadExtension(
extension_path, extension_path, load_flags,
base::BindOnce( base::BindOnce(
[](gin_helper::Promise<const extensions::Extension*> promise, [](gin_helper::Promise<const extensions::Extension*> promise,
const extensions::Extension* extension, const extensions::Extension* extension,

View file

@ -136,7 +136,8 @@ class Session : public gin::Wrappable<Session>,
#endif #endif
#if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS) #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
v8::Local<v8::Promise> LoadExtension(const base::FilePath& extension_path); v8::Local<v8::Promise> LoadExtension(const base::FilePath& extension_path,
gin::Arguments* args);
void RemoveExtension(const std::string& extension_id); void RemoveExtension(const std::string& extension_id);
v8::Local<v8::Value> GetExtension(const std::string& extension_id); v8::Local<v8::Value> GetExtension(const std::string& extension_id);
v8::Local<v8::Value> GetAllExtensions(); v8::Local<v8::Value> GetAllExtensions();

View file

@ -30,7 +30,8 @@ using LoadErrorBehavior = ExtensionRegistrar::LoadErrorBehavior;
namespace { namespace {
std::pair<scoped_refptr<const Extension>, std::string> LoadUnpacked( std::pair<scoped_refptr<const Extension>, std::string> LoadUnpacked(
const base::FilePath& extension_dir) { const base::FilePath& extension_dir,
int load_flags) {
// app_shell only supports unpacked extensions. // app_shell only supports unpacked extensions.
// NOTE: If you add packed extension support consider removing the flag // NOTE: If you add packed extension support consider removing the flag
// FOLLOW_SYMLINKS_ANYWHERE below. Packed extensions should not have symlinks. // FOLLOW_SYMLINKS_ANYWHERE below. Packed extensions should not have symlinks.
@ -40,7 +41,6 @@ std::pair<scoped_refptr<const Extension>, std::string> LoadUnpacked(
return std::make_pair(nullptr, err); return std::make_pair(nullptr, err);
} }
int load_flags = Extension::FOLLOW_SYMLINKS_ANYWHERE;
std::string load_error; std::string load_error;
scoped_refptr<Extension> extension = file_util::LoadExtension( scoped_refptr<Extension> extension = file_util::LoadExtension(
extension_dir, Manifest::COMMAND_LINE, load_flags, &load_error); extension_dir, Manifest::COMMAND_LINE, load_flags, &load_error);
@ -75,10 +75,11 @@ ElectronExtensionLoader::~ElectronExtensionLoader() = default;
void ElectronExtensionLoader::LoadExtension( void ElectronExtensionLoader::LoadExtension(
const base::FilePath& extension_dir, const base::FilePath& extension_dir,
int load_flags,
base::OnceCallback<void(const Extension*, const std::string&)> cb) { base::OnceCallback<void(const Extension*, const std::string&)> cb) {
base::PostTaskAndReplyWithResult( base::PostTaskAndReplyWithResult(
GetExtensionFileTaskRunner().get(), FROM_HERE, GetExtensionFileTaskRunner().get(), FROM_HERE,
base::BindOnce(&LoadUnpacked, extension_dir), base::BindOnce(&LoadUnpacked, extension_dir, load_flags),
base::BindOnce(&ElectronExtensionLoader::FinishExtensionLoad, base::BindOnce(&ElectronExtensionLoader::FinishExtensionLoad,
weak_factory_.GetWeakPtr(), std::move(cb))); weak_factory_.GetWeakPtr(), std::move(cb)));
} }
@ -174,9 +175,13 @@ void ElectronExtensionLoader::LoadExtensionForReload(
LoadErrorBehavior load_error_behavior) { LoadErrorBehavior load_error_behavior) {
CHECK(!path.empty()); CHECK(!path.empty());
// TODO(nornagon): we should save whether file access was granted
// when loading this extension and retain it here. As is, reloading an
// extension will cause the file access permission to be dropped.
int load_flags = Extension::FOLLOW_SYMLINKS_ANYWHERE;
base::PostTaskAndReplyWithResult( base::PostTaskAndReplyWithResult(
GetExtensionFileTaskRunner().get(), FROM_HERE, GetExtensionFileTaskRunner().get(), FROM_HERE,
base::BindOnce(&LoadUnpacked, path), base::BindOnce(&LoadUnpacked, path, load_flags),
base::BindOnce(&ElectronExtensionLoader::FinishExtensionReload, base::BindOnce(&ElectronExtensionLoader::FinishExtensionReload,
weak_factory_.GetWeakPtr(), extension_id)); weak_factory_.GetWeakPtr(), extension_id));
did_schedule_reload_ = true; did_schedule_reload_ = true;

View file

@ -37,6 +37,7 @@ class ElectronExtensionLoader : public ExtensionRegistrar::Delegate {
// Loads an unpacked extension from a directory synchronously. Returns the // Loads an unpacked extension from a directory synchronously. Returns the
// extension on success, or nullptr otherwise. // extension on success, or nullptr otherwise.
void LoadExtension(const base::FilePath& extension_dir, void LoadExtension(const base::FilePath& extension_dir,
int load_flags,
base::OnceCallback<void(const Extension* extension, base::OnceCallback<void(const Extension* extension,
const std::string&)> cb); const std::string&)> cb);

View file

@ -55,8 +55,9 @@ ElectronExtensionSystem::~ElectronExtensionSystem() = default;
void ElectronExtensionSystem::LoadExtension( void ElectronExtensionSystem::LoadExtension(
const base::FilePath& extension_dir, const base::FilePath& extension_dir,
int load_flags,
base::OnceCallback<void(const Extension*, const std::string&)> cb) { base::OnceCallback<void(const Extension*, const std::string&)> cb) {
extension_loader_->LoadExtension(extension_dir, std::move(cb)); extension_loader_->LoadExtension(extension_dir, load_flags, std::move(cb));
} }
void ElectronExtensionSystem::FinishInitialization() { void ElectronExtensionSystem::FinishInitialization() {

View file

@ -41,6 +41,7 @@ class ElectronExtensionSystem : public ExtensionSystem {
// success, or nullptr otherwise. // success, or nullptr otherwise.
void LoadExtension( void LoadExtension(
const base::FilePath& extension_dir, const base::FilePath& extension_dir,
int load_flags,
base::OnceCallback<void(const Extension*, const std::string&)> cb); base::OnceCallback<void(const Extension*, const std::string&)> cb);
// Finish initialization for the shell extension system. // Finish initialization for the shell extension system.