Jak wysłać polecenie terminalu do terminala TTY

8

Szukam sposobu na wysłanie wykonanego polecenia do terminala TTY. Powiedzmy na przykład, że otwieram terminal gnome i w oknie, które piszę sudo aptitude update && sudo aptitude upgrade.

Jak mogę wysłać to do terminala TTY zamiast pracować z nim w środowisku GUI TTY7?

Luis Alvarado
źródło

Odpowiedzi:

7

Radziłbym nie robić tego bezpośrednio, ale jako pośrednika użyj terminalowego multipleksera, takiego jak tmux.

W terminalu, który powinien otrzymać polecenie start tmux z identyfikatorem:

tmux new-session -s MYSES

Wysyłaj do niego polecenia za pomocą:

tmux send-keys -t MYSES "sudo aptitude update && sudo aptitude upgrade"$'\n'
Thor
źródło
jeśli używasz byobu, to również działa! I możesz upuścić -t MYSES, jeśli zdarza się, że sesja już działa, a nie znasz nazwy. Dziękuję!
ExploWare
10

To jest program, który pomoże:

#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);
}

Skopiuj powyższy kod do pliku C (np. Ttyecho.c). Uruchom następującą komendę w katalogu, w którym utworzono plik C, aby skompilować kod.

make ttyecho

Skopiuj ten plik do katalogu bin w katalogu domowym. Utwórz katalog, jeśli nie istnieje. Dobrą praktyką jest przechowywanie wszystkich niestandardowych plików binarnych / plików wykonywalnych w tym katalogu bin.

Uruchom inny terminal lub przełącz się na inny otwarty terminal, którym chcesz sterować, i wykonaj polecenie tty. Poniżej możesz zobaczyć przykładowe wyjście.

@~$ tty

/ dev / pts / 5

Teraz, aby wykonać polecenie na / dev / pts / 5 , uruchom następującą komendę w terminalu control / original.

sudo ttyecho -n /dev/pts/5 ls

Zobaczysz, że polecenie ls jest wykonywane w / dev / pts / 5 . Opcja -n powoduje, że ttyecho wysyła nowy wiersz po poleceniu, aby polecenie zostało wykonane, a nie tylko wstawione. Tego narzędzia można używać do przesyłania dowolnych danych do innych terminali. Na przykład, możesz otworzyć vim w / dev / pts / 5, a następnie uruchomić następującą komendę w terminalu sterującym, aby spowodować, że vim wyjdzie z / dev / pts / 5 .

sudo ttyecho -n /dev/pts/5 :q

Aby uniknąć korzystania z sudo przez cały czas, aby komenda była łatwa do skryptowania, zmień właścicieli / uprawnienia tego pliku wykonywalnego za pomocą następujących poleceń.

sudo chown root:root ttyecho
sudo chmod u+s ttyecho

Uwaga: Ustawienie bitu setuid może stać się zagrożeniem bezpieczeństwa.

Źródło: http://www.humbug.in/2010/utility-to-send-commands-or-data-to-other-terminals-ttypts/

Eyal Levin
źródło