Linux display命令在终端中工作,但在systemd服务中不起作用

时间:2015-12-13 23:00:16

标签: linux ubuntu service go systemd

我制作了一个网络应用来关闭我的电脑屏幕,有一些不同的技术,但它相当简单:

我有一个html / js前端,可以检测按钮点击(Screens On / Screens Off),它通过ajax将选项发送到PHP后端

然后,php通过tcp端口连接,将选项发送到用golang编写的程序

然后我的golang程序执行命令关闭/打开屏幕。 它运行的命令是(" xset -display:0 dpms force off")

我遇到的问题是该命令仅在我在终端中运行golang程序时才有效,但当我将其设置为服务时,该命令将不起作用。

这是golang代码:

package main

import (
    "os/exec"
    "net"
    "fmt"
    "bufio"
)

func main() {
    fmt.Println("Launching server")

    ln, _ := net.Listen("tcp", ":7777")
    fmt.Println("Listening...\n")

    for {
        // accept connection on port
        conn, _ := ln.Accept()
        fmt.Println("New connection")

        // listen for message ending in \n
        message, _ := bufio.NewReader(conn).ReadString('\n')
        rec := string(message)

        // remove trailing \n
        rec = rec[:len(rec)-1]

        fmt.Println("Message Received: ", "\""+rec+"\"")

        returnMessage := "fail"

        if (rec == "screensOff") {
            fmt.Println("Turning off screens...")

            //execute screens off command
            cmd := exec.Command("xset", "-display", ":0", "dpms", "force", "off")
            stdout, err := cmd.Output()

            if err != nil {
                fmt.Println(err.Error())
            } else {
                fmt.Println(string(stdout))
                returnMessage = "done"
            }
        } else if (rec == "screensOn") {
            fmt.Println("Turning on screens...");

            //execute screens on command
            cmd := exec.Command("xset", "-display", ":0", "dpms", "force", "on")

            stdout, err := cmd.Output()
            if err != nil {
                fmt.Println(err.Error())
            } else {
                fmt.Println(string(stdout))
                returnMessage = "done"
            }
            returnMessage = "done"
        } 

        conn.Write([]byte(returnMessage + "\n"))

        conn.Close()
        fmt.Println("Connection closed\n")
    }
}

相关的PHP代码:

<?php
function sendServiceMessage($message) {
    $host = "localhost";
    $port = 7777;
    $timeout = 30;

    // connect to service
    $socket = fsockopen($host, $port, $errnum, $errstr, $timeout);
    if (!is_resource($socket)) {
        exit("connection fail: ".$errnum." ".$errstr);
    }
    else {
        // send message
        fputs($socket, $message."\n");

        // receive return message
        $recieved = "";
        while (!feof($socket)) {
            $recieved .= fgets ($socket, 1024);
        }
    }

    // close connection
    fclose($socket);
    if ($recieved == "done") {
        return true;
    }
    return false;   
}

sendServiceMessage("screensOff");

我使用systemd来设置服务,所以在构建程序并将其放在/ usr / bin /

之后
...$ go build screenControl.go
...$ sudo cp screenControl /usr/bin/screenControl

我可以在终端中运行screenControl程序,然后选择&#34;屏幕关闭&#34;在网络应用程序中,它都按预期工作:

...$ screenControl
Launching server
Listening...

New Connection
Message Received:  "screensOff"
Turning off screens...

Connection closed

然后我创建了一个systemd单元文件(/etc/systemd/system/screenControl.service):

[Unit]
Description=Screen control service

[Service]
ExecStart=/usr/bin/screenControl
Restart=on-abort

[Install]
WantedBy=multi-user.target

我启动了服务并检查了它:

...$ systemctl start screenControl
...$ systemctl status screenControl
● screenControl.service - Screen control service
   Loaded: loaded (/etc/systemd/system/screenControl.service; disabled; vendor preset: enabled)
   Active: active (running) since Sun 2015-12-13 22:31:54 GMT; 6s ago
 Main PID: 19871 (screenControl)
   CGroup: /system.slice/screenControl.service
           └─19871 /usr/bin/screenControl

