Easyland is a Python framework to manage your wayland compositor (Hyprland, Sway) configuration by reacting to events. With Easyland, you can dismiss many side tools like Kanshi, hypridle, swayidle, etc. and script your environment according to your preferences.
- Hyprland IPC event list
- Sway IPC
- Systemd signals
- Native Wayland Idle system (ext_idle_notify_v1)
The tool allows to listen for these events and to execute commands in response.
Good question.
Initially, I was a bit stressed by the number of tools needed with Hyprland (Kanshi & hypridle notably), and also by the number of bugs despite the awesome efforts of the developers.
I wanted to have a deeper control on my system, and to be able to script it as I wanted.
To give an example, my laptop screen brightness was always at 100% when I undock it, and Kanshi does not allow to add shell commands. This is only one small example of the numerous limitations I met during my setup of Hyprland.
By scripting my Desktop in Python, I have more control to implement what I want.
- Install Easyland in a python environment
pip3 -i easyland
-
Copy an example of configuration files from here
-
Modify it according to your needs
-
Launch
easyland -c <path_to_your_config_file
This program needs the following external tools:
Depending if you use Hyprland or Sway, you will need hyprctl
or swaymsg
.
If it's not done automatically, before using PyWayland, you will need to execute pywayland.scanner
to generate all protocols:
python -m pywayland.scanner
or
pywayland-scanner
The easyland package provides the easyland
CLI command, which loads your custom Python file configuration with the -c parameter:
easyland -c ~/home/.config/hyprland/myconfig.py
Where myconfig.py
contains such a content, explained in details below:
from easyland import logger, command
###############################################################################
# Set active listeners
###############################################################################
listeners = {
"hyprland": { "socket_path": "/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock" },
'systemd_logind': {},
'idle': {}
}
###############################################################################
# Method executed at start
###############################################################################
def init():
set_monitors()
###############################################################################
# Idle configuration
# Format: [timeout in seconds, [commands to run], [commands to run on resume]]
###############################################################################
def idle_config():
return [
[150, ['brightnessctl -s set 0'], ['brightnessctl -r']],
[600, ['pidof hyprlock || hyprlock']],
[720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]
]
###############################################################################
# Handler of Hyprland IPC events
# List of events: https://wiki.hyprland.org/IPC/
###############################################################################
def on_hyprland_event(event, argument):
if event in [ "monitoradded", "monitorremoved" ]:
logger.info('Handling hyprland event: ' + event)
set_monitors()
###############################################################################
# Handlers of Systemd logind events
###############################################################################
def on_PrepareForSleep(payload):
if 'true' in payload:
logger.info("Locking the screen before suspend")
command.exec("pidof hyprlock || hyprlock", True)
# To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock && loginctl unlock-session
# def on_Unlock():
# logger.info("Unlocking the screen")
# To use this handler, you need to launch your locker like this: loginctl lock-session
# def on_lock():
# logger.info("Locking the screen")
###############################################################################
# Various methods
###############################################################################
def set_monitors():
logger.info('Setting monitors')
if command.hyprland_get_monitor(description="HP 22es") is not None:
command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
# command.exec('hyprctl keyword monitor "eDP-1,disable"')
else:
command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
command.exec("brightnessctl -s set 0")
You can find an example in the config_examples
folder. We'll explain step-by-step the configuration called Hyprland.py
.
from easyland import logger, command
Easyland has helper tools to log everything (console and file) and execute commands. Just import the two.
listeners = {
"hyprland": { "socket_path": "/tmp/hypr/$HYPRLAND_INSTANCE_SIGNATURE/.socket2.sock" },
'systemd_logind': {},
'idle': {}
}
The listeners to launch at the startup. There are currently three listeners:
hyprland
to listen Hyprland IPC eventssystemd_logind
to monitor for Systemd Logind eventsidle
which allows you to react when your computer has no activity
Each one can take take on or more parameters. See reference below.
def idle_config():
return [
[150, ['brightnessctl -s set 0'], ['brightnessctl -r']],
[600, ['pidof hyprlock || hyprlock']],
[720, ['hyprctl dispatch dpms off'], ['hyprctl dispatch dpms on']]
]
This method configure the Idle part. It should return the list of your idle actions. Each action has three parameters:
- The timeout in seconds
- The list of commands to execute when the timeout occurs
- The optional list of commands when the timeout is resumed (eg. once there is user activity after the timeout)
Here, at 720 seconds, the screens are turned off. Then, if some activities are detected, they are turned on.
def on_hyprland_event(event, argument):
if event in [ "monitoradded", "monitorremoved" ]:
logger.info('Handling hyprland event: ' + event)
set_monitors()
The method on_hyprland_event
allows you to handle Hyprland IPC events. All events are avalable here.
In this case, when we connect or disconnect a monitor, we call a method to set our monitors according to our preferences. This method is as follows.
def set_monitors(self):
logger.info('Setting monitors')
if command.hyprland_get_monitor(description="HP 22es") is not None:
command.exec('hyprctl keyword monitor "eDP-1,disable"')
else:
command.exec('hyprctl keyword monitor "eDP-1,preferred,auto,2"')
command.exec("brightnessctl -s set 0")
We use the hyprland_get_monitor
command helper to get the configuration of a particular monitor. hyprland_get_monitor
accepts the name of the monitor, its description, the maker or the model. If the screen is not found, this method returns None.
So, when we detect a "HP 22es" monitor, we disable the screen of the laptop. Otherwise, we turn on the monitor of the laptop, and we put the brightness at the lowest level (in my configuration, when I undock my laptop, the brightness is at 100%)
For Sway, you have the sway_get_monitor
helper method.
def on_PrepareForSleep(payload):
if 'true' in payload:
logger.info("Locking the screen before suspend")
command.exec("pidof hyprlock || hyprlock", True)
The methods on_Whatever(payload)
are automatically called when the signal Whatever is sent by Systemd Logind. Here, we are listening for the signal "PrepareForSleep" which is called just before you computer is suspending.
The second parameter of the command.exec
helper allows you to execute a command in the background. It's necessary here, otherwhise Easyland will wait indefinitely until the screen is unlocked.
You may be interested to listen for Lock and Unlock events emitted from Systemd when you call loginctl lock-session
and loginctl unlock-session
. However, keep in mind that hyprlock
and swaylock
do not send any signal for these events, so you need to hack that.
# To use this handler, you need to launch your locker (hyprlock or swaylock) like this: hyprlock && loginctl unlock-session
def on_Unlock():
logger.info("Unlocking the screen")
To receive the Systemd Unlock
signal, you should launch your screen locker with the following command: hyprlock && loginctl unlock-session
, so Systemd will send the Unlock
signal when the screen is unlocked.
For locking, keep in mind that hyprlock
and swaylock
do not listen for the Systemd Lock
event, so you need to it manually.
# To use this handler, you need to launch your locker like this: loginctl lock-session
def on_Lock():
logger.info("Locking the screen")
command.exec('pidof hyprlock || hyprlock', True)
# Do other actions if needed
Alternatively to write several methods to listen for Systemd events, you can also define method on_systemd_event
and add a condition to achieve what you want:
def on_systemd_event(sender, signal, payload)
if signal == 'Lock':
...
if signal == 'PrepareForSleep':
...
socket_path
: The path of the Hyprland IPC socket
event_types
: The type of events to be listened for
None.
None.
Sender | Handler method to add to your class | Arguments |
---|---|---|
Hyprland | on_hyprland_event | event, argument |
Sway | on_sway_event_[type] | payload |
Systemd Logind | on_systemd_event | sender, signal, payload |
Systemd Logind | on_[signal] | payload |
They are well documented here.
For Sway, the current event types are those defined in the IPC manual
These events are called "signals" in the Systemd terminology.
They are not well documented but you can try to read that (good luck).
Some examples that can be useful:
Member | Description |
---|---|
PrepareForShutdown | Sent before a shutdown |
PrepareForSleep | Sent before suspend |
Lock | Sent when a lock is requested, eg loginctrl lock-session |
Unlock | Sent when an unlock is requested |
SessionNew | When a session is created |
Keep in mind that these signals are independent from Wayland/Hyprland/Sway. My recommendation would be to always configure your compositor to use loginctl to send the signals, and add a listener in Easyland to achieve what you want.
Method | Usage | Arguments |
---|---|---|
command.exec | Execute a command, eventually in the background | cmd (string), background (bool, default False), decode_json (bool, default False) |
command.hyprland_get_all_monitors | Get all monitors and their configuration through Hyprland IPC | None |
command.hyprland_get_monitor | Get the config of one monitor, None if not found | name, description, make, model |
command.sway_get_all_monitors | Get all monitors and their configuration through Hyprland IPC | None |
command.sway_get_monitor | Get the config of one monitor, None if not found | name, make, model |
logger | Log messages to easyland.log and to STDOUT | use logger.info, logger.error, for the severity etc. |
idle_config | Set the idle configuration | None |
- Integrating other DBUS services should be easy with Easyland (type
dbusctl
to list all avalable DBUS on your system). Do not hesitate to let me know what you need. - Better tests for Sway. I use Hyprland so feel free to submit bugs if you are using Sway and see an issue.
If you see some bugs or propose patches, feel free to contribute.
Thanks to the developer(s) of Hyprland for their fantastic compositor. I tried so many ones in the past, and this has been Hyprland that convinced me to do the switch from KDE :-)