feat: new makeSingleInstance API (#12782)

* Refactor app.makeSingleInstance
* new API `app.isPrimaryInstance()`
* new API `app.isSingleInstance()`
* new event `app.on('second-instance')`
* deprecated old syntax `app.makeSingleInstance(cb)`
* deprecated old syntax of `app.makeSingleInstance() --> bool` in favor
of `app.isPrimaryInstance()`
* Fix spec, we don't need process.nextTick hacks any more
* Make deprecation TODO for the return value of makeSingleInstance
* Refactor makeSingleInstance to requestSingleInstanceLock and add appropriate deprecation comments
* I swear this isn't tricking the linter
* Make const
* Add deprecation warnings for release, and add to planned-breaking-changes

BREAKING CHANGE
This commit is contained in:
Samuel Attard 2018-05-08 01:29:18 +10:00 committed by GitHub
parent 9c8952aef0
commit 5b5c161601
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
9 changed files with 176 additions and 52 deletions

View file

@ -350,8 +350,8 @@ struct Converter<content::CertificateRequestResultType> {
namespace atom { namespace atom {
ProcessMetric::ProcessMetric(int type, ProcessMetric::ProcessMetric(int type,
base::ProcessId pid, base::ProcessId pid,
std::unique_ptr<base::ProcessMetrics> metrics) { std::unique_ptr<base::ProcessMetrics> metrics) {
this->type = type; this->type = type;
this->pid = pid; this->pid = pid;
this->metrics = std::move(metrics); this->metrics = std::move(metrics);
@ -422,7 +422,9 @@ int GetPathConstant(const std::string& name) {
} }
bool NotificationCallbackWrapper( bool NotificationCallbackWrapper(
const ProcessSingleton::NotificationCallback& callback, const base::Callback<
void(const base::CommandLine::StringVector& command_line,
const base::FilePath& current_directory)>& callback,
const base::CommandLine::StringVector& cmd, const base::CommandLine::StringVector& cmd,
const base::FilePath& cwd) { const base::FilePath& cwd) {
// Make sure the callback is called after app gets ready. // Make sure the callback is called after app gets ready.
@ -864,30 +866,43 @@ std::string App::GetLocale() {
return g_browser_process->GetApplicationLocale(); return g_browser_process->GetApplicationLocale();
} }
bool App::MakeSingleInstance( void App::OnSecondInstance(const base::CommandLine::StringVector& cmd,
const ProcessSingleton::NotificationCallback& callback) { const base::FilePath& cwd) {
Emit("second-instance", cmd, cwd);
}
bool App::HasSingleInstanceLock() const {
if (process_singleton_) if (process_singleton_)
return false; return true;
return false;
}
bool App::RequestSingleInstanceLock() {
if (HasSingleInstanceLock())
return true;
base::FilePath user_dir; base::FilePath user_dir;
PathService::Get(brightray::DIR_USER_DATA, &user_dir); PathService::Get(brightray::DIR_USER_DATA, &user_dir);
auto cb = base::Bind(&App::OnSecondInstance, base::Unretained(this));
process_singleton_.reset(new ProcessSingleton( process_singleton_.reset(new ProcessSingleton(
user_dir, base::Bind(NotificationCallbackWrapper, callback))); user_dir, base::Bind(NotificationCallbackWrapper, cb)));
switch (process_singleton_->NotifyOtherProcessOrCreate()) { switch (process_singleton_->NotifyOtherProcessOrCreate()) {
case ProcessSingleton::NotifyResult::LOCK_ERROR: case ProcessSingleton::NotifyResult::LOCK_ERROR:
case ProcessSingleton::NotifyResult::PROFILE_IN_USE: case ProcessSingleton::NotifyResult::PROFILE_IN_USE:
case ProcessSingleton::NotifyResult::PROCESS_NOTIFIED: { case ProcessSingleton::NotifyResult::PROCESS_NOTIFIED: {
process_singleton_.reset(); process_singleton_.reset();
return true; return false;
} }
case ProcessSingleton::NotifyResult::PROCESS_NONE: case ProcessSingleton::NotifyResult::PROCESS_NONE:
default: // Shouldn't be needed, but VS warns if it is not there. default: // Shouldn't be needed, but VS warns if it is not there.
return false; return true;
} }
} }
void App::ReleaseSingleInstance() { void App::ReleaseSingleInstanceLock() {
if (process_singleton_) { if (process_singleton_) {
process_singleton_->Cleanup(); process_singleton_->Cleanup();
process_singleton_.reset(); process_singleton_.reset();
@ -1252,8 +1267,9 @@ void App::BuildPrototype(v8::Isolate* isolate,
#if defined(USE_NSS_CERTS) #if defined(USE_NSS_CERTS)
.SetMethod("importCertificate", &App::ImportCertificate) .SetMethod("importCertificate", &App::ImportCertificate)
#endif #endif
.SetMethod("makeSingleInstance", &App::MakeSingleInstance) .SetMethod("hasSingleInstanceLock", &App::HasSingleInstanceLock)
.SetMethod("releaseSingleInstance", &App::ReleaseSingleInstance) .SetMethod("requestSingleInstanceLock", &App::RequestSingleInstanceLock)
.SetMethod("releaseSingleInstanceLock", &App::ReleaseSingleInstanceLock)
.SetMethod("relaunch", &App::Relaunch) .SetMethod("relaunch", &App::Relaunch)
.SetMethod("isAccessibilitySupportEnabled", .SetMethod("isAccessibilitySupportEnabled",
&App::IsAccessibilitySupportEnabled) &App::IsAccessibilitySupportEnabled)

View file

@ -178,9 +178,11 @@ class App : public AtomBrowserClient::Delegate,
void SetDesktopName(const std::string& desktop_name); void SetDesktopName(const std::string& desktop_name);
std::string GetLocale(); std::string GetLocale();
bool MakeSingleInstance( void OnSecondInstance(const base::CommandLine::StringVector& cmd,
const ProcessSingleton::NotificationCallback& callback); const base::FilePath& cwd);
void ReleaseSingleInstance(); bool HasSingleInstanceLock() const;
bool RequestSingleInstanceLock();
void ReleaseSingleInstanceLock();
bool Relaunch(mate::Arguments* args); bool Relaunch(mate::Arguments* args);
void DisableHardwareAcceleration(mate::Arguments* args); void DisableHardwareAcceleration(mate::Arguments* args);
void DisableDomainBlockingFor3DAPIs(mate::Arguments* args); void DisableDomainBlockingFor3DAPIs(mate::Arguments* args);

View file

@ -373,6 +373,22 @@ app.on('session-created', (event, session) => {
}) })
``` ```
### Event: 'second-instance'
Returns:
* `argv` String[] - An array of the second instance's command line arguments
* `workingDirectory` String - The second instance's working directory
This event will be emitted inside the primary instance of your application
when a second instance has been executed. `argv` is an Array of the second instance's
command line arguments, and `workingDirectory` is its current working directory. Usually
applications respond to this by making their primary window focused and
non-minimized.
This event is guaranteed to be emitted after the `ready` event of `app`
gets emitted.
## Methods ## Methods
The `app` object has the following methods: The `app` object has the following methods:
@ -742,32 +758,24 @@ app.setJumpList([
]) ])
``` ```
### `app.makeSingleInstance(callback)` ### `app.requestSingleInstanceLock()`
* `callback` Function Returns `Boolean`
* `argv` String[] - An array of the second instance's command line arguments
* `workingDirectory` String - The second instance's working directory
Returns `Boolean`.
This method makes your application a Single Instance Application - instead of This method makes your application a Single Instance Application - instead of
allowing multiple instances of your app to run, this will ensure that only a allowing multiple instances of your app to run, this will ensure that only a
single instance of your app is running, and other instances signal this single instance of your app is running, and other instances signal this
instance and exit. instance and exit.
`callback` will be called by the first instance with `callback(argv, workingDirectory)` The return value of this method indicates whether or not this instance of your
when a second instance has been executed. `argv` is an Array of the second instance's application successfully obtained the lock. If it failed to obtain the lock
command line arguments, and `workingDirectory` is its current working directory. Usually you can assume that another instance of your application is already running with
applications respond to this by making their primary window focused and the lock and exit immediately.
non-minimized.
The `callback` is guaranteed to be executed after the `ready` event of `app` I.e. This method returns `true` if your process is the primary instance of your
gets emitted. application and your app should continue loading. It returns `false` if your
process should immediately quit as it has sent its parameters to another
This method returns `false` if your process is the primary instance of the instance that has already acquired the lock.
application and your app should continue loading. And returns `true` if your
process has sent its parameters to another instance, and you should immediately
quit.
On macOS the system enforces single instance automatically when users try to open On macOS the system enforces single instance automatically when users try to open
a second instance of your app in Finder, and the `open-file` and `open-url` a second instance of your app in Finder, and the `open-file` and `open-url`
@ -782,27 +790,38 @@ starts:
const {app} = require('electron') const {app} = require('electron')
let myWindow = null let myWindow = null
const isSecondInstance = app.makeSingleInstance((commandLine, workingDirectory) => { const gotTheLock = app.requestSingleInstanceLock()
// Someone tried to run a second instance, we should focus our window.
if (myWindow) {
if (myWindow.isMinimized()) myWindow.restore()
myWindow.focus()
}
})
if (isSecondInstance) { if (!gotTheLock) {
app.quit() app.quit()
} } else {
app.on('second-instance', (commandLine, workingDirectory) => {
// Someone tried to run a second instance, we should focus our window.
if (myWindow) {
if (myWindow.isMinimized()) myWindow.restore()
myWindow.focus()
}
})
// Create myWindow, load the rest of the app, etc... // Create myWindow, load the rest of the app, etc...
app.on('ready', () => { app.on('ready', () => {
}) })
}
``` ```
### `app.releaseSingleInstance()` ### `app.hasSingleInstanceLock()`
Releases all locks that were created by `makeSingleInstance`. This will allow Returns `Boolean`
multiple instances of the application to once again run side by side.
This method returns whether or not this instance of your app is currently
holding the single instance lock. You can request the lock with
`app.requestSingleInstanceLock()` and release with
`app.releaseSingleInstanceLock()`
### `app.releaseSingleInstanceLock()`
Releases all locks that were created by `requestSingleInstanceLock`. This will
allow multiple instances of the application to once again run side by side.
### `app.setUserActivity(type, userInfo[, webpageURL])` _macOS_ ### `app.setUserActivity(type, userInfo[, webpageURL])` _macOS_

View file

@ -1,4 +1,4 @@
# Planned Breaking API Changes # Planned Breaking API Changes (3.0)
The following list includes the APIs that will be removed in Electron 3.0. The following list includes the APIs that will be removed in Electron 3.0.
@ -150,3 +150,33 @@ Replace with: https://atom.io/download/electron
The `FIXME` string is used in code comments to denote things that should be The `FIXME` string is used in code comments to denote things that should be
fixed for the 3.0 release. See fixed for the 3.0 release. See
https://github.com/electron/electron/search?q=fixme https://github.com/electron/electron/search?q=fixme
# Planned Breaking API Changes (4.0)
The following list includes the APIs that will be removed in Electron 4.0.
There is no timetable for when this release will occur but deprecation
warnings will be added at least [one major version](electron-versioning.md#semver) beforehand.
## `app.makeSingleInstance`
```js
// Deprecated
app.makeSingleInstance(function (argv, cwd) {
})
// Replace with
app.requestSingleInstanceLock()
app.on('second-instance', function (argv, cwd) {
})
```
## `app.releaseSingleInstance`
```js
// Deprecated
app.releaseSingleInstance()
// Replace with
app.releaseSingleInstanceLock()
```

View file

@ -109,6 +109,21 @@ for (let name of events) {
}) })
} }
// TODO(MarshallOfSound): Remove in 4.0
app.releaseSingleInstance = () => {
deprecate.warn('app.releaseSingleInstance(cb)', 'app.releaseSingleInstanceLock()')
app.releaseSingleInstanceLock()
}
// TODO(MarshallOfSound): Remove in 4.0
app.makeSingleInstance = (oldStyleFn) => {
deprecate.warn('app.makeSingleInstance(cb)', 'app.requestSingleInstanceLock() and app.on(\'second-instance\', cb)')
if (oldStyleFn && typeof oldStyleFn === 'function') {
app.on('second-instance', (event, ...args) => oldStyleFn(...args))
}
return !app.requestSingleInstanceLock()
}
// Wrappers for native classes. // Wrappers for native classes.
const {DownloadItem} = process.atomBinding('download_item') const {DownloadItem} = process.atomBinding('download_item')
Object.setPrototypeOf(DownloadItem.prototype, EventEmitter.prototype) Object.setPrototypeOf(DownloadItem.prototype, EventEmitter.prototype)

View file

@ -185,7 +185,29 @@ describe('app module', () => {
}) })
}) })
// TODO(MarshallOfSound) - Remove in 4.0.0
describe('app.makeSingleInstance', () => { describe('app.makeSingleInstance', () => {
it('prevents the second launch of app', function (done) {
this.timeout(120000)
const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton-old')
// First launch should exit with 0.
const first = ChildProcess.spawn(remote.process.execPath, [appPath])
first.once('exit', (code) => {
assert.equal(code, 0)
})
// Start second app when received output.
first.stdout.once('data', () => {
// Second launch should exit with 1.
const second = ChildProcess.spawn(remote.process.execPath, [appPath])
second.once('exit', (code) => {
assert.equal(code, 1)
done()
})
})
})
})
describe('app.requestSingleInstanceLock', () => {
it('prevents the second launch of app', function (done) { it('prevents the second launch of app', function (done) {
this.timeout(120000) this.timeout(120000)
const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton') const appPath = path.join(__dirname, 'fixtures', 'api', 'singleton')

13
spec/fixtures/api/singleton-old/main.js vendored Normal file
View file

@ -0,0 +1,13 @@
const {app} = require('electron')
app.once('ready', () => {
console.log('started') // ping parent
})
const shouldExit = app.makeSingleInstance(() => {
setImmediate(() => app.exit(0))
})
if (shouldExit) {
app.exit(1)
}

View file

@ -0,0 +1,5 @@
{
"name": "electron-app-singleton",
"main": "main.js"
}

View file

@ -4,10 +4,12 @@ app.once('ready', () => {
console.log('started') // ping parent console.log('started') // ping parent
}) })
const shouldExit = app.makeSingleInstance(() => { const gotTheLock = app.requestSingleInstanceLock()
process.nextTick(() => app.exit(0))
app.on('second-instance', () => {
setImmediate(() => app.exit(0))
}) })
if (shouldExit) { if (!gotTheLock) {
app.exit(1) app.exit(1)
} }