Dec 13 22:31:54 User systemd[1]: Started Screen control service.
Dec 13 22:31:54 User screenControl[19871]: Launching server
Dec 13 22:31:54 User screenControl[19871]: Listening...

所以它正在运行,但是当我现在在网络应用程序中关闭屏幕时,没有任何反应......我再次检查服务状态并且它正在接收消息以关闭屏幕但命令是退出时出错:

...
Dec 13 22:31:54 User screenControlTest[19871]: Launching server
Dec 13 22:31:54 User screenControlTest[19871]: Listening...
Dec 13 22:32:25 User screenControlTest[19871]: New connection
Dec 13 22:32:25 User screenControlTest[19871]: Message Received:  "screensOff"
Dec 13 22:32:25 User screenControlTest[19871]: Turning off screens...
Dec 13 22:32:25 User screenControlTest[19871]: exit status 1
Dec 13 22:32:25 User screenControlTest[19871]: Connection closed

这里的问题是什么?如何让该命令作为服务工作?一旦这个工作,我希望在机器开启时自动启动服务,尽管使用systemd我认为这很简单:

...$ systemctl enable screenControl

任何帮助都会很棒,谢谢:)

修改

让golang程序向我展示xset命令的stderr后,我现在也有错误消息:

xset:  unable to open display ""

2 个答案:

答案 0 :(得分:1)

xset命令只是X服务器的客户端。它通过检查DISPLAY环境变量来确定要与哪个X服务器通信,当您将命令作为系统服务运行时,该变量将不会被设置。

即使您确保在运行守护程序时设置了DISPLAY,它也可能作为不同的用户帐户运行,并且默认情况下拒绝访问显示。

更好的选择是将您的守护程序作为用户会话的一部分运行。这将解决身份验证问题(它将像您一样运行),以及定位显示的能力(环境变量应该是可见的)。当您未登录时,守护程序将不会运行,但这可能与此特定用例无关。

您已使用“Ubuntu”标记了您的问题,其中会话仍由Upstart管理。您可以通过在~/.config/upstart中创建文件来创建新的用户会话作业。可以在init(5) man page

中找到文件格式的详细信息

答案 1 :(得分:0)

根据David Budworth的评论,修复非常简单;由于服务在root下运行,因此它没有设置DISPLAY环境变量。

在go中你可以在使用exec时设置环境变量:

//execute screens off command
cmd := exec.Command("xset", "-display", ":0", "dpms", "force", "off")
cmd.Env = []string{"DISPLAY=:0"} // set the display before executing
stdout, stderr := cmd.CombinedOutput() //execute and return all output

来自James Henstridge的回答我发现我还需要运行xhost +SI:localuser:root以允许root用户访问X服务器。

您可以在用户登录后通过将此行添加到/etc/profile文件的顶部

来为用户执行此操作
xhost +SI:localuser:root > /dev/null 2>&1

即使没有用户登录(显示登录屏幕时),您也可以使用它

首先我创建了目录/opt/scripts 然后创建了文件/opt/scripts/xhost.sh 并使用chmod +x /opt/scripts/xhost.sh

为其提供了可执行权限

在这个文件中只有一行:

xhost +SI:localuser:root > /dev/null 2>&1

然后编辑文件/etc/lightdm/lightdm.conf(我必须创建它,但如果它在那里就编辑它)并添加该行 display-setup-script=/opt/scripts/xhost.sh

所以我的lightdm.conf文件如下所示:

[SeatDefaults]
greeter-session=unity-greeter
user-session=ubuntu
display-setup-script=/opt/scripts/xhost.sh

这告诉LightDM(在Ubuntu中运行的显示管理器)在X服务器启动之后但在其他任何事情之前运行脚本/opt/scripts/xhost.sh,因此root会直接获得xhost授权!

注意:

display-setup-script在X服务器启动之后但在运行用户会话/ greeter之前运行。如果需要在X服务器中配置任何特殊内容,请设置此项。它以root身份运行。如果此命令返回错误代码,则X服务器将停止。 来源:https://wiki.ubuntu.com/LightDM