我在/home/jack/Code/Service/main.py
中有以下Python脚本:
from subprocess import Popen
Popen(["/usr/bin/gedit"])
while True:
pass
我正在使用此systemd服务运行它:
[Unit]
Description=Test
[Service]
Type=simple
Restart=no
User=jack
WorkingDirectory=/home/jack/Code/Service
ExecStart=/usr/bin/python3 main.py
[Install]
WantedBy=multi-user.target
当我使用main.py
从命令行运行python3 main.py
时,它将运行并打开gedit。当我启动systemd服务并查看日志时,我看到:
mars 10 11:19:55 Ubuntu-Tower systemd[1]: Started Test.
mars 10 11:19:55 Ubuntu-Tower python3[8827]: Unable to init server: Could not connect: Connection refused
mars 10 11:19:55 Ubuntu-Tower gedit[8835]: cannot open display:
请注意,gedit
进程实际上在服务运行时确实出现在系统监视器中(并且只有在我停止服务时才消失),因此这似乎确实是特定于GUI的问题。
这是怎么回事,如何获得所需的行为?
答案 0 :(得分:1)
要在X11,Linux和其他系统的Windows系统中打开一个窗口,必须指定显示。通常,这是在DISPLAY环境变量中定义的。在我的情况下,它设置为“:0.0”。如果未设置此变量,则程序将不知道在何处绘制窗口:
marco$ DISPLAY= xeyes
Error: Can't open display:
marco$
某些程序允许您通过-display
参数指定显示。仅此信息还不够:X11包含了可以连接到特定服务器的允许客户端的访问控制列表。
marco$ su - test
Password:
test$ env | grep DISPLAY
DISPLAY=:0.0
test$ xeyes
No protocol specified
Error: Can't open display: :0.0
xhost
是操纵ACL的工具:
marco$ xhost +
access control disabled, clients can connect from any host
marco$ su - test
Password:
# disable X11 acl with xhost +
test:~$ xeyes
^C
无论如何,请记住X11服务器必须正在运行才能打开窗口,如果启动服务时服务器未在监听,则可能会引发其他类型的错误(无法初始化服务器:无法连接:连接被拒绝)