xrandr は udev から実行すると失敗する

xrandr は udev から実行すると失敗する

画面レイアウトを検出し、xrandr を使用して更新することを目的とした Python 3 スクリプト「myscript」からの次の抜粋は、 または で実行すると正常に機能しsudo /usr/local/bin/myscriptます/usr/local/bin/myscript

xrandr_cmd = Popen("xrandr", shell=True, stdout=PIPE, stderr=STDOUT)

ただし、次の udev ルールの結果として実行される場合:

ACTION="change", SUBSYSTEM="drm", ENV{HOTPLUG}=="1", RUN+="/usr/local/bin/myscript"

失敗し、xrandr が「ディスプレイを開けません」というメッセージとともに 1 を返したと表示されます。

udev ルールから実行したときに xrandr が失敗する理由を知っている人はいますか?

興味のある人のために、全文を紹介します。

#! /usr/bin/env python3
import os
from subprocess import Popen, PIPE, STDOUT

def log(s):
    home_dir = os.path.expanduser("~")
    #with open (f"{home_dir}/monitor_script.log", "a+") as f:
    with open (f"/home/vedantroy/monitor_script.log", "a+") as f:
        f.write(s)
xrandr = "/usr/bin/xrandr"
xrandr_cmd = Popen(xrandr, shell=True, stdout=PIPE, stderr=STDOUT)
retval = xrandr_cmd.wait()
lines = map(lambda l: l.decode('ascii'), xrandr_cmd.stdout.readlines())
if retval != 0:
    nl = "\n"
    log(f"xrandr returned {retval} with output:\n{nl.join(lines)}")
else:
    layout_cmds = [
        # No monitors plugged in
        f"{xrandr} --auto",
        # Thinkpad T580
        # Monitor plugged into HDMI port
        # Monitor to right of laptop
        f"{xrandr} --output HDMI2 --primary --auto --right-of eDP1"
    ]
    layout = 0
    for line in lines:
        if "HDMI2 connected" in line:
            layout = 1
            break

    layout_cmd_str = layout_cmds[layout]
    layout_cmd = Popen(layout_cmd_str, shell=True)
    retval = layout_cmd.wait()
    if retval != 0:
        log(f"{layout_cmd_str} returnd {retval}")

答え1

udev は X にアクセスできないため、環境変数を指定する必要がありDISPLAYますXAUTHORITY

KERNEL=="card0", SUBSYSTEM=="drm", ACTION=="change", ENV{DISPLAY}=":0", ENV{XAUTHORITY}="/home/ben/.Xauthority", RUN+="/bin/bash /path/to/script.sh"

(出典:https://frdmtoplay.com/i3-udev-xrandr-ホットプラグ出力切り替え/

もちろん、.Xauthorityファイルとスクリプトへのパスを調整する必要があります。

関連情報