如何将命令发送到特定的终端窗口?


13


我想编写一个脚本,用于在单独的终端中同时打开多个程序(服务器)-无关紧要-并通过在正确终端内“登陆”的命令将不同的命令分配给不同的终端。这可能吗?
也许是这样的:

  1. 打开终端1
  2. //与1同时打开terminal2。
  3. command1 //在terminal1中执行而无需打开新的终端窗口
  4. command2 //在terminal2中执行而无需打开新的终端窗口
  5. ...

我可以以某种方式标记终端窗口,以便在正确的终端内执行命令吗?

我还想在它们的程序运行时观察所有终端-我的程序有一个参数,用于将跟踪/调试打印到终端。所以我想看看它们之间交换了什么消息。

注意:我不太担心交换数据的安全性,因为该脚本应充当“模拟”。我已经将每个服务器配置为从本地主机上分配的端口运行。


检查pssh ....
heemayl

时间应该有多精确;假设2秒钟(每个终端)合适吗?
Jacob Vlijm 2015年

@JacobVlijm:对我来说更重要的是正确地根据终端“窗口”分配命令
Aliakbar Ahmadi 2015年

1
可以完成,尤其是在模拟时,将回发:)
Jacob Vlijm 2015年

1
@JacomVlijm:实际上我的问题被偶然解决了:为了将命令发送到正确的实例,每个命令必须以实例开始的datadir前缀!但是,幸运的是,这是在比特币中实现的,但我只是不回答这个问题..也许有人对任何程序都提出了更一般的想法!:)但是,谢谢!
Aliakbar Ahmadi 2015年

Answers:


14

自从您提到以来,您已针对自己的具体情况解决了该问题,下面提供了一个通用解决方案。多亏了xdotool--sync选择,它在我运行的测试中仍然非常可靠;我可以将命令“发送”到特定的终端窗口,并且运行无异常。

在实践中如何运作

该解决方案由一个脚本,它可以与两个选项运行中存在 -set-run

  1. 设置(打开)任意数量的终端窗口,在此示例中:3

    target_term -set 3

    将打开三个新的终端,它们的窗口ID会在一个隐藏文件中被记住:

    在此处输入图片说明

    为了清楚起见,我最小化了终端窗口:

  2. 现在,我创建了三个窗口,我可以使用run命令(例如)将命令发送到其中一个:

    target_term -run 2 echo "Monkey eats banana since it ran out of peanuts"

    如下所示,该命令在第二个终端中运行:

    在此处输入图片说明

    随后,我可以向第一个终端发送命令:

     target_term -run 1 sudo apt-get update

    使sudo apt-get update运行在终端1:

    在此处输入图片说明

    等等...

如何设定

  1. 该脚本同时需要wmctrlxdotool

    sudo apt-get install wmctrl xdotool
  2. 将以下脚本复制到一个空文件中,将其安全保存为target_term(无扩展名!)~/bin(在~/bin必要时创建目录)中。

  3. 使脚本可执行(不要忘记),然后注销/登录或运行:

    source ~/.profile
  4. 现在,以所需的窗口数作为参数来设置终端窗口:

    target_term -set <number_of_windows>
  5. 现在,您可以使用以下命令将命令“发送”到任一终端:

    target_term -run <terminal_number> <command_to_run>

剧本

#!/usr/bin/env python3
import subprocess
import os
import sys
import time
#--- set your terminal below
application = "gnome-terminal"
#---

option = sys.argv[1]
data = os.environ["HOME"]+"/.term_list"

def current_windows():
    w_list = subprocess.check_output(["wmctrl", "-lp"]).decode("utf-8")
    w_lines = [l for l in w_list.splitlines()]
    try:
        pid = subprocess.check_output(["pgrep", application]).decode("utf-8").strip()
        return [l for l in w_lines if str(pid) in l]
    except subprocess.CalledProcessError:
        return []

def arr_windows(n):
    w_count1 = current_windows()
    for requested in range(n):
        subprocess.Popen([application])
    called = []
    while len(called) < n:
        time.sleep(1)
        w_count2 = current_windows()
        add = [w for w in w_count2 if not w in w_count1]
        [called.append(w.split()[0]) for w in add if not w in called]
        w_count1 = w_count2

    return called

def run_intterm(w, command):
    subprocess.call(["xdotool", "windowfocus", "--sync", w])
    subprocess.call(["xdotool", "type", command+"\n"]) 

if option == "-set":
    open(data, "w").write("")
    n = int(sys.argv[2])
    new = arr_windows(n)
    for w in new:
        open(data, "a").write(w+"\n")
elif option == "-run":
    t_term = open(data).read().splitlines()[int(sys.argv[2])-1]
    command = (" ").join(sys.argv[3:])
    run_intterm(t_term, command)

笔记

  • 将该脚本设置为gnome-terminal,但可以通过更改application脚本头部的将该脚本用于任何终端(或其他程序):

    #--- set your terminal below
    application = "gnome-terminal"
    #---
  • 上面的命令(当然)也可以从脚本运行,以防您愿意将其用于某种模拟。
  • 该脚本将一直等待,直到目标窗口都具有焦点并且命令完成键入为止,因此该命令将始终位于正确的终端窗口中。
  • 无需说该脚本仅适用于命令调用的终端设置(Windows):

    target_term -set

    就像您在问题中提到的那样,终端窗口将由脚本“标记”。

  • 万一您开始一个新的target_term会话,由脚本创建的隐藏文件将被简单地覆盖,因此不需要删除它。

很好,谢谢!还应注意,python 3.x也是此脚本正常工作的要求。
pompalini
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.