Blame lib/ssh.exp

Packit 62fe53
# Copyright (C) 2016 Free Software Foundation, Inc.
Packit 62fe53
#
Packit 62fe53
# This file is part of DejaGnu.
Packit 62fe53
#
Packit 62fe53
# DejaGnu is free software; you can redistribute it and/or modify it
Packit 62fe53
# under the terms of the GNU General Public License as published by
Packit 62fe53
# the Free Software Foundation; either version 3 of the License, or
Packit 62fe53
# (at your option) any later version.
Packit 62fe53
#
Packit 62fe53
# DejaGnu is distributed in the hope that it will be useful, but
Packit 62fe53
# WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 62fe53
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 62fe53
# General Public License for more details.
Packit 62fe53
#
Packit 62fe53
# You should have received a copy of the GNU General Public License
Packit 62fe53
# along with DejaGnu; if not, write to the Free Software Foundation,
Packit 62fe53
# Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
Packit 62fe53
Packit 62fe53
# Connect using ssh(1).
Packit 62fe53
Packit 62fe53
set ssh_initialized "no"
Packit 62fe53
set ssh_useropts " -o ControlPersist=yes -o ControlMaster=auto -o ControlPath=\"/tmp/ssh-%r@%h:%p\""
Packit 62fe53
Packit 62fe53
# Default to the ssh and scp in the user's path.
Packit 62fe53
set SSH ssh
Packit 62fe53
set SCP scp
Packit 62fe53
Packit 62fe53
# Download SRCFILE to DESTFILE on DESTHOST.
Packit 62fe53
#
Packit 62fe53
proc ssh_download {desthost srcfile destfile} {
Packit 62fe53
    global SSH SCP ssh_initialized timeout
Packit 62fe53
Packit 62fe53
    set ssh_port ""
Packit 62fe53
    set ssh_user ""
Packit 62fe53
    set ssh_useropts ""
Packit 62fe53
    set name ""
Packit 62fe53
    set hostname ""
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists scp_prog]} {
Packit 62fe53
	set SCP [board_info $desthost scp_prog]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists ssh_prog]} {
Packit 62fe53
	set SSH [board_info $desthost ssh_prog]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    # The default user name is the person running the tests
Packit 62fe53
    if {[board_info $desthost exists username]} {
Packit 62fe53
	set ssh_user "[board_info $desthost username]@"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists ssh_opts]} {
Packit 62fe53
	append ssh_useropts " [board_info $desthost ssh_opts]"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    # The default SSH port is 22
Packit 62fe53
    if {[board_info $desthost exists port]} {
Packit 62fe53
	set ssh_port "[board_info $desthost port]"
Packit 62fe53
    } else {
Packit 62fe53
	set ssh_port 22
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists name]} {
Packit 62fe53
	set name [board_info $desthost name]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists hostname]} {
Packit 62fe53
	set hostname [board_info $desthost hostname]
Packit 62fe53
    } else {
Packit 62fe53
	set hostname $desthost
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    append ssh_useropts " -o ControlPersist=yes -o ControlMaster=auto -o ControlPath=/tmp/ssh-%r@%h:%p"
Packit 62fe53
Packit 62fe53
    set ret [local_exec "$SCP -P $ssh_port $ssh_useropts $srcfile $ssh_user$hostname:$destfile" "" "" $timeout]
Packit 62fe53
    set status [lindex $ret 0]
Packit 62fe53
    set output [lindex $ret 1]
Packit 62fe53
    if { $status == 0 } {
Packit 62fe53
	set ssh_initialized "yes"
Packit 62fe53
	verbose "Copied $srcfile to $desthost:$destfile" 2
Packit 62fe53
	return $destfile
Packit 62fe53
    } else {
Packit 62fe53
	verbose "Download via ssh to $desthost failed."
Packit 62fe53
	return ""
Packit 62fe53
    }
Packit 62fe53
}
Packit 62fe53
Packit 62fe53
proc ssh_upload {desthost srcfile destfile} {
Packit 62fe53
    global SSH SCP
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists scp_prog]} {
Packit 62fe53
	set SCP [board_info $desthost scp_prog]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists username]} {
Packit 62fe53
	set ssh_user "[board_info $desthost username]@"
Packit 62fe53
    } else {
Packit 62fe53
	set ssh_user ""
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists name]} {
Packit 62fe53
	set desthost [board_info $desthost name]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists hostname]} {
Packit 62fe53
	set desthost [board_info $desthost hostname]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    set status [catch "exec $SCP $ssh_user$desthost:$srcfile $destfile" output]
Packit 62fe53
    if { $status == 0 } {
Packit 62fe53
	verbose "Copied $desthost:$srcfile to $destfile" 2
Packit 62fe53
	return $destfile
Packit 62fe53
    } else {
Packit 62fe53
	verbose "Upload from $desthost failed, $output."
Packit 62fe53
	return ""
Packit 62fe53
    }
Packit 62fe53
}
Packit 62fe53
Packit 62fe53
# Execute CMD on BOARDNAME.
Packit 62fe53
#
Packit 62fe53
proc ssh_exec { boardname program pargs inp outp } {
Packit 62fe53
    global SSH timeout
Packit 62fe53
Packit 62fe53
    set ssh_port ""
Packit 62fe53
    set scp_port ""
Packit 62fe53
    set ssh_user ""
Packit 62fe53
    set ssh_useropts ""
Packit 62fe53
    set name ""
Packit 62fe53
    set hostname ""
Packit 62fe53
Packit 62fe53
    verbose "Executing on $boardname: $program $pargs"
Packit 62fe53
Packit 62fe53
    if {![board_info $boardname exists ssh_prog]} {
Packit 62fe53
	set SSH ssh
Packit 62fe53
    } else {
Packit 62fe53
	set SSH [board_info $boardname ssh_prog]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $boardname exists username]} {
Packit 62fe53
	set ssh_user "[board_info $boardname username]@"
Packit 62fe53
    } else {
Packit 62fe53
	set ssh_user ""
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $boardname exists ssh_useropts]} {
Packit 62fe53
	append ssh_useropts " [board_info $boardname ssh_opts]"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $boardname exists name]} {
Packit 62fe53
	set boardname [board_info $boardname name]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $boardname exists hostname]} {
Packit 62fe53
	set hostname [board_info $boardname hostname]
Packit 62fe53
    } else {
Packit 62fe53
	set hostname $boardname
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $boardname  exists port]} {
Packit 62fe53
	append ssh_useropts " -p [board_info $boardname port]"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    append ssh_useropts " -o ControlPersist=yes -o ControlMaster=auto -o ControlPath=\"/tmp/ssh-%r@%h:%p\""
Packit 62fe53
Packit 62fe53
    # If CMD sends any output to stderr, exec will think it failed.
Packit 62fe53
    # More often than not that will be true, but it doesn't catch the
Packit 62fe53
    # case where there is no output but the exit code is non-zero.
Packit 62fe53
    if { $inp == "" } {
Packit 62fe53
	set inp "/dev/null"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    # We use && here, as otherwise the echo always works, which makes it look
Packit 62fe53
    # like execution succeeded when in reality it failed.
Packit 62fe53
    set ret [local_exec "$SSH $ssh_useropts $ssh_user$hostname sh -c '$program $pargs && echo XYZ\\\${?}ZYX \\; rm -f $program'" $inp $outp $timeout]
Packit 62fe53
    set status [lindex $ret 0]
Packit 62fe53
    set output [lindex $ret 1]
Packit 62fe53
Packit 62fe53
    verbose "$SSH status is $status, output is $output"
Packit 62fe53
Packit 62fe53
    # `status' doesn't mean much here other than ssh worked ok.
Packit 62fe53
    # What we want is whether $program ran ok.  Return $status
Packit 62fe53
    # if the program timed out, status will be 1 indicating that
Packit 62fe53
    # ssh ran and failed.  If ssh fails, we will get FAIL rather
Packit 62fe53
    # than UNRESOLVED - this will help the problem be noticed.
Packit 62fe53
    if { $status != 0 } {
Packit 62fe53
	regsub "XYZ(\[0-9\]*)ZYX\n?" $output "" output
Packit 62fe53
	return [list $status "$SSH to $boardname failed for $program, $output"]
Packit 62fe53
    }
Packit 62fe53
    if { [regexp "XYZ(\[0-9\]*)ZYX" $output junk status] == 0 } {
Packit 62fe53
	set status ""
Packit 62fe53
    }
Packit 62fe53
    verbose "ssh_exec: status:$status text:$output" 4
Packit 62fe53
    if { $status == "" } {
Packit 62fe53
	return [list -1 "Couldn't parse $SSH output, $output."]
Packit 62fe53
    }
Packit 62fe53
    regsub "XYZ(\[0-9\]*)ZYX\n?" $output "" output
Packit 62fe53
    # Delete one trailing \n because that is what `exec' will do and we want
Packit 62fe53
    # to behave identical to it.
Packit 62fe53
    regsub "\n$" $output "" output
Packit 62fe53
    return [list [expr {$status != 0}] $output]
Packit 62fe53
}
Packit 62fe53
Packit 62fe53
proc ssh_close { desthost } {
Packit 62fe53
    global SSH ssh_initialized
Packit 62fe53
Packit 62fe53
    verbose "Closing the SSH connection to $desthost"
Packit 62fe53
    
Packit 62fe53
    set ssh_port ""
Packit 62fe53
    set scp_port ""
Packit 62fe53
    set ssh_user ""
Packit 62fe53
    set ssh_useropts ""
Packit 62fe53
    set name ""
Packit 62fe53
    set hostname ""
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists username]} {
Packit 62fe53
	set ssh_useropts "-l [board_info $desthost username]"
Packit 62fe53
	set ssh_user "[board_info $desthost username]@"
Packit 62fe53
    } else {
Packit 62fe53
	set ssh_user ""
Packit 62fe53
	set ssh_useropts ""
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists hostname]} {
Packit 62fe53
	set hostname [board_info $desthost hostname]
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists ssh_opts]} {
Packit 62fe53
	append ssh_useropts " [board_info $desthost ssh_opts]"
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    if {[board_info $desthost exists port]} {
Packit 62fe53
	set ssh_port " -p [board_info $desthost port]"
Packit 62fe53
    } else {
Packit 62fe53
	set ssh_port ""
Packit 62fe53
    }
Packit 62fe53
Packit 62fe53
    set args "$ssh_user$hostname $ssh_port"
Packit 62fe53
Packit 62fe53
   # Kill the remote server
Packit 62fe53
    set status [catch "exec ssh $ssh_port -o ControlPath=/tmp/ssh-%r@%h:%p -O exit $args"]
Packit 62fe53
    set ssh_initialized "no"
Packit 62fe53
Packit 62fe53
     return ""
Packit 62fe53
}