屏幕锁定/关闭时,cmd.Run()会永久悬挂在Macos的golang中

时间:2019-03-24 07:22:07

标签: macos go osascript

我正在Macos上运行golang应用程序。它具有一些类似于以下的代码:

for {
    time.Sleep(time.Second * 5)
    cmd := exec.Command("/usr/bin/osascript", "-e", `display dialog "hello" with title "hello"`)
    err := cmd.Run()
}

如果我不锁定屏幕(屏幕始终处于打开状态),则效果很好。但是,如果屏幕被锁定,则代码err := cmd.Run()将永远挂起,并在该行执行时关闭。当我解锁屏幕(打开屏幕)时,for循环会永远挂在那里,永远不会继续执行。

我不确定此问题是否属于golang或MacOS如何处理osascript。谁能告诉我如何解决呢?非常感谢。

PS:我在Linux中使用相同的代码,并将/usr/bin/osascript替换为/usr/bin/xmessage,即使在Linux中锁定/关闭屏幕,这也可以正常工作而不会出现任何问题。

已编辑:

我的解决方案,请改用chrome:

cmd := exec.Command(`/Applications/Google Chrome.app/Contents/MacOS/Google Chrome`, "-new-window", "/path/hello.html")

2 个答案:

答案 0 :(得分:3)

这似乎与MacOS在屏幕锁定时如何使进程处于空闲状态有关。它使osasscript子进程永远不会完成执行并阻塞for循环。

您可以做的一件事是在超时上下文中运行命令。我已经尝试过了,而且效果很好。当屏幕解锁并且超时到期时,执行将恢复。

示例:

package main

import (
    "context"
    "fmt"
    "os/exec"
    "time"
)

func main() {
    for {
        time.Sleep(time.Second * 5)

        // run your command with a timeout
        ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
        cmd := exec.CommandContext(
            ctx,
            "/usr/bin/osascript",
            "-e",
            `display dialog "hello" with title "hello"`,
        )

        err := cmd.Run()
        if err != nil {
            fmt.Println(err)
        }
        // don't forget to cancel your context to avoid context leak
        cancel()
    }
}

或者,如果您不希望超时,则可以在尝试调用显示对话框之前检查屏幕是否被锁定。

package main

import (
    "context"
    "fmt"
    "os"
    "os/exec"
    "strings"
    "time"
)

func main() {
    for {
        time.Sleep(time.Second * 5)

        ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
        cmd := exec.CommandContext(
            ctx,
            "python",
            "-c",
            "import sys,Quartz; d=Quartz.CGSessionCopyCurrentDictionary(); print d",
        )

        var err error
        var b []byte
        if b, err = cmd.CombinedOutput(); err != nil {
            cancel()
            continue
        }
        cancel()

        // if screen is not locked
        if !strings.Contains(string(b), "CGSSessionScreenIsLocked = 1") {
            cmd = exec.Command(
                "/usr/bin/osascript",
                "-e",
                "display dialog \"Hello\"",
            )
            cmd.Stdout = os.Stdout
            cmd.Stderr = os.Stderr

            err = cmd.Run()
            if err != nil {
                fmt.Println("err: ", err)
            }
        }
    }
}

答案 1 :(得分:0)

您可以在命令末尾使用 &,这将导致程序在后台运行。

 for {
    time.Sleep(time.Second * 5)
    cmd := exec.Command("/usr/bin/osascript", "-e", `display dialog "hello" with title "hello"`, "&")
    err := cmd.Run()
}