Я хотел бы запустить экран, если сеанс Gnome заблокирован и разблокирован. Есть ли способ, которым я могу перехватить это и выполнить определенные действия, когда рабочий стол заблокирован или разблокирован?
Я хотел бы запустить экран, если сеанс Gnome заблокирован и разблокирован. Есть ли способ, которым я могу перехватить это и выполнить определенные действия, когда рабочий стол заблокирован или разблокирован?
Ответы:
Gnome-заставка генерирует некоторые сигналы на dbus, когда что-то происходит.
Здесь документация (с некоторыми примерами).
Вы могли бы написать сценарии, которые выполняются:
dbus-monitor --session "type='signal',interface='org.gnome.ScreenSaver'"
и это делает то, что вам нужно, в любое время dbus-monitor
печатает строку о заблокированном / разблокированном экране.
Вот команда bash, чтобы сделать то, что вам нужно:
dbus-monitor --session "type='signal',interface='org.gnome.ScreenSaver'" |
while read x; do
case "$x" in
*"boolean true"*) echo SCREEN_LOCKED;;
*"boolean false"*) echo SCREEN_UNLOCKED;;
esac
done
Просто замените echo SCREEN_LOCKED
и echo SCREEN_UNLOCKED
тем, что вам нужно.
gnome-screensaver-command
уже есть. Переходя -a
к gnome-screensaver-command
блокировке экрана, вы разблокируете его с помощью -d
. В любом случае, большинство приложений для гномов широко используют dbus, поэтому вы сможете делать с ним много удивительных вещей.
В Ubuntu 14.04 изменилось событие DBus для разблокировки экрана, и новый скрипт для привязки к событиям блокировки экрана и разблокировки выглядит следующим образом
dbus-monitor --session "type='signal',interface='com.ubuntu.Upstart0_6'" | \
(
while true; do
read X
if echo $X | grep "desktop-lock" &> /dev/null; then
SCREEN_LOCKED;
elif echo $X | grep "desktop-unlock" &> /dev/null; then
SCREEN_UNLOCKED;
fi
done
)
В настоящее время я думаю, что лучше слушать сообщения LockedHint
, чем заставки. Таким образом, вы не привязаны к реализации заставки.
Вот простой скрипт для этого:
gdbus monitor -y -d org.freedesktop.login1 | grep LockedHint
Дает это:
/org/freedesktop/login1/session/_32: org.freedesktop.DBus.Properties.PropertiesChanged ('org.freedesktop.login1.Session', {'LockedHint': <true>}, @as [])
/org/freedesktop/login1/session/_32: org.freedesktop.DBus.Properties.PropertiesChanged ('org.freedesktop.login1.Session', {'LockedHint': <false>}, @as [])
Ubuntu 16.04: решение ozma у меня не сработало, но вот это:
dbus-monitor --session "type=signal,interface=com.canonical.Unity.Session,member=Unlocked" |
while read MSG; do
LOCK_STAT=`echo $MSG | awk '{print $NF}'`
if [[ "$LOCK_STAT" == "member=Unlocked" ]]; then
echo "was unlocked"
fi
done
Расширение на уже предоставленный ответ.
Если вы попытаетесь запустить сценарий из сеанса screen
или tmux
, вам нужно $DBUS_SESSION_BUS_ADDRESS
сначала найти правильный и передать его в качестве аргумента dbus-monitor
вместо --session
. Также, если вы запускаете его как демон, вы должны убедиться, что одновременно запускается только один экземпляр (например, с файлом блокировки), и что сценарий очищается после себя trap
. Следующий пример будет работать как демон в большинстве современных сред Gnome (протестировано в Ubuntu GNOME 16.04):
#!/bin/bash
set -o nounset # good practice, exit if unset variable used
pidfile=/tmp/lastauth.pid # lock file path
logfile=/tmp/lastauth.log # log file path
cleanup() { # when cleaning up:
rm -f $pidfile # * remove the lock file
trap - INT TERM EXIT # * reset kernel signal catching
exit # * stop the daemon
}
log() { # simple logging format example
echo $(date +%Y-%m-%d\ %X) -- $USER -- "$@" >> $logfile
}
if [ -e "$pidfile" ]; then # if lock file exists, exit
log $0 already running...
exit
fi
trap cleanup INT TERM EXIT # call cleanup() if e.g. killed
log daemon started...
echo $$ > $pidfile # create lock file with own PID inside
# usually `dbus-daemon` address can be guessed (`-s` returns 1st PID found)
export $(grep -z DBUS_SESSION_BUS_ADDRESS /proc/$(pidof -s dbus-daemon)/environ)
expr='type=signal,interface=org.gnome.ScreenSaver' # DBus watch expression here
dbus-monitor --address $DBUS_SESSION_BUS_ADDRESS "$expr" | \
while read line; do
case "$line" in
*"boolean true"*) log session locked;;
*"boolean false"*) log session unlocked;;
esac
done
cleanup # let's not leave orphaned lock file when the loop ends (e.g. dbus dies)
Если это не работает для вас, это, вероятно, потому что:
Если вы используете Kubuntu или используете KDE / Plasma в качестве среды рабочего стола, вы должны прослушивать интерфейс org.freedesktop.ScreenSaver
, поэтому сценарий прослушивания этого события будет выглядеть следующим образом:
dbus-monitor --session "type='signal',interface='org.freedesktop.ScreenSaver'" |
while read x; do
case "$x" in
*"boolean true"*) echo SCREEN_LOCKED;;
*"boolean false"*) echo SCREEN_UNLOCKED;;
esac
done
Personalization>Notifications>Notifications>Screensaver
.
upstart
сессионный Иов Поддержит desktop-lock
и desktop-unlock
событие в start on
строфе. Просто создать работу .conf для вашего пользователя с соответствующими триггерами и командами вызова в соответствии $XDG_CONFIG_HOME/upstart/
или $HOME/.config/upstart
как пример ниже:
description "some job description"
start on desktop-lock
script
/path/to/your/executable
end script
это может быть не самое чистое решение, но оно работает для меня, когда dbus-monitor ...
нет:
gdbus monitor -y -d org.freedesktop.login1 |\
grep --line-buffered -i "LockedHint" |\
sed -uE 's/.*LockedHint.*<(.*)>.*/\1/g'
Должен возвращать поток строк со строками 'true' или 'false'
Использование: Fedora релиз 30 (тридцать)
это то, что у меня сработало в Ubuntu 16.04
dbus-monitor --session "type=signal,interface=org.gnome.ScreenSaver" |
while read MSG; do
LOCK_STAT=`echo $MSG | grep boolean | awk '{print $2}'`
if [[ "$LOCK_STAT" == "true" ]]; then
echo "was locked"
else
echo "was un-locked"
fi
done