Выполнить команду в другом терминале через /dev/pts

У меня есть терминал, который использует STDIN 3 (/proc/xxxx/fd/0 -> /dev/pts/3)

Итак, если (в другом терминале) я делаю:

echo 'do_something_command' > /dev/pts/3

Команда отображается в моем первом (pts/3) терминале, но команда не выполняется. И если (в этом терминале pts/3) я в программе жду каких-то данных от stdin, данные записываются на экран, но программа не захватывает их с stdin.

Что я хочу сделать, так это выполнить команду "do_something_command", а не только показать ее.

Может кто-нибудь объяснить мне такое поведение? Как мне достичь своего намерения?


person ton    schedule 13.07.2014    source источник
comment
Нужно ли нажимать ввод в /dev/pts/3 после получения данных?   -  person Nullpointer    schedule 13.07.2014
comment
Я пробовал... если я вхожу на терминал (pts 3) после отправки данных в /dev/pts/3, принимается только ввод. а не do_something_command. как будто его и не было... только показывали.   -  person ton    schedule 13.07.2014
comment
реальное намерение состоит в том, чтобы управлять mpg123 -R с другого терминала. но мне интересно узнать о функционале pts и что можно или нельзя с ним делать. и почему.   -  person ton    schedule 13.07.2014
comment
гм... не решение для этого, но команда mpg123 имеет параметр fifo, который создает именованный канал для решения моей проблемы.   -  person ton    schedule 13.07.2014
comment
Вы можете посмотреть на xdotool. На самом деле pts не действует как труба.   -  person Mark Plotnick    schedule 14.10.2014
comment
Это хороший инструмент, но он предназначен только для Xorg, я хочу сделать это, например, в ssh-соединениях.   -  person ton    schedule 14.10.2014
comment
ХОРОШО. Затем вы можете добиться того, чего хотите, используя программу, которая использует ioctl TIOCSTI для имитации ввода с терминала. На stackoverflow.com есть несколько примеров на нескольких языках. Я не знаю, какие языки вы предпочитаете, поэтому попробуйте в окне поиска найти TIOCSTI и посмотрите, подходит ли вам один из них.   -  person Mark Plotnick    schedule 19.10.2014


Ответы (2)


Я полностью понимаю, о чем вы спрашиваете. Вы можете добиться этого, самостоятельно написав и выполнив небольшой фрагмент кода на C. Это должно дать вам некоторое представление.

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <string.h>
#include <unistd.h>

void print_help(char *prog_name) {
        printf("Usage: %s [-n] DEVNAME COMMAND\n", prog_name);
        printf("Usage: '-n' is an optional argument if you want to push a new line at the end of the text\n");
        printf("Usage: Will require 'sudo' to run if the executable is not setuid root\n");
        exit(1);
}

int main (int argc, char *argv[]) {
    char *cmd, *nl = "\n";
    int i, fd;
    int devno, commandno, newline;
    int mem_len;
    devno = 1; commandno = 2; newline = 0;
    if (argc < 3) {
        print_help(argv[0]);
    }
    if (argc > 3 && argv[1][0] == '-' && argv[1][1] == 'n') {
        devno = 2; commandno = 3; newline=1;
    } else if (argc > 3 && argv[1][0] == '-' && argv[1][1] != 'n') {
        printf("Invalid Option\n");
        print_help(argv[0]);
    }
    fd = open(argv[devno],O_RDWR);
    if(fd == -1) {
        perror("open DEVICE");
        exit(1);
    }
    mem_len = 0;
    for (i = commandno; i < argc; i++) {
        mem_len += strlen(argv[i]) + 2;
        if (i > commandno) {
            cmd = (char *)realloc((void *)cmd, mem_len);
        } else { // i == commandno
            cmd = (char *)malloc(mem_len);
        }

        strcat(cmd, argv[i]);
        strcat(cmd, " ");
    }
  if (newline == 0)
        usleep(225000);
    for (i = 0; cmd[i]; i++)
        ioctl (fd, TIOCSTI, cmd+i);
    if (newline == 1)
        ioctl (fd, TIOCSTI, nl);
    close(fd);
    free((void *)cmd);
    exit (0);
}

Скомпилируйте и выполните его с разрешениями sudo. Например, если вы хотите выполнить команду на /dev/pts/3, просто выполните sudo ./a.out -n /dev/pts/3 whoami, запустите whoami на /dev/pts/3.

Этот код был полностью взят из эта страница.

person John Strood    schedule 22.10.2016
comment
так что мне делать, чтобы открыть интерактивную оболочку другого терминала? - person rojen; 04.10.2020
comment
Также, если вы хотите лучше понять это, я бы предложил man ioctl_tty и обратил внимание на вариант TIOCSTI. - person ChalkTalk; 15.12.2020

Кажется, вы используете неправильные кавычки вокруг команды. Либо удалите кавычки и команду эха, либо используйте эхо и обратные галочки (`).

Пытаться:

echo `date` > /dev/pts/3

или просто

date > /dev/pts/3

Обратите внимание, что все, что запускается в /dev/pts/3, не сможет прочитать то, что всплывает «сзади».

person Rein    schedule 14.10.2014
comment
Спасибо, но это не то, что я ищу... ваше предложение только перенаправляет вывод команды в /dev/pts/3, я пытаюсь выполнить команду в /dev/pts/3. - person ton; 14.10.2014