Fix HDMI audio after suspend/resume on PipeWire + WirePlumber
Project description
aproman
Fix HDMI audio after suspend/resume on Linux systems running PipeWire + WirePlumber.
The Problem
When a Linux system resumes from suspend, HDMI audio devices often lose their
connection. WirePlumber tries to link to stale node proxies, resulting in
silence. The only manual fix is to open your audio settings and switch the card
profile away (for example to off) and back, forcing a full teardown and
rebuild of the audio nodes.
How It Works
aproman runs as a service (systemd or OpenRC) and:
- Auto-detects your HDMI audio card, or uses the one saved in the config file
- Monitors D-Bus for
PrepareForSleepsignals from systemd-logind (or elogind) - On wake, waits briefly for HDMI to renegotiate, then cycles the card profile off and back on
- Monitors PipeWire via
pw-dumpfor nodes entering an error state, and automatically restarts PipeWire to recover (with a 30-second cooldown to prevent restart loops)
This forces PipeWire and WirePlumber to rebuild fresh nodes, restoring audio without manual intervention.
Requirements
- PipeWire with WirePlumber, or PulseAudio compatibility via PipeWire
pactldbus-monitorpw-dump(optional, for node error monitoring)- A Linux distribution with systemd or OpenRC (elogind for OpenRC)
Installation
systemd
uv tool install aproman
aproman install-service
systemctl --user start aproman.service
This installs aproman to ~/.local/bin/, copies the systemd user service into
place, and enables it.
OpenRC user service (0.60+, Alpine edge, etc.)
uv tool install aproman
aproman install-service
rc-service --user aproman start
On OpenRC 0.60 or newer, install-service automatically installs a user-level
service to ~/.config/rc/init.d/aproman. Make sure ~/.local/bin is on your
PATH.
OpenRC system service (older OpenRC)
sudo uv pip install --system --break-system-packages aproman
sudo aproman install-service
sudo rc-service aproman start
On OpenRC versions before 0.60, install-service installs a system-level init
script to /etc/init.d/aproman and adds it to the default runlevel. The service
uses supervise-daemon for process supervision with automatic restart.
To configure the user and environment for the daemon, create
/etc/conf.d/aproman:
command_user="youruser"
supervise_daemon_args="--env XDG_RUNTIME_DIR=/run/user/1000"
Replace 1000 with your user's UID (id -u youruser).
Alternative: install.sh (systemd)
git clone https://github.com/mwolson/aproman.git
cd aproman
./install.sh
systemctl --user start aproman.service
This copies aproman to ~/.local/bin/ and installs and enables the user
service.
Optional: set defaults
After installing, you can optionally save your preferred card and profile so that aproman uses them instead of auto-detecting:
aproman list-cards
aproman set-default-card alsa_card.pci-0000_01_00.1
aproman list-profiles
aproman set-default-profile pro-audio
These write to ~/.config/aproman.conf and signal the running daemon to pick up
the changes. Without defaults, aproman auto-detects the first HDMI card and uses
its active profile at startup.
Usage
The service runs automatically. To check status:
systemd
systemctl --user status aproman.service
journalctl --user -u aproman.service -f
OpenRC (user, 0.60+)
rc-service --user aproman status
OpenRC (system, older)
rc-service aproman status
Commands
aproman uses subcommands for one-off operations. With no subcommand, it runs as a daemon.
aproman Run as a daemon (default)
aproman cycle Cycle the card profile off and back on
aproman get-default-card Print the default card from the config file
aproman get-default-profile Print the default profile from the config file
aproman install-service Install and enable the service (systemd or OpenRC)
aproman list-cards List available audio cards
aproman list-profiles List available profiles for the card
aproman set-default-card CARD Save default card and signal the daemon
aproman set-default-profile PROFILE Save default profile and signal the daemon
aproman uninstall-service Disable and remove the service (systemd or OpenRC)
Daemon options
These flags apply to the daemon and to cycle:
--card CARD PipeWire/PulseAudio card name (default: config file, then auto-detect HDMI)
--profile PROFILE Desired audio profile (default: config file, then active profile)
--wake-delay SECONDS Seconds to wait after wake before cycling (default: 3.0)
Configuration File
aproman reads defaults from ~/.config/aproman.conf (or
$XDG_CONFIG_HOME/aproman.conf). The file uses one flag per line:
--card=alsa_card.pci-0000_01_00.1
--profile=pro-audio
Only --card and --profile are supported. Unrecognized flags cause an error
at startup. Command-line arguments always take precedence over the config file.
When the daemon receives a reload signal (sent automatically by
set-default-card and set-default-profile via the Unix socket, or manually
via kill -HUP), it reloads the config file and updates the card and profile
for future suspend/resume cycles.
One-Shot Fix
If audio breaks and the daemon missed the resume event (for example, after a service restart), you can manually trigger a single profile cycle:
aproman cycle
This sends a cycle request to the running daemon via its Unix socket. If the daemon is unavailable, it falls back to running the cycle directly.
When the card is stuck in the off state, cycle automatically selects the
highest-priority available profile. You can override with --profile:
aproman --profile pro-audio cycle
Example: Custom Card and Profile
aproman set-default-card alsa_card.pci-0000_01_00.1
aproman set-default-profile output:hdmi-stereo
Uninstall
aproman uninstall-service
uv tool uninstall aproman # or: rm ~/.local/bin/aproman
rm -f ~/.config/aproman.conf
Testing
bun run test # unit tests
bun run test:integration # Docker-based integration tests
bun run test:all # both
Hooks
bun run hooks:check # run checks against working tree
lefthook install # install git hooks
The pre-commit hook runs uvx ruff check, uvx ty check, and the unit test
suite.
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file aproman-0.5.2.tar.gz.
File metadata
- Download URL: aproman-0.5.2.tar.gz
- Upload date:
- Size: 24.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f6d86908787df92e4b35f937b43e13a05a4530786bfa5d3ff16036eb73ad9a95
|
|
| MD5 |
444989f8425f856908a41bd8b0b664a1
|
|
| BLAKE2b-256 |
d3001b191e1bee15cab19fdbf95445845a7100e21e2a5d215f7ec9e89064330f
|
Provenance
The following attestation bundles were made for aproman-0.5.2.tar.gz:
Publisher:
publish.yml on mwolson/aproman
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
aproman-0.5.2.tar.gz -
Subject digest:
f6d86908787df92e4b35f937b43e13a05a4530786bfa5d3ff16036eb73ad9a95 - Sigstore transparency entry: 1280811069
- Sigstore integration time:
-
Permalink:
mwolson/aproman@b0a25853ab1f8a8cd1bc95c9c6345619bff12b0a -
Branch / Tag:
refs/tags/v0.5.2 - Owner: https://github.com/mwolson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@b0a25853ab1f8a8cd1bc95c9c6345619bff12b0a -
Trigger Event:
push
-
Statement type:
File details
Details for the file aproman-0.5.2-py3-none-any.whl.
File metadata
- Download URL: aproman-0.5.2-py3-none-any.whl
- Upload date:
- Size: 13.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7368baf5b7dbab490de1db1090cd3e5e977370ab8bc37c4876195f0ddf7058b1
|
|
| MD5 |
82d3b5355c80d8f7527354335ef3e88d
|
|
| BLAKE2b-256 |
bc7a2a0440419c8b66231d79b98397d891d953c3e8ae182ec6abdc6b7dcd672e
|
Provenance
The following attestation bundles were made for aproman-0.5.2-py3-none-any.whl:
Publisher:
publish.yml on mwolson/aproman
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
aproman-0.5.2-py3-none-any.whl -
Subject digest:
7368baf5b7dbab490de1db1090cd3e5e977370ab8bc37c4876195f0ddf7058b1 - Sigstore transparency entry: 1280811081
- Sigstore integration time:
-
Permalink:
mwolson/aproman@b0a25853ab1f8a8cd1bc95c9c6345619bff12b0a -
Branch / Tag:
refs/tags/v0.5.2 - Owner: https://github.com/mwolson
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@b0a25853ab1f8a8cd1bc95c9c6345619bff12b0a -
Trigger Event:
push
-
Statement type: