diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml
index a53ab78..f201d30 100644
--- a/.pre-commit-config.yaml
+++ b/.pre-commit-config.yaml
@@ -20,4 +20,4 @@ repos:
     hooks:
       - id: mypy
         args: [--strict]
-        additional_dependencies: [paho-mqtt]
+        additional_dependencies: [paho-mqtt, pillow]
diff --git a/PKGBUILD b/PKGBUILD
index 2f1bf0c..cbabed1 100644
--- a/PKGBUILD
+++ b/PKGBUILD
@@ -15,7 +15,7 @@ arch=(any)
 url="https://git.edgarpierre.fr/edpibu/hasspy"
 license=('GPL-3.0-or-later')
 groups=()
-depends=('python-paho-mqtt')
+depends=('python-paho-mqtt' 'python-pillow')
 makedepends=('git' 'uv' 'python-installer') # 'bzr', 'git', 'mercurial' or 'subversion'
 provides=("${pkgname%-git}")
 conflicts=("${pkgname%-git}")
diff --git a/hasspy/__init__.py b/hasspy/__init__.py
index 8a97f14..7807b61 100644
--- a/hasspy/__init__.py
+++ b/hasspy/__init__.py
@@ -1,4 +1,5 @@
 import logging
+import logging.config
 import signal
 import tomllib
 from argparse import ArgumentParser
@@ -10,7 +11,6 @@ log = logging.getLogger(__name__)
 
 
 def main() -> int:
-    log.info("Starting HassPy")
     parser = ArgumentParser(
         prog="HassPy",
         description="Home Assistant MQTT client",
@@ -42,9 +42,11 @@ def main() -> int:
 
     if isinstance(config.get("log_level"), str):
         config["log_level"] = getattr(logging, config["log_level"])
-    logging.basicConfig(
-        level=config.get("log_level", logging.INFO) - (args.verbose * 10)
-    )
+    config["log_level"] = config.get("log_level", logging.INFO) - (args.verbose * 10)
+
+    logging.basicConfig(level=config["log_level"])
+
+    log.info("Starting HassPy")
 
     ha: HassClient
     if not args.user:
diff --git a/hasspy/mqtt.py b/hasspy/mqtt.py
index d6dc613..dbb70eb 100644
--- a/hasspy/mqtt.py
+++ b/hasspy/mqtt.py
@@ -1,11 +1,14 @@
+import io
 import json
 import logging
+import re
 from subprocess import run
-from threading import Timer
+from threading import Thread, Timer
 from typing import Any, Mapping
 
 from paho.mqtt.client import Client, MQTTMessage, MQTTMessageInfo
 from paho.mqtt.enums import CallbackAPIVersion, MQTTErrorCode
+from PIL import Image
 
 log = logging.getLogger(__name__)
 
@@ -23,7 +26,9 @@ class HassClient(Client):
 
         self.interval = self.config.get("interval", 60)
         self.power_on = True
-        self.timer = Timer(self.interval, self.publish_state)
+        self.timer = Timer(0, self.publish_state)
+
+        self.cover = ""
 
         self.connect()
 
@@ -58,7 +63,6 @@ class HassClient(Client):
         self.message_callback_add(self.command_topic, self.on_command)
 
     def publish_state(self) -> MQTTMessageInfo:
-        self.timer.cancel()
         self.timer = Timer(self.interval, self.publish_state)
         self.timer.start()
 
@@ -74,7 +78,9 @@ class HassClient(Client):
 
         self.do_command(*payload.split(":"))
 
-        self.publish_state()
+        self.timer.cancel()
+        self.timer = Timer(1, self.publish_state)
+        self.timer.start()
 
     def do_command(self, cmd: str, value: str = "") -> None:
         pass
@@ -85,7 +91,7 @@ class HassClient(Client):
         self.publish_availability()
         self.init_subs()
 
-        self.publish_state()
+        self.timer.start()
 
     @property
     def state_topic(self) -> str:
@@ -139,9 +145,7 @@ class HassSystemClient(HassClient):
     def do_command(self, cmd: str, value: str = "") -> None:
         if cmd in self.commands:
             log.debug(f"Executing command: {cmd}")
-            proc = run(self.commands[cmd])
-            if proc.returncode != 0:
-                log.error(f"Failed to execute command: {cmd}")
+            run_command(self.commands[cmd])
 
         if cmd == "POWER_ON":
             self.power_on = True
@@ -179,6 +183,9 @@ class HassSystemClient(HassClient):
 class HassUserClient(HassClient):
     commands = {
         "PLAY_PAUSE": ["playerctl", "play-pause"],
+        "PLAY_NEXT": ["playerctl", "next"],
+        "PLAY_PREV": ["playerctl", "previous"],
+        "PLAY_STOP": ["playerctl", "stop"],
     }
 
     def __init__(self, node_id: str, config: Mapping[str, Any]) -> None:
@@ -187,14 +194,13 @@ class HassUserClient(HassClient):
     def do_command(self, cmd: str, value: str = "") -> None:
         if cmd in self.commands:
             log.debug(f"Executing command: {cmd}")
-            proc = run(self.commands[cmd])
-            if proc.returncode != 0:
-                log.error(f"Failed to execute command: {cmd}")
+            run_command(self.commands[cmd])
 
         match [cmd, value]:
             case ["VOLUME", value]:
                 log.debug(f"Executing command: {cmd}:{value}")
-                proc = run(
+
+                run_command(
                     [
                         "wpctl",
                         "set-volume",
@@ -202,8 +208,6 @@ class HassUserClient(HassClient):
                         f"{int(value) / 100:.2f}",
                     ]
                 )
-                if proc.returncode != 0:
-                    log.error(f"Failed to set volume: {value}")
 
     @property
     def availability_topic(self) -> str:
@@ -219,6 +223,44 @@ class HassUserClient(HassClient):
                 "icon": "mdi:play-pause",
                 "payload_press": "PLAY_PAUSE",
             },
+            "next": {
+                "unique_id": f"{self.node_id}_next",
+                "p": "button",
+                "name": "Next",
+                "icon": "mdi:skip-next",
+                "payload_press": "PLAY_NEXT",
+            },
+            "prev": {
+                "unique_id": f"{self.node_id}_prev",
+                "p": "button",
+                "name": "Previous",
+                "icon": "mdi:skip-previous",
+                "payload_press": "PLAY_PREV",
+            },
+            "stop": {
+                "unique_id": f"{self.node_id}_stop",
+                "p": "button",
+                "name": "Stop",
+                "icon": "mdi:stop",
+                "payload_press": "PLAY_STOP",
+            },
+            "player": {
+                "unique_id": f"{self.node_id}_player",
+                "p": "sensor",
+                "name": "Player",
+                "icon": "mdi:music",
+                "value_template": "{{ value_json.player.value }}",
+                "json_attributes_topic": self.state_topic,
+                "json_attributes_template": "{{ value_json.player.attributes | to_json }}",
+            },
+            "cover": {
+                "unique_id": f"{self.node_id}_cover",
+                "p": "image",
+                "name": "Cover",
+                "icon": "mdi:disc-player",
+                "content_type": "image/webp",
+                "image_topic": self.cover_topic,
+            },
             "volume": {
                 "unique_id": f"{self.node_id}_volume",
                 "p": "number",
@@ -237,13 +279,59 @@ class HassUserClient(HassClient):
     def state_payload(self) -> dict[str, Any]:
         return {
             "volume": self.volume_value,
+            "player": self.player_value,
         }
 
     @property
     def volume_value(self) -> int:
-        proc = run(["wpctl", "get-volume", "@DEFAULT_AUDIO_SINK@"], capture_output=True)
-        if proc.returncode != 0:
-            log.error("Failed to get volume")
-            return 0
+        vol = run_command(["wpctl", "get-volume", "@DEFAULT_AUDIO_SINK@"])
 
-        return int(float(proc.stdout.decode("utf-8").split(": ")[1]) * 100)
+        return int(float(vol.split(": ")[1]) * 100)
+
+    @property
+    def player_value(self) -> str | dict[str, str | dict[str, str]]:
+        return {
+            "value": run_command(["playerctl", "status"]),
+            "attributes": {
+                k: run_command(["playerctl", "metadata", k])
+                for k in ["title", "album", "artist"]
+            },
+        }
+
+    def publish_state(self) -> MQTTMessageInfo:
+        Thread(target=self.publish_cover).start()
+        return super().publish_state()
+
+    @property
+    def cover_topic(self) -> str:
+        return f"{self.node_id}/image/cover"
+
+    def publish_cover(self) -> None:
+        log.debug("Publishing cover image")
+        out = run_command(["playerctl", "metadata"])
+
+        artUrl = re.compile(r"mpris:artUrl\s+file://(.*)").search(out)
+        if not artUrl:
+            return
+
+        art = artUrl.group(1)
+
+        if art == self.cover:
+            return
+
+        self.cover = art
+        by = io.BytesIO()
+        with Image.open(art) as im:
+            im.save(by, format="webp")
+
+        by.seek(0)
+        self.publish(self.cover_topic, by.read())
+
+
+def run_command(cmd: list[str]) -> str:
+    proc = run(cmd, capture_output=True)
+    if proc.returncode != 0:
+        log.error(f"Failed to execute command: {cmd}")
+        return "null"
+
+    return proc.stdout.decode("utf-8")
diff --git a/pyproject.toml b/pyproject.toml
index 6078950..60fb4c8 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -6,6 +6,7 @@ readme = "README.md"
 requires-python = ">=3.13"
 dependencies = [
     "paho-mqtt>=2.1.0",
+    "pillow>=11.1.0",
 ]
 
 [dependency-groups]
diff --git a/uv.lock b/uv.lock
index e0c824a..6e05769 100644
--- a/uv.lock
+++ b/uv.lock
@@ -8,6 +8,7 @@ version = "0.1.0"
 source = { virtual = "." }
 dependencies = [
     { name = "paho-mqtt" },
+    { name = "pillow" },
 ]
 
 [package.dev-dependencies]
@@ -17,7 +18,10 @@ dev = [
 ]
 
 [package.metadata]
-requires-dist = [{ name = "paho-mqtt", specifier = ">=2.1.0" }]
+requires-dist = [
+    { name = "paho-mqtt", specifier = ">=2.1.0" },
+    { name = "pillow", specifier = ">=11.1.0" },
+]
 
 [package.metadata.requires-dev]
 dev = [
@@ -62,6 +66,33 @@ wheels = [
     { url = "https://files.pythonhosted.org/packages/c4/cb/00451c3cf31790287768bb12c6bec834f5d292eaf3022afc88e14b8afc94/paho_mqtt-2.1.0-py3-none-any.whl", hash = "sha256:6db9ba9b34ed5bc6b6e3812718c7e06e2fd7444540df2455d2c51bd58808feee", size = 67219 },
 ]
 
+[[package]]
+name = "pillow"
+version = "11.1.0"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/f3/af/c097e544e7bd278333db77933e535098c259609c4eb3b85381109602fb5b/pillow-11.1.0.tar.gz", hash = "sha256:368da70808b36d73b4b390a8ffac11069f8a5c85f29eff1f1b01bcf3ef5b2a20", size = 46742715 }
+wheels = [
+    { url = "https://files.pythonhosted.org/packages/b3/31/9ca79cafdce364fd5c980cd3416c20ce1bebd235b470d262f9d24d810184/pillow-11.1.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:ae98e14432d458fc3de11a77ccb3ae65ddce70f730e7c76140653048c71bfcbc", size = 3226640 },
+    { url = "https://files.pythonhosted.org/packages/ac/0f/ff07ad45a1f172a497aa393b13a9d81a32e1477ef0e869d030e3c1532521/pillow-11.1.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:cc1331b6d5a6e144aeb5e626f4375f5b7ae9934ba620c0ac6b3e43d5e683a0f0", size = 3101437 },
+    { url = "https://files.pythonhosted.org/packages/08/2f/9906fca87a68d29ec4530be1f893149e0cb64a86d1f9f70a7cfcdfe8ae44/pillow-11.1.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:758e9d4ef15d3560214cddbc97b8ef3ef86ce04d62ddac17ad39ba87e89bd3b1", size = 4326605 },
+    { url = "https://files.pythonhosted.org/packages/b0/0f/f3547ee15b145bc5c8b336401b2d4c9d9da67da9dcb572d7c0d4103d2c69/pillow-11.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b523466b1a31d0dcef7c5be1f20b942919b62fd6e9a9be199d035509cbefc0ec", size = 4411173 },
+    { url = "https://files.pythonhosted.org/packages/b1/df/bf8176aa5db515c5de584c5e00df9bab0713548fd780c82a86cba2c2fedb/pillow-11.1.0-cp313-cp313-manylinux_2_28_aarch64.whl", hash = "sha256:9044b5e4f7083f209c4e35aa5dd54b1dd5b112b108648f5c902ad586d4f945c5", size = 4369145 },
+    { url = "https://files.pythonhosted.org/packages/de/7c/7433122d1cfadc740f577cb55526fdc39129a648ac65ce64db2eb7209277/pillow-11.1.0-cp313-cp313-manylinux_2_28_x86_64.whl", hash = "sha256:3764d53e09cdedd91bee65c2527815d315c6b90d7b8b79759cc48d7bf5d4f114", size = 4496340 },
+    { url = "https://files.pythonhosted.org/packages/25/46/dd94b93ca6bd555588835f2504bd90c00d5438fe131cf01cfa0c5131a19d/pillow-11.1.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:31eba6bbdd27dde97b0174ddf0297d7a9c3a507a8a1480e1e60ef914fe23d352", size = 4296906 },
+    { url = "https://files.pythonhosted.org/packages/a8/28/2f9d32014dfc7753e586db9add35b8a41b7a3b46540e965cb6d6bc607bd2/pillow-11.1.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:b5d658fbd9f0d6eea113aea286b21d3cd4d3fd978157cbf2447a6035916506d3", size = 4431759 },
+    { url = "https://files.pythonhosted.org/packages/33/48/19c2cbe7403870fbe8b7737d19eb013f46299cdfe4501573367f6396c775/pillow-11.1.0-cp313-cp313-win32.whl", hash = "sha256:f86d3a7a9af5d826744fabf4afd15b9dfef44fe69a98541f666f66fbb8d3fef9", size = 2291657 },
+    { url = "https://files.pythonhosted.org/packages/3b/ad/285c556747d34c399f332ba7c1a595ba245796ef3e22eae190f5364bb62b/pillow-11.1.0-cp313-cp313-win_amd64.whl", hash = "sha256:593c5fd6be85da83656b93ffcccc2312d2d149d251e98588b14fbc288fd8909c", size = 2626304 },
+    { url = "https://files.pythonhosted.org/packages/e5/7b/ef35a71163bf36db06e9c8729608f78dedf032fc8313d19bd4be5c2588f3/pillow-11.1.0-cp313-cp313-win_arm64.whl", hash = "sha256:11633d58b6ee5733bde153a8dafd25e505ea3d32e261accd388827ee987baf65", size = 2375117 },
+    { url = "https://files.pythonhosted.org/packages/79/30/77f54228401e84d6791354888549b45824ab0ffde659bafa67956303a09f/pillow-11.1.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:70ca5ef3b3b1c4a0812b5c63c57c23b63e53bc38e758b37a951e5bc466449861", size = 3230060 },
+    { url = "https://files.pythonhosted.org/packages/ce/b1/56723b74b07dd64c1010fee011951ea9c35a43d8020acd03111f14298225/pillow-11.1.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:8000376f139d4d38d6851eb149b321a52bb8893a88dae8ee7d95840431977081", size = 3106192 },
+    { url = "https://files.pythonhosted.org/packages/e1/cd/7bf7180e08f80a4dcc6b4c3a0aa9e0b0ae57168562726a05dc8aa8fa66b0/pillow-11.1.0-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9ee85f0696a17dd28fbcfceb59f9510aa71934b483d1f5601d1030c3c8304f3c", size = 4446805 },
+    { url = "https://files.pythonhosted.org/packages/97/42/87c856ea30c8ed97e8efbe672b58c8304dee0573f8c7cab62ae9e31db6ae/pillow-11.1.0-cp313-cp313t-manylinux_2_28_x86_64.whl", hash = "sha256:dd0e081319328928531df7a0e63621caf67652c8464303fd102141b785ef9547", size = 4530623 },
+    { url = "https://files.pythonhosted.org/packages/ff/41/026879e90c84a88e33fb00cc6bd915ac2743c67e87a18f80270dfe3c2041/pillow-11.1.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:e63e4e5081de46517099dc30abe418122f54531a6ae2ebc8680bcd7096860eab", size = 4465191 },
+    { url = "https://files.pythonhosted.org/packages/e5/fb/a7960e838bc5df57a2ce23183bfd2290d97c33028b96bde332a9057834d3/pillow-11.1.0-cp313-cp313t-win32.whl", hash = "sha256:dda60aa465b861324e65a78c9f5cf0f4bc713e4309f83bc387be158b077963d9", size = 2295494 },
+    { url = "https://files.pythonhosted.org/packages/d7/6c/6ec83ee2f6f0fda8d4cf89045c6be4b0373ebfc363ba8538f8c999f63fcd/pillow-11.1.0-cp313-cp313t-win_amd64.whl", hash = "sha256:ad5db5781c774ab9a9b2c4302bbf0c1014960a0a7be63278d13ae6fdf88126fe", size = 2631595 },
+    { url = "https://files.pythonhosted.org/packages/cf/6c/41c21c6c8af92b9fea313aa47c75de49e2f9a467964ee33eb0135d47eb64/pillow-11.1.0-cp313-cp313t-win_arm64.whl", hash = "sha256:67cd427c68926108778a9005f2a04adbd5e67c442ed21d95389fe1d595458756", size = 2377651 },
+]
+
 [[package]]
 name = "ruff"
 version = "0.9.10"