a3db909383
The LoginManager abstraction is still mostly callback-based, not least because the methods are thin wrappers around logind D-Bus calls. However as gjs' dbus wrapper now generates promised-based wrappers as well, we can implement a proper async API just as naturally. Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/2344>
236 lines
8.0 KiB
JavaScript
236 lines
8.0 KiB
JavaScript
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
|
/* exported canLock, getLoginManager, registerSessionWithGDM */
|
|
|
|
const { GLib, Gio } = imports.gi;
|
|
const Signals = imports.misc.signals;
|
|
|
|
const { loadInterfaceXML } = imports.misc.fileUtils;
|
|
|
|
const SystemdLoginManagerIface = loadInterfaceXML('org.freedesktop.login1.Manager');
|
|
const SystemdLoginSessionIface = loadInterfaceXML('org.freedesktop.login1.Session');
|
|
const SystemdLoginUserIface = loadInterfaceXML('org.freedesktop.login1.User');
|
|
|
|
const SystemdLoginManager = Gio.DBusProxy.makeProxyWrapper(SystemdLoginManagerIface);
|
|
const SystemdLoginSession = Gio.DBusProxy.makeProxyWrapper(SystemdLoginSessionIface);
|
|
const SystemdLoginUser = Gio.DBusProxy.makeProxyWrapper(SystemdLoginUserIface);
|
|
|
|
function haveSystemd() {
|
|
return GLib.access("/run/systemd/seats", 0) >= 0;
|
|
}
|
|
|
|
function versionCompare(required, reference) {
|
|
required = required.split('.');
|
|
reference = reference.split('.');
|
|
|
|
for (let i = 0; i < required.length; i++) {
|
|
let requiredInt = parseInt(required[i]);
|
|
let referenceInt = parseInt(reference[i]);
|
|
if (requiredInt != referenceInt)
|
|
return requiredInt < referenceInt;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function canLock() {
|
|
try {
|
|
let params = GLib.Variant.new('(ss)', ['org.gnome.DisplayManager.Manager', 'Version']);
|
|
let result = Gio.DBus.system.call_sync('org.gnome.DisplayManager',
|
|
'/org/gnome/DisplayManager/Manager',
|
|
'org.freedesktop.DBus.Properties',
|
|
'Get', params, null,
|
|
Gio.DBusCallFlags.NONE,
|
|
-1, null);
|
|
|
|
let version = result.deep_unpack()[0].deep_unpack();
|
|
return haveSystemd() && versionCompare('3.5.91', version);
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
async function registerSessionWithGDM() {
|
|
log("Registering session with GDM");
|
|
try {
|
|
await Gio.DBus.system.call(
|
|
'org.gnome.DisplayManager',
|
|
'/org/gnome/DisplayManager/Manager',
|
|
'org.gnome.DisplayManager.Manager',
|
|
'RegisterSession',
|
|
GLib.Variant.new('(a{sv})', [{}]), null,
|
|
Gio.DBusCallFlags.NONE, -1, null);
|
|
} catch (e) {
|
|
if (!e.matches(Gio.DBusError, Gio.DBusError.UNKNOWN_METHOD))
|
|
log(`Error registering session with GDM: ${e.message}`);
|
|
else
|
|
log('Not calling RegisterSession(): method not exported, GDM too old?');
|
|
}
|
|
}
|
|
|
|
let _loginManager = null;
|
|
|
|
/**
|
|
* getLoginManager:
|
|
* An abstraction over systemd/logind and ConsoleKit.
|
|
* @returns {object} - the LoginManager singleton
|
|
*
|
|
*/
|
|
function getLoginManager() {
|
|
if (_loginManager == null) {
|
|
if (haveSystemd())
|
|
_loginManager = new LoginManagerSystemd();
|
|
else
|
|
_loginManager = new LoginManagerDummy();
|
|
}
|
|
|
|
return _loginManager;
|
|
}
|
|
|
|
var LoginManagerSystemd = class extends Signals.EventEmitter {
|
|
constructor() {
|
|
super();
|
|
|
|
this._proxy = new SystemdLoginManager(Gio.DBus.system,
|
|
'org.freedesktop.login1',
|
|
'/org/freedesktop/login1');
|
|
this._userProxy = new SystemdLoginUser(Gio.DBus.system,
|
|
'org.freedesktop.login1',
|
|
'/org/freedesktop/login1/user/self');
|
|
this._proxy.connectSignal('PrepareForSleep',
|
|
this._prepareForSleep.bind(this));
|
|
}
|
|
|
|
async getCurrentSessionProxy() {
|
|
if (this._currentSession)
|
|
return this._currentSession;
|
|
|
|
let sessionId = GLib.getenv('XDG_SESSION_ID');
|
|
if (!sessionId) {
|
|
log('Unset XDG_SESSION_ID, getCurrentSessionProxy() called outside a user session. Asking logind directly.');
|
|
let [session, objectPath] = this._userProxy.Display;
|
|
if (session) {
|
|
log(`Will monitor session ${session}`);
|
|
sessionId = session;
|
|
} else {
|
|
log('Failed to find "Display" session; are we the greeter?');
|
|
|
|
for ([session, objectPath] of this._userProxy.Sessions) {
|
|
let sessionProxy = new SystemdLoginSession(Gio.DBus.system,
|
|
'org.freedesktop.login1',
|
|
objectPath);
|
|
log(`Considering ${session}, class=${sessionProxy.Class}`);
|
|
if (sessionProxy.Class == 'greeter') {
|
|
log(`Yes, will monitor session ${session}`);
|
|
sessionId = session;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!sessionId) {
|
|
log('No, failed to get session from logind.');
|
|
return null;
|
|
}
|
|
}
|
|
}
|
|
|
|
try {
|
|
const [objectPath] = await this._proxy.GetSessionAsync(sessionId);
|
|
this._currentSession = new SystemdLoginSession(Gio.DBus.system,
|
|
'org.freedesktop.login1', objectPath);
|
|
return this._currentSession;
|
|
} catch (error) {
|
|
logError(error, 'Could not get a proxy for the current session');
|
|
return null;
|
|
}
|
|
}
|
|
|
|
async canSuspend() {
|
|
try {
|
|
const [result] = await this._proxy.CanSuspendAsync();
|
|
const needsAuth = result === 'challenge';
|
|
const canSuspend = needsAuth || result === 'yes';
|
|
return [canSuspend, needsAuth];
|
|
} catch (error) {
|
|
return [false, false];
|
|
}
|
|
}
|
|
|
|
async canRebootToBootLoaderMenu() {
|
|
try {
|
|
const [result] = await this._proxy.CanRebootToBootLoaderMenuAsync();
|
|
const needsAuth = result[0] === 'challenge';
|
|
const canRebootToBootLoaderMenu = needsAuth || result[0] === 'yes';
|
|
return [canRebootToBootLoaderMenu, needsAuth];
|
|
} catch (error) {
|
|
return [false, false];
|
|
}
|
|
}
|
|
|
|
setRebootToBootLoaderMenu() {
|
|
/* Parameter is timeout in usec, show to menu for 60 seconds */
|
|
this._proxy.SetRebootToBootLoaderMenuAsync(60000000);
|
|
}
|
|
|
|
async listSessions() {
|
|
try {
|
|
const [sessions] = await this._proxy.ListSessionsAsync();
|
|
return sessions;
|
|
} catch (e) {
|
|
return [];
|
|
}
|
|
}
|
|
|
|
suspend() {
|
|
this._proxy.SuspendAsync(true);
|
|
}
|
|
|
|
async inhibit(reason, cancellable) {
|
|
const inVariant = new GLib.Variant('(ssss)',
|
|
['sleep', 'GNOME Shell', reason, 'delay']);
|
|
const [outVariant_, fdList] =
|
|
await this._proxy.call_with_unix_fd_list('Inhibit',
|
|
inVariant, 0, -1, null, cancellable);
|
|
const [fd] = fdList.steal_fds();
|
|
return new Gio.UnixInputStream({ fd });
|
|
}
|
|
|
|
_prepareForSleep(proxy, sender, [aboutToSuspend]) {
|
|
this.emit('prepare-for-sleep', aboutToSuspend);
|
|
}
|
|
};
|
|
|
|
var LoginManagerDummy = class extends Signals.EventEmitter {
|
|
getCurrentSessionProxy() {
|
|
// we could return a DummySession object that fakes whatever callers
|
|
// expect (at the time of writing: connect() and connectSignal()
|
|
// methods), but just never settling the promise should be safer
|
|
return new Promise(() => {});
|
|
}
|
|
|
|
canSuspend() {
|
|
return new Promise(resolve => resolve([false, false]));
|
|
}
|
|
|
|
canRebootToBootLoaderMenu() {
|
|
return new Promise(resolve => resolve([false, false]));
|
|
}
|
|
|
|
setRebootToBootLoaderMenu() {
|
|
}
|
|
|
|
listSessions() {
|
|
return new Promise(resolve => resolve([]));
|
|
}
|
|
|
|
suspend() {
|
|
this.emit('prepare-for-sleep', true);
|
|
this.emit('prepare-for-sleep', false);
|
|
}
|
|
|
|
/* eslint-disable-next-line require-await */
|
|
async inhibit() {
|
|
return null;
|
|
}
|
|
};
|