2011-09-28 13:16:26 +00:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2011-06-20 19:16:40 +00:00
|
|
|
|
|
|
|
const Lang = imports.lang;
|
|
|
|
const DBus = imports.dbus;
|
|
|
|
const Mainloop = imports.mainloop;
|
|
|
|
const Gio = imports.gi.Gio;
|
|
|
|
const Params = imports.misc.params;
|
|
|
|
|
|
|
|
const Main = imports.ui.main;
|
2011-06-22 20:43:16 +00:00
|
|
|
const ShellMountOperation = imports.ui.shellMountOperation;
|
2011-06-20 19:16:40 +00:00
|
|
|
const ScreenSaver = imports.misc.screenSaver;
|
|
|
|
|
|
|
|
// GSettings keys
|
|
|
|
const SETTINGS_SCHEMA = 'org.gnome.desktop.media-handling';
|
|
|
|
const SETTING_ENABLE_AUTOMOUNT = 'automount';
|
|
|
|
|
2011-06-22 13:45:03 +00:00
|
|
|
const AUTORUN_EXPIRE_TIMEOUT_SECS = 10;
|
|
|
|
|
2011-06-20 19:16:40 +00:00
|
|
|
const ConsoleKitSessionIface = {
|
|
|
|
name: 'org.freedesktop.ConsoleKit.Session',
|
|
|
|
methods: [{ name: 'IsActive',
|
|
|
|
inSignature: '',
|
|
|
|
outSignature: 'b' }],
|
|
|
|
signals: [{ name: 'ActiveChanged',
|
|
|
|
inSignature: 'b' }]
|
|
|
|
};
|
|
|
|
|
|
|
|
const ConsoleKitSessionProxy = DBus.makeProxyClass(ConsoleKitSessionIface);
|
|
|
|
|
|
|
|
const ConsoleKitManagerIface = {
|
|
|
|
name: 'org.freedesktop.ConsoleKit.Manager',
|
|
|
|
methods: [{ name: 'GetCurrentSession',
|
|
|
|
inSignature: '',
|
|
|
|
outSignature: 'o' }]
|
|
|
|
};
|
|
|
|
|
|
|
|
function ConsoleKitManager() {
|
|
|
|
this._init();
|
|
|
|
};
|
|
|
|
|
|
|
|
ConsoleKitManager.prototype = {
|
|
|
|
_init: function() {
|
|
|
|
this.sessionActive = true;
|
|
|
|
|
|
|
|
DBus.system.proxifyObject(this,
|
|
|
|
'org.freedesktop.ConsoleKit',
|
|
|
|
'/org/freedesktop/ConsoleKit/Manager');
|
|
|
|
|
|
|
|
DBus.system.watch_name('org.freedesktop.ConsoleKit',
|
|
|
|
false, // do not launch a name-owner if none exists
|
|
|
|
Lang.bind(this, this._onManagerAppeared),
|
|
|
|
Lang.bind(this, this._onManagerVanished));
|
|
|
|
},
|
|
|
|
|
|
|
|
_onManagerAppeared: function(owner) {
|
|
|
|
this.GetCurrentSessionRemote(Lang.bind(this, this._onCurrentSession));
|
|
|
|
},
|
|
|
|
|
|
|
|
_onManagerVanished: function(oldOwner) {
|
|
|
|
this.sessionActive = true;
|
|
|
|
},
|
|
|
|
|
|
|
|
_onCurrentSession: function(session) {
|
|
|
|
this._ckSession = new ConsoleKitSessionProxy(DBus.system, 'org.freedesktop.ConsoleKit', session);
|
|
|
|
|
|
|
|
this._ckSession.connect
|
|
|
|
('ActiveChanged', Lang.bind(this, function(object, isActive) {
|
|
|
|
this.sessionActive = isActive;
|
|
|
|
}));
|
|
|
|
this._ckSession.IsActiveRemote(Lang.bind(this, function(isActive) {
|
|
|
|
this.sessionActive = isActive;
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
DBus.proxifyPrototype(ConsoleKitManager.prototype, ConsoleKitManagerIface);
|
|
|
|
|
|
|
|
function AutomountManager() {
|
|
|
|
this._init();
|
|
|
|
}
|
|
|
|
|
|
|
|
AutomountManager.prototype = {
|
|
|
|
_init: function() {
|
|
|
|
this._settings = new Gio.Settings({ schema: SETTINGS_SCHEMA });
|
|
|
|
this._volumeQueue = [];
|
|
|
|
|
|
|
|
this.ckListener = new ConsoleKitManager();
|
|
|
|
|
|
|
|
this._ssProxy = new ScreenSaver.ScreenSaverProxy();
|
2011-07-13 19:07:39 +00:00
|
|
|
this._ssProxy.connect('ActiveChanged',
|
2011-06-20 19:16:40 +00:00
|
|
|
Lang.bind(this,
|
|
|
|
this._screenSaverActiveChanged));
|
|
|
|
|
|
|
|
this._volumeMonitor = Gio.VolumeMonitor.get();
|
|
|
|
|
|
|
|
this._volumeMonitor.connect('volume-added',
|
|
|
|
Lang.bind(this,
|
|
|
|
this._onVolumeAdded));
|
|
|
|
this._volumeMonitor.connect('volume-removed',
|
|
|
|
Lang.bind(this,
|
|
|
|
this._onVolumeRemoved));
|
2011-06-27 15:51:23 +00:00
|
|
|
this._volumeMonitor.connect('drive-connected',
|
|
|
|
Lang.bind(this,
|
|
|
|
this._onDriveConnected));
|
|
|
|
this._volumeMonitor.connect('drive-disconnected',
|
|
|
|
Lang.bind(this,
|
|
|
|
this._onDriveDisconnected));
|
2011-06-27 16:13:54 +00:00
|
|
|
this._volumeMonitor.connect('drive-eject-button',
|
|
|
|
Lang.bind(this,
|
|
|
|
this._onDriveEjectButton));
|
2011-06-20 19:16:40 +00:00
|
|
|
|
|
|
|
Mainloop.idle_add(Lang.bind(this, this._startupMountAll));
|
|
|
|
},
|
|
|
|
|
|
|
|
_screenSaverActiveChanged: function(object, isActive) {
|
|
|
|
if (!isActive) {
|
|
|
|
this._volumeQueue.forEach(Lang.bind(this, function(volume) {
|
|
|
|
this._checkAndMountVolume(volume);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear the queue anyway
|
|
|
|
this._volumeQueue = [];
|
|
|
|
},
|
|
|
|
|
|
|
|
_startupMountAll: function() {
|
|
|
|
let volumes = this._volumeMonitor.get_volumes();
|
|
|
|
volumes.forEach(Lang.bind(this, function(volume) {
|
|
|
|
this._checkAndMountVolume(volume, { checkSession: false,
|
|
|
|
useMountOp: false });
|
|
|
|
}));
|
|
|
|
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
|
2011-06-27 15:51:23 +00:00
|
|
|
_onDriveConnected: function() {
|
|
|
|
// if we're not in the current ConsoleKit session,
|
|
|
|
// or screensaver is active, don't play sounds
|
|
|
|
if (!this.ckListener.sessionActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (this._ssProxy.screenSaverActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
global.play_theme_sound(0, 'device-added-media');
|
|
|
|
},
|
|
|
|
|
|
|
|
_onDriveDisconnected: function() {
|
|
|
|
// if we're not in the current ConsoleKit session,
|
|
|
|
// or screensaver is active, don't play sounds
|
|
|
|
if (!this.ckListener.sessionActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (this._ssProxy.screenSaverActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
global.play_theme_sound(0, 'device-removed-media');
|
|
|
|
},
|
|
|
|
|
2011-06-27 16:13:54 +00:00
|
|
|
_onDriveEjectButton: function(monitor, drive) {
|
|
|
|
// TODO: this code path is not tested, as the GVfs volume monitor
|
|
|
|
// doesn't emit this signal just yet.
|
|
|
|
if (!this.ckListener.sessionActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// we force stop/eject in this case, so we don't have to pass a
|
|
|
|
// mount operation object
|
|
|
|
if (drive.can_stop()) {
|
|
|
|
drive.stop
|
|
|
|
(Gio.MountUnmountFlags.FORCE, null, null,
|
|
|
|
Lang.bind(this, function(drive, res) {
|
|
|
|
try {
|
|
|
|
drive.stop_finish(res);
|
|
|
|
} catch (e) {
|
|
|
|
log("Unable to stop the drive after drive-eject-button " + e.toString());
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
} else if (drive.can_eject()) {
|
|
|
|
drive.eject_with_operation
|
|
|
|
(Gio.MountUnmountFlags.FORCE, null, null,
|
|
|
|
Lang.bind(this, function(drive, res) {
|
|
|
|
try {
|
|
|
|
drive.eject_with_operation_finish(res);
|
|
|
|
} catch (e) {
|
|
|
|
log("Unable to eject the drive after drive-eject-button " + e.toString());
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-06-20 19:16:40 +00:00
|
|
|
_onVolumeAdded: function(monitor, volume) {
|
|
|
|
this._checkAndMountVolume(volume);
|
|
|
|
},
|
|
|
|
|
|
|
|
_checkAndMountVolume: function(volume, params) {
|
|
|
|
params = Params.parse(params, { checkSession: true,
|
|
|
|
useMountOp: true });
|
|
|
|
|
|
|
|
if (params.checkSession) {
|
|
|
|
// if we're not in the current ConsoleKit session,
|
|
|
|
// don't attempt automount
|
|
|
|
if (!this.ckListener.sessionActive)
|
|
|
|
return;
|
|
|
|
|
2011-07-13 19:07:39 +00:00
|
|
|
if (this._ssProxy.screenSaverActive) {
|
2011-06-20 19:16:40 +00:00
|
|
|
if (this._volumeQueue.indexOf(volume) == -1)
|
|
|
|
this._volumeQueue.push(volume);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-28 20:29:45 +00:00
|
|
|
// Volume is already mounted, don't bother.
|
|
|
|
if (volume.get_mount())
|
|
|
|
return;
|
|
|
|
|
2011-06-22 13:45:03 +00:00
|
|
|
if (!this._settings.get_boolean(SETTING_ENABLE_AUTOMOUNT) ||
|
|
|
|
!volume.should_automount() ||
|
|
|
|
!volume.can_mount()) {
|
2011-09-28 20:29:45 +00:00
|
|
|
// allow the autorun to run anyway; this can happen if the
|
2011-06-22 13:45:03 +00:00
|
|
|
// mount gets added programmatically later, even if
|
|
|
|
// should_automount() or can_mount() are false, like for
|
|
|
|
// blank optical media.
|
|
|
|
this._allowAutorun(volume);
|
|
|
|
this._allowAutorunExpire(volume);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-06-22 20:43:16 +00:00
|
|
|
if (params.useMountOp) {
|
|
|
|
let operation = new ShellMountOperation.ShellMountOperation(volume);
|
|
|
|
this._mountVolume(volume, operation.mountOp);
|
|
|
|
} else {
|
|
|
|
this._mountVolume(volume, null);
|
|
|
|
}
|
2011-06-20 19:16:40 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_mountVolume: function(volume, operation) {
|
2011-06-22 13:45:03 +00:00
|
|
|
this._allowAutorun(volume);
|
2011-06-20 19:16:40 +00:00
|
|
|
volume.mount(0, operation, null,
|
|
|
|
Lang.bind(this, this._onVolumeMounted));
|
|
|
|
},
|
|
|
|
|
|
|
|
_onVolumeMounted: function(volume, res) {
|
2011-06-22 13:45:03 +00:00
|
|
|
this._allowAutorunExpire(volume);
|
|
|
|
|
2011-06-20 19:16:40 +00:00
|
|
|
try {
|
|
|
|
volume.mount_finish(res);
|
|
|
|
} catch (e) {
|
2011-07-12 15:29:49 +00:00
|
|
|
let string = e.toString();
|
|
|
|
|
|
|
|
// FIXME: needs proper error code handling instead of this
|
|
|
|
// See https://bugzilla.gnome.org/show_bug.cgi?id=591480
|
|
|
|
if (string.indexOf('No key available with this passphrase') != -1)
|
|
|
|
this._reaskPassword(volume);
|
|
|
|
else
|
|
|
|
log('Unable to mount volume ' + volume.get_name() + ': ' + string);
|
2011-06-20 19:16:40 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onVolumeRemoved: function(monitor, volume) {
|
|
|
|
this._volumeQueue =
|
|
|
|
this._volumeQueue.filter(function(element) {
|
|
|
|
return (element != volume);
|
|
|
|
});
|
2011-06-22 13:45:03 +00:00
|
|
|
},
|
|
|
|
|
2011-07-12 15:29:49 +00:00
|
|
|
_reaskPassword: function(volume) {
|
|
|
|
let operation = new ShellMountOperation.ShellMountOperation(volume, { reaskPassword: true });
|
|
|
|
this._mountVolume(volume, operation.mountOp);
|
|
|
|
},
|
|
|
|
|
2011-06-22 13:45:03 +00:00
|
|
|
_allowAutorun: function(volume) {
|
|
|
|
volume.allowAutorun = true;
|
|
|
|
},
|
|
|
|
|
|
|
|
_allowAutorunExpire: function(volume) {
|
|
|
|
Mainloop.timeout_add_seconds(AUTORUN_EXPIRE_TIMEOUT_SECS, function() {
|
|
|
|
volume.allowAutorun = false;
|
|
|
|
return false;
|
|
|
|
});
|
2011-06-20 19:16:40 +00:00
|
|
|
}
|
|
|
|
}
|