Avendo recentemente riscontrato questo problema (in un altro modo: Vim in esecuzione su un server remoto e avevo dimenticato lo schermo), ho deciso di cercare un modo.
La prima idea era quella di cercare i descrittori di file utilizzati da Vim e provare a scriverlo. Le fds di Vim indicano il psedoterminale aperto dall'emulatore terminale, abbastanza naturalmente:
$ ls -l /proc/$(pgrep -n vim)/fd/
total 0
lrwx------ 1 muru muru 64 Nov 17 01:25 0 -> /dev/pts/14
lrwx------ 1 muru muru 64 Nov 17 01:25 1 -> /dev/pts/14
lrwx------ 1 muru muru 64 Nov 17 01:25 2 -> /dev/pts/14
lrwx------ 1 muru muru 64 Nov 17 01:25 3 -> socket:[99564312]
Tuttavia, i miei primi tentativi falliti:
echo '^[:wq^M' > /proc/$(pgrep -n vim)/fd/0
echo ':wq^M' > /proc/$(pgrep -n vim)/fd/0
echo ':wq^M' > /proc/$(pgrep -n vim)/fd/0
echo '^C' > /proc/$(pgrep -n vim)/fd/0
printf "%s" '^[:wqa!^M' > /proc/$(pgrep -n vim)/fd/0
Gli ^[
e ^M
sono stati ottenuti da CtrlVEsce CtrlVEnter, rispettivamente.
Tutti hanno portato alla comparsa dei personaggi sul terminale (lo stavo testando localmente, prima di applicarlo alla sessione remota). Cercando su Google, ho trovato questo post SO , usando Python per scrivere sul dispositivo pseudoterminale:
#!/usr/bin/python
import sys,os,fcntl,termios
if len(sys.argv) != 3:
sys.stderr.write("usage: ttyexec.py tty command\n")
sys.exit(1)
fd = os.open("/dev/" + sys.argv[1], os.O_RDWR)
cmd=sys.argv[2]
for i in range(len(cmd)):
fcntl.ioctl(fd, termios.TIOCSTI, cmd[i])
fcntl.ioctl(fd, termios.TIOCSTI, '\n')
os.close(fd)
E provarlo su una shell interattiva in pitone ha funzionato:
$ sudo python3
Python 3.5.0 (default, Sep 20 2015, 11:28:25)
[GCC 5.2.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import os, fcntl, termios
>>> fd = os.open('/dev/pts/14', os.O_RDWR)
>>> a = '\033:wqa!\n'
>>> for i in a: fcntl.ioctl(fd, termios.TIOCSTI, i);
...
b'\x1b'
b':'
b'w'
b'q'
b'a'
b'!'
b'\n'
>>>
Fatto!