user2033758
user2033758

Reputation: 1986

Running process of remote SSH server in the background using Python Paramiko

I am finding hard to run a process on a remote SSH server at background using Paramiko. I used :

stdin, stdout, stderr = ssh.exec_command('executefile.py &') 

and found that no process of executefile.py was found running.

Then I tried using other way as including a backward slash:

stdin, stdout, stderr = ssh.exec_command('executefile.py \&') 

This method worked. There was an instance running on machine but no surprise, it was not running at background. I could come to know as it is not running at background as when code stuck at second line after this code. It was

all_inf = stdout.readlines()

Now code was not going beyond above line unless the process of the script was killed.

I am learning Paramiko, any help is appreciated.

Upvotes: 11

Views: 22406

Answers (5)

Ariel Camino
Ariel Camino

Reputation: 279

I've tried all the methods described here and here without success, and finally realized that you need to use channels instead of using the SSHClient directly for calling exec_command (this does not work in background):

client = paramiko.SSHClient()
client.connect(ip_address, username='root', pkey=paramiko_key, timeout=5)
client.exec_command('python script.py > /dev/null 2>&1 &')

You should create and use a channel, this works in background:

client = paramiko.SSHClient()
client.connect(ip_address, username='root', pkey=paramiko_key, timeout=5)
transport = client.get_transport()
channel = transport.open_session()
channel.exec_command('python script.py > /dev/null 2>&1 &')

So nohup, dtach, screen, etc, are actually not necessary.

Upvotes: 27

shivram
shivram

Reputation: 469

You could try using screen

screen -d -m ping 8.8.8.8

This is would start a screen and ping 8.8.8.8. You can view this screen by using

screen -ls

and attach using

screen -D <<screen_name>>

Note that the screen will terminate after the command has finished executing.

Upvotes: 1

cizixs
cizixs

Reputation: 13981

You can try:

stdin, stdout, stderr = ssh.exec_command('nohup python executefile.py >/dev/null 2>&1 &') 

Upvotes: 5

Zeinab Abbasimazar
Zeinab Abbasimazar

Reputation: 10439

I tried transport class and it was really great. Here's the code I used:

ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(hostname = "host_ip", username = "un"], password = "up")
channel = ssh.get_transport().open_session()
pty = channel.get_pty()
shell = ssh.invoke_shell()
shell.send("cd /my/directory/; nohup ./exec_name > /dev/null 2>&1 &\n")

But I still don't know how to kill it using python scripts; I have an open question about it here.

EDIT 1:

I have solved my problem about killing the process somehow; you can check it.

Upvotes: 1

JimB
JimB

Reputation: 109325

exec_command isn't executing the command in an interactive shell, so "running a process in the background" doesn't really make sense.

If you really want to do this, you could use the command nohup to start your process, and keep it alive when the session exits. Remember that you can't get stdin, stdout, or stderr when you do this, since you are detaching the process from the shell, so redirect them accordingly.

Upvotes: 3

Related Questions