Tcl
Tcl
SSH Parallel Remote Commands on Multiple Servers
See more SSH Examples
Shows how to execute a command in parallel on multiple servers.Chilkat Tcl Downloads
load ./chilkat.dll
set success 0
# This example requires the Chilkat API to have been previously unlocked.
# See Global Unlock Sample for sample code.
# Executing a command on multiple servers simultaneously is straightforward.
# It's just a matter of using one SSH object per server..
set ssh1 [new_CkSsh]
set ssh2 [new_CkSsh]
set ssh3 [new_CkSsh]
set port 22
set success [CkSsh_Connect $ssh1 "ssh-server1.com" $port]
if {$success != 1} then {
puts [CkSsh_lastErrorText $ssh1]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
# Authenticate using login/password:
set success [CkSsh_AuthenticatePw $ssh1 "sshLogin1" "sshPassword1"]
if {$success != 1} then {
puts [CkSsh_lastErrorText $ssh1]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
# Connect and authenticate with 2 more servers.
# For brevity, the success/failure won't be checked...
set success [CkSsh_Connect $ssh2 "ssh-server2.com" $port]
set success [CkSsh_AuthenticatePw $ssh2 "sshLogin2" "sshPassword2"]
set success [CkSsh_Connect $ssh3 "ssh-server3.com" $port]
set success [CkSsh_AuthenticatePw $ssh3 "sshLogin3" "sshPassword3"]
# Note: If we wanted, we could've used ConnectAsync and AuthenticatePwAsync
# to do the connecting and authenticating in parallel...
# The command to be run on each SSH server will sleep for 5 seconds,
# and then show the current system date/time.
set cmd "sleep 5; date"
# Start each command
set ssh1Channel [CkSsh_QuickCmdSend $ssh1 $cmd]
if {$ssh1Channel < 0} then {
puts [CkSsh_lastErrorText $ssh1]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
# For brevity, we're not checking the return values here:
set ssh2Channel [CkSsh_QuickCmdSend $ssh2 $cmd]
set ssh3Channel [CkSsh_QuickCmdSend $ssh3 $cmd]
# OK, at this point the command is running simultaneously on each server.
# Now collect the results of each command.
set pollTimeoutMs 50
set numFinished 0
# Note: You would rewrite this code to use arrays.
set ssh1Finished 0
set ssh2Finished 0
set ssh3Finished 0
while {$numFinished < 3} {
# Check to see if anything has finished.
# QuickCmdCheck returns -1 if there are no errors and nothing else finished
# QuickCmdCheck returns -2 if there was an error (such as a lost connection)
# QuickCmdCheck returns a channel number if a channel finished.
if {$ssh1Finished != 1} then {
set channel [CkSsh_QuickCmdCheck $ssh1 $pollTimeoutMs]
if {$channel == -2} then {
puts [CkSsh_lastErrorText $ssh1]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
if {$channel == $ssh1Channel} then {
puts "---- ssh1 channel $channel finished ----"
puts [CkSsh_getReceivedText $ssh1 $channel ansi]
set numFinished [expr $numFinished + 1]
set ssh1Finished 1
}
}
if {$ssh2Finished != 1} then {
set channel [CkSsh_QuickCmdCheck $ssh2 $pollTimeoutMs]
if {$channel == -2} then {
puts [CkSsh_lastErrorText $ssh2]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
if {$channel == $ssh2Channel} then {
puts "---- ssh2 channel $channel finished ----"
puts [CkSsh_getReceivedText $ssh2 $channel ansi]
set numFinished [expr $numFinished + 1]
set ssh2Finished 1
}
}
if {$ssh3Finished != 1} then {
set channel [CkSsh_QuickCmdCheck $ssh3 $pollTimeoutMs]
if {$channel == -2} then {
puts [CkSsh_lastErrorText $ssh3]
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3
exit
}
if {$channel == $ssh3Channel} then {
puts "---- ssh3 channel $channel finished ----"
puts [CkSsh_getReceivedText $ssh3 $channel ansi]
set numFinished [expr $numFinished + 1]
set ssh3Finished 1
}
}
}
# --------------
# Sample output:
# ---- ssh2 channel 101 finished ----
# Fri Dec 23 00:25:48 UTC 2016
#
# ---- ssh3 channel 102 finished ----
# Thu Dec 22 18:25:12 CST 2016
#
# ---- ssh1 channel 100 finished ----
# Thu Dec 22 18:25:48 CST 2016
delete_CkSsh $ssh1
delete_CkSsh $ssh2
delete_CkSsh $ssh3