2010-11-22 22:57:00 +01:00
# Source `init.tcl' again to restore the `unknown' procedure
# NOTE: DejaGnu has an old `unknown' procedure which unfortunately disables
# tcl auto-loading.
2009-07-12 21:37:57 +02:00
source [file join [info library] init.tcl]
2010-10-31 17:51:14 +01:00
package require cmdline
2009-07-12 21:37:57 +02:00
package require textutil::string
2009-12-10 23:53:22 +02:00
# Execute a bash command and make sure the exit status is successful.
2009-06-09 22:49:53 +02:00
# If not, output the error message.
2009-12-10 23:53:22 +02:00
# @param string $cmd Bash command line to execute. If empty string (""), the
2009-06-09 22:49:53 +02:00
# exit status of the previously executed bash command will be
# checked; specify `title' to adorn the error message.
# @param string $title (optional) Command title. If empty, `cmd' is used.
2009-07-17 16:15:02 +02:00
# @param string $prompt (optional) Bash prompt. Default is "/@"
2010-01-24 10:32:41 +01:00
# @param mixed $out (optional) Reference to (tcl) variable to hold output.
2009-09-26 11:00:02 +02:00
# If variable equals -1 (default) the bash command is expected
# to return no output. If variable equals 0, any output
# from the bash command is disregarded.
proc assert_bash_exec {{aCmd ""} {title ""} {prompt /@} {out -1}} {
if {$out != 0 && $out != -1} {upvar $out results}
2009-06-09 22:49:53 +02:00
if {[string length $aCmd] != 0} {
send "$aCmd\r"
expect -ex "$aCmd\r\n"
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
if {[string length $title] == 0} {set title $aCmd}
2009-07-17 16:15:02 +02:00
expect -ex $prompt
2009-09-20 13:07:04 +02:00
set results $expect_out(buffer); # Catch output
# Remove $prompt suffix from output
set results [
string range $results 0 [
2009-10-02 10:53:00 +02:00
expr [string length $results] - [string length $prompt] - 1
2009-09-20 13:07:04 +02:00
]
]
2009-09-26 11:00:02 +02:00
if {$out == -1 && [string length $results] > 0} {
2010-11-05 21:11:57 +01:00
fail "ERROR Unexpected output from bash command \"$title\""
2010-06-18 17:21:38 +02:00
}
2009-09-20 13:07:04 +02:00
2009-06-09 22:49:53 +02:00
set cmd "echo $?"
send "$cmd\r"
expect {
2009-07-17 16:15:02 +02:00
-ex "$cmd\r\n0\r\n$prompt" {}
2010-11-05 21:11:57 +01:00
$prompt {fail "ERROR executing bash command \"$title\""}
2010-06-18 17:21:38 +02:00
}
}
2009-06-09 22:49:53 +02:00
# Test `type ...' in bash
# Indicate "unsupported" if `type' exits with error status.
# @param string $command Command to locate
proc assert_bash_type {command} {
set test "$command should be available in bash"
set cmd "type $command &> /dev/null && echo -n 0 || echo -n 1"
send "$cmd\r"
expect "$cmd\r\n"
expect {
-ex 0 { set result true }
-ex 1 { set result false; unsupported "$test" }
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
expect "/@"
return $result
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
2010-10-31 17:51:14 +01:00
# Make sure the expected list matches the real list, as returned by executing
# the specified bash command.
# Specify `-sort' if the real list is sorted.
# @param list $expected Expected list items
# @param string $cmd Bash command to execute in order to generate real list
# items
# @param string $test Test title. Becomes "$cmd should show expected output"
# if empty string.
# @param list $args Options:
# -sort Compare list sorted. Default is unsorted
# -prompt Bash prompt. Default is `/@'
# -chunk-size N Compare list N items at a time. Default
# is 20.
proc assert_bash_list {expected cmd test {args {}}} {
array set arg [::cmdline::getoptions args {
{sort "compare list sorted"}
{prompt.arg /@ "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
}]
set prompt $arg(prompt)
2009-08-23 09:38:19 +02:00
if {$test == ""} {set test "$cmd should show expected output"}
2010-02-07 15:18:58 +01:00
if {[llength $expected] == 0} {
assert_no_output $cmd $test $prompt
2009-08-23 09:38:19 +02:00
} else {
2010-02-07 15:18:58 +01:00
send "$cmd\r"
expect -ex "$cmd\r\n"
2010-11-17 23:36:58 +01:00
if {$arg(sort)} {set bash_sort "-bash-sort"} {set bash_sort ""}
2010-10-31 17:51:14 +01:00
if {[
2010-11-17 23:36:58 +01:00
eval match_items \$expected $bash_sort -chunk-size \
\$arg(chunk-size) -end-newline -end-prompt \
-prompt \$prompt
2010-10-31 17:51:14 +01:00
]} {
2010-11-17 23:36:58 +01:00
pass "$test"
2010-02-07 15:18:58 +01:00
} else {
fail "$test"
}
}
}
2009-08-23 09:38:19 +02:00
2010-10-31 17:51:14 +01:00
# Make sure the expected list matches the real list, as returned by executing
# the specified bash command within the specified directory.
# Specify `-sort' if the real list is sorted.
# @param list $expected Expected list items
# @param string $cmd Bash command to generate real list items
# @param string $dir Directory to execute $cmd within
# @param string $test Test title. Becomes "$cmd should show expected output"
# if empty string.
# @param list $args Options:
# -sort Compare list sorted. Default is unsorted
2010-11-16 23:06:13 +01:00
# -prompt Bash prompt. Default is `/@'
2010-10-31 17:51:14 +01:00
# -chunk-size N Compare list N items at a time. Default
# is 20.
proc assert_bash_list_dir {expected cmd dir test {args {}}} {
array set arg [::cmdline::getoptions args {
2010-11-16 23:06:13 +01:00
{sort "compare list sorted"}
{prompt.arg "/@" "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
2010-10-31 17:51:14 +01:00
}]
2010-11-16 23:06:13 +01:00
set prompt $arg(prompt)
2010-10-31 17:51:14 +01:00
if {$arg(sort)} {set arg_sort "-sort"} else {set arg_sort ""}
2009-12-24 09:41:22 +01:00
assert_bash_exec "cd $dir" "" $prompt
2010-10-31 17:51:14 +01:00
assert_bash_list $expected $cmd $test $arg_sort \
-chunk-size $arg(chunk-size) -prompt $prompt
2009-12-24 09:41:22 +01:00
sync_after_int $prompt
2010-02-02 11:16:29 +02:00
assert_bash_exec {cd "$TESTDIR"}
2010-06-18 17:21:38 +02:00
}
2009-12-24 09:41:22 +01:00
2009-07-17 16:15:02 +02:00
# Make sure the expected items are returned by TAB-completing the specified
2010-11-22 22:57:00 +01:00
# command. If the number of expected items is one, expected is:
#
# $cmd<TAB>$expected[<SPACE>]
#
# SPACE is not expected if -nospace is specified.
#
# If the number of expected items is greater than one, expected is:
#
# $cmd<TAB>\n
# $expected\n
# $prompt + ($cmd - AUTO) + longest-common-prefix-of-$expected
#
# AUTO is calculated like this: If $cmd ends with non-whitespace, and
# the last argument of $cmd equals the longest-common-prefix of
# $expected, $cmd minus this argument will be expected.
#
# If the algorithm above fails, you can manually specify the CWORD to be
# subtracted from $cmd specifying `-expect-cmd-minus CWORD'. Known cases where
# this is useful are when:
# - the last whitespace is escaped, e.g. "finger foo\ " or "finger
# 'foo "
#
2010-01-29 23:23:30 +01:00
# @param list $expected Expected completions.
2009-06-09 22:49:53 +02:00
# @param string $cmd Command given to generate items
2010-11-17 23:36:58 +01:00
# @param string $test Test title
# @param list $args Options:
2010-11-22 22:57:00 +01:00
# -prompt PROMPT Bash prompt. Default is `/@'
2010-11-17 23:36:58 +01:00
# -chunk-size CHUNK-SIZE Compare list CHUNK-SIZE items at
# a time. Default is 20.
2010-11-22 22:57:00 +01:00
# -nospace Don't expect space character to be output after completion match.
# Valid only if a single completion is expected.
# -ltrim-colon-completions Left-trim completions with cword containing
# colon (:)
# -expect-cmd-minus DWORD Expect $cmd minus DWORD to be echoed.
# Expected is:
#
# $cmd<TAB>\n
# $expected\n
# $prompt + ($cmd - DWORD) + longest-common-prefix-of-$expected
#
2010-11-17 23:36:58 +01:00
proc assert_complete {expected cmd {test ""} {args {}}} {
2010-11-22 22:57:00 +01:00
set args_orig $args
2010-11-17 23:36:58 +01:00
array set arg [::cmdline::getoptions args {
{prompt.arg "/@" "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
{nospace "don't expect space after completion"}
2010-11-22 22:57:00 +01:00
{ltrim-colon-completions "left-trim completions with cword containing :"}
{expect-cmd-minus.arg "" "Expect cmd minus DWORD after prompt"}
2010-11-17 23:36:58 +01:00
}]
2010-01-29 23:23:30 +01:00
if {[llength $expected] == 0} {
assert_no_complete $cmd $test
2010-11-22 22:57:00 +01:00
} elseif {[llength $expected] == 1} {
eval assert_complete_one \$expected \$cmd \$test $args_orig
2009-06-14 12:18:24 +02:00
} else {
2010-11-22 22:57:00 +01:00
eval assert_complete_many \$expected \$cmd \$test $args_orig
}
}
2010-01-29 23:23:30 +01:00
2009-11-07 09:57:11 +01:00
2010-11-22 22:57:00 +01:00
# Make sure the expected multiple items are returned by TAB-completing the
# specified command.
# @see assert_complete()
proc assert_complete_many {expected cmd {test ""} {args {}}} {
array set arg [::cmdline::getoptions args {
{prompt.arg "/@" "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
{nospace "don't expect space after completion"}
{ltrim-colon-completions "left-trim completions with cword containing :"}
{expect-cmd-minus.arg "" "Expect cmd minus CWORD after prompt"}
}]
if {$test == ""} {set test "$cmd should show completions"}
set prompt $arg(prompt)
set dword ""
if {$arg(expect-cmd-minus) != ""} {set dword $arg(expect-cmd-minus)}
send "$cmd\t"
expect -ex "$cmd\r\n"
# Make sure expected items are unique
set expected [lsort -unique $expected]
# Determine common prefix of completions
set common [::textutil::string::longestCommonPrefixList $expected]
if {$arg(ltrim-colon-completions)} {
# If partial contains colon (:), remove partial from begin of items
_ltrim_colon_completions $cmd expected dword
}
set cmd2 [_remove_cword_from_cmd $cmd $dword $common]
set prompt "$prompt$cmd2$common"
if {$arg(nospace)} {set endspace ""} else {set endspace "-end-space"}
set endprompt "-end-prompt"
if {[
eval match_items \$expected -bash-sort -chunk-size \
\$arg(chunk-size) $endprompt $endspace -prompt \$prompt
]} {
pass "$test"
} else {
fail "$test"
}
}
# Make sure the expected single item is returned by TAB-completing the
# specified command.
# @see assert_complete()
proc assert_complete_one {expected cmd {test ""} {args {}}} {
array set arg [::cmdline::getoptions args {
{prompt.arg "/@" "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
{nospace "don't expect space after completion"}
2010-11-23 09:17:53 +02:00
{ltrim-colon-completions "left-trim completions with cword containing :"}
2010-11-22 22:57:00 +01:00
{expect-cmd-minus.arg "" "Expect cmd minus CWORD after prompt"}
}]
set prompt $arg(prompt)
if {$test == ""} {set test "$cmd should show completion"}
send "$cmd\t"
expect -ex "$cmd"
2010-11-23 09:17:53 +02:00
if {$arg(ltrim-colon-completions)} {
2010-11-22 22:57:00 +01:00
# If partial contains colon (:), remove partial from begin of items
2010-11-23 09:17:53 +02:00
_ltrim_colon_completions $cmd expected cword
2010-11-22 22:57:00 +01:00
} else {
set cur ""; # Default to empty word to complete on
set words [split_words_bash $cmd]
if {[llength $words] > 1} {
# Assume last word of `$cmd' is word to complete on.
set index [expr [llength $words] - 1]
set cur [lindex $words $index]
}
# Remove second word from beginning of $expected
if {[string first $cur $expected] == 0} {
set expected [list [string range $expected [string length $cur] end]]
2010-06-18 17:21:38 +02:00
}
}
2010-11-22 22:57:00 +01:00
if {$arg(nospace)} {set endspace ""} else {set endspace "-end-space"}
if {[
eval match_items \$expected -bash-sort -chunk-size \
\$arg(chunk-size) $endspace -prompt \$prompt
]} {
pass "$test"
} else {
fail "$test"
}
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
2010-11-22 22:57:00 +01:00
# @param string $cmd Command to remove current-word-to-complete from.
# @param string $dword (optional) Manually specify current-word-to-complete,
# i.e. word to remove from $cmd. If empty string (default),
# `_remove_cword_from_cmd' autodetects if the last argument is the
# current-word-to-complete by checking if $cmd doesn't end with whitespace.
# Specifying `dword' is only necessary if this autodetection fails, e.g.
2009-11-07 09:57:11 +01:00
# when the last whitespace is escaped or quoted, e.g. "finger foo\ " or
# "finger 'foo "
2010-11-22 22:57:00 +01:00
# @param string $common (optional) Common prefix of expected completions.
# @return string Command with current-word-to-complete removed
proc _remove_cword_from_cmd {cmd {dword ""} {common ""}} {
2009-11-07 09:57:11 +01:00
set cmd2 $cmd
2010-11-22 22:57:00 +01:00
# Is $dword specified?
if {[string length $dword] > 0} {
# Remove $dword from end of $cmd
if {[string last $dword $cmd] == [string length $cmd] - [string length $dword]} {
set cmd2 [string range $cmd 0 [expr [string last $dword $cmd] - 1]]
2010-06-18 17:21:38 +02:00
}
2009-11-07 09:57:11 +01:00
} else {
2010-11-22 22:57:00 +01:00
# No, $dword not specified;
# Check if last argument is really a word-to-complete, i.e.
2009-11-07 09:57:11 +01:00
# doesn't end with whitespace.
# NOTE: This check fails if trailing whitespace is escaped or quoted,
# e.g. "finger foo\ " or "finger 'foo ". Specify parameter
2010-11-22 22:57:00 +01:00
# $dword in those cases.
2009-11-07 09:57:11 +01:00
# Is last char whitespace?
if {! [string is space [string range $cmd end end]]} {
# No, last char isn't whitespace;
2010-11-22 22:57:00 +01:00
set cmds [split $cmd]
# Does word-to-complete start with $common?
if {[string first $common [lrange $cmds end end]] == 0} {
# Remove word-to-complete from end of $cmd
set cmd2 [lrange $cmds 0 end-1]
append cmd2 " "
}
2010-06-18 17:21:38 +02:00
}
}
2009-11-07 09:57:11 +01:00
return $cmd2
2010-06-18 17:21:38 +02:00
}
2009-11-07 09:57:11 +01:00
2010-01-29 23:23:30 +01:00
# Escape regexp special characters
proc _escape_regexp_chars {var} {
upvar $var str
regsub -all {([\^$+*?.|(){}[\]\\])} $str {\\\1} str
}
2009-06-14 16:22:25 +02:00
# Make sure any completions are returned
proc assert_complete_any {cmd {test ""} {prompt /@}} {
if {$test == ""} {set test "$cmd should show completions"}
send "$cmd\t"
expect -ex "$cmd"
2010-01-29 23:23:30 +01:00
_escape_regexp_chars cmd
2009-06-14 16:22:25 +02:00
expect {
2009-07-18 16:55:23 +02:00
-timeout 1
# Match completions, multiple words
2009-07-17 23:15:15 +02:00
# NOTE: The `\S*' (zero or more non-whitespace characters) matches a
2009-07-18 16:55:23 +02:00
# longest common prefix of the completions shown.
2009-07-17 23:15:15 +02:00
# E.g. `fmt -' becomes `fmt --' (two dashes) when completing
-re "^\r\n.*$prompt$cmd\\S*$" { pass "$test" }
2009-07-18 16:55:23 +02:00
timeout {
expect {
# Match completion, single word. This word is shown on the
# same line as the command.
-re "^\\w+ $" { pass "$test" }
# Try matching multiple words again, with new timeout
-re "^\r\n.*$prompt$cmd\\S*$" { pass "$test" }
}
}
2009-07-17 16:15:02 +02:00
-re $prompt { unresolved "$test at prompt" }
2009-07-18 16:55:23 +02:00
eof { unresolved "eof" }
2010-06-18 17:21:38 +02:00
}
}
2009-06-14 16:22:25 +02:00
2009-07-19 14:48:55 +02:00
# Make sure the expected files are returned by TAB-completing the
# specified command in the specified subdirectory.
# @param list $expected
# @param string $cmd Command given to generate items
# @param string $dir Subdirectory to attempt completion in. The directory must be relative from the $TESTDIR and without a trailing slash. E.g. `fixtures/evince'
2010-11-17 23:36:58 +01:00
# @param string $test Test title
# @param list $args See: assert_complete()
2009-07-19 14:48:55 +02:00
# @result boolean True if successful, False if not
2010-11-17 23:36:58 +01:00
proc assert_complete_dir {expected cmd dir {test ""} {args {}}} {
2010-11-12 23:35:36 +01:00
set prompt "/@"
2009-07-19 14:48:55 +02:00
assert_bash_exec "cd $dir" "" $prompt
2010-11-17 23:36:58 +01:00
assert_complete $expected $cmd $test $args
2009-07-19 14:48:55 +02:00
sync_after_int $prompt
2010-02-02 11:16:29 +02:00
assert_bash_exec {cd "$TESTDIR"}
2010-06-18 17:21:38 +02:00
}
2009-07-19 14:48:55 +02:00
# Make sure a partial argument is completed.
# A completion is tried with `$partial', or if this is empty, the first
# character of the first item of `$expected'. Only the items from $expected,
# starting with this character are then expected as completions.
# @param list $expected List of all completions.
# @param string $cmd Command given to generate items
2009-11-07 09:57:11 +01:00
# @param string $partial Word to complete
2010-11-17 23:36:58 +01:00
# @param string $test Test title
# @param list $args See: assert_complete()
proc assert_complete_partial {expected cmd {partial ""} {test ""} {args {}}} {
2009-07-19 14:48:55 +02:00
if {$test == ""} {set test "$cmd should complete partial argument"}
if {[llength $expected] == 0} {
unresolved "$test"
} else {
set pick {}
2009-11-29 14:40:46 +01:00
# Make sure expected items are unique
set expected [lsort -unique $expected]
2009-07-19 14:48:55 +02:00
foreach item $expected {
if {$partial == ""} {set partial [string range $item 0 0]}
2010-11-22 22:57:00 +01:00
# Only append item if starting with $partial
2009-07-19 14:48:55 +02:00
if {[string range $item 0 [expr [string length $partial] - 1]] == "$partial"} {
lappend pick $item
2010-06-18 17:21:38 +02:00
}
}
2010-11-22 22:57:00 +01:00
# NOTE: The `eval' is necessary to flatten the $args list
# See also: http://wiki.tcl.tk/11787 - {expand}
eval assert_complete \$pick \"\$cmd \$partial\" \$test $args; #"
2010-06-18 17:21:38 +02:00
}
}
2009-07-19 14:48:55 +02:00
2010-11-22 22:57:00 +01:00
# If cword contains colon (:), left-trim completions with cword
# @param string $cmd Command to complete
# @param list $items Reference to list of completions to trim
# @param string $dword Reference to variable to contain word to remove from
# expected cmd.
2009-12-02 21:12:40 +01:00
# See also: bash_completion._ltrim_colon_completions
2010-11-22 22:57:00 +01:00
proc _ltrim_colon_completions {cmd items dword} {
2009-12-02 21:12:40 +01:00
upvar 1 $items items_out
2010-11-22 22:57:00 +01:00
upvar 1 $dword dword_out
set cur ""; # Default to empty word to complete on
set words [split_words_bash $cmd]
if {[llength $words] > 1} {
# Assume last word of `$cmd' is word to complete on.
set index [expr [llength $words] - 1]
set cur [lindex $words $index]
}
2009-12-02 21:12:40 +01:00
# If word-to-complete contains a colon,
2011-04-21 12:20:59 +03:00
# and COMP_WORDBREAKS contains a colon
2009-12-02 21:12:40 +01:00
if {
2011-04-21 12:20:59 +03:00
[string first : $cur] > -1 && [string first ":" $::COMP_WORDBREAKS] > -1
2009-12-02 21:12:40 +01:00
} {
2010-11-22 22:57:00 +01:00
set dword_out $cur
2009-12-02 21:12:40 +01:00
for {set i 0} {$i < [llength $items_out]} {incr i} {
set item [lindex $items_out $i]
2010-11-22 22:57:00 +01:00
if {[string first $cur $item] == 0} {
2009-12-02 21:12:40 +01:00
# Strip colon-prefix
2010-11-22 22:57:00 +01:00
lset items_out $i [string range $item [string length $cur] end]
2010-06-18 17:21:38 +02:00
}
}
}
}
2009-12-02 21:12:40 +01:00
2009-06-09 22:49:53 +02:00
# Make sure the bash environment hasn't changed between now and the last call
# to `save_env()'.
# @param string $sed Sed commands to preprocess diff output.
2009-07-17 16:15:02 +02:00
# Example calls:
#
# # Replace `COMP_PATH=.*' with `COMP_PATH=PATH'
# assert_env_unmodified {s/COMP_PATH=.*/COMP_PATH=PATH/}
#
2009-07-31 12:20:51 +02:00
# # Remove lines containing `OLDPWD='
2009-07-17 16:15:02 +02:00
# assert_env_unmodified {/OLDPWD=/d}
#
2009-06-09 22:49:53 +02:00
# @param string $file Filename to generate environment save file from. See
# `gen_env_filename()'.
# @param string $diff Expected diff output (after being processed by $sed)
# @see save_env()
proc assert_env_unmodified {{sed ""} {file ""} {diff ""}} {
set test "Environment should not be modified"
_save_env [gen_env_filename $file 2]
# Prepare sed script
# Escape special bash characters ("\)
2009-09-20 13:07:04 +02:00
regsub -all {([\"\\])} $sed {\\\1} sed; #"# (fix Vim syntax highlighting)
2009-06-09 22:49:53 +02:00
# Escape newlines
regsub -all {\n} [string trim $sed] "\r\n" sed
# Prepare diff script
# If diff is filled, escape newlines and make sure it ends with a newline
if {[string length [string trim $diff]]} {
regsub -all {\n} [string trim $diff] "\r\n" diff
append diff "\r\n"
} else {
set diff ""
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Execute diff
2009-09-16 22:14:53 +02:00
# NOTE: The dummy argument 'LAST-ARG' sets bash variable $_ (last argument) to
# 'LAST-ARG' so that $_ doesn't mess up the diff (as it would if $_
# was the (possibly multi-lined) sed script).
set cmd "diff_env \"[gen_env_filename $file 1]\" \"[gen_env_filename $file 2]\" \"$sed\" LAST-ARG"
2009-06-09 22:49:53 +02:00
send "$cmd\r"
2009-09-16 22:14:53 +02:00
expect "LAST-ARG\r\n"
2009-06-09 22:49:53 +02:00
expect {
-re "^$diff[wd]@$" { pass "$test" }
-re [wd]@ {
fail "$test"
# Show diff to user
set diff $expect_out(buffer)
# Remove possible `\r\n[wd]@' from end of diff
if {[string last "\r\n[wd]@" $diff] == [string length $diff] - [string length "\r\n[wd]@"]} {
set diff [string range $diff 0 [expr [string last "\r\n[wd]@" $diff] - 1]]
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
send_user $diff;
}
2010-06-18 17:21:38 +02:00
}
}
2009-06-09 22:49:53 +02:00
# Make sure the specified command executed from within Tcl/Expect.
# Fail the test with status UNSUPPORTED if Tcl fails with error "POSIX/ENOENT
2010-01-30 14:56:39 +02:00
# (No such file or directory)", or with the given Tcl failure status command
# (default "unresolved") if other error occurs.
2009-06-09 22:49:53 +02:00
# NOTE: Further tests are assumed if executing the command is successful. The
# test isn't immediately declared to have PASSED if the command is
# executed successful.
# @param string $command
# @param string $stdout (optional) Reference to variable to hold stdout.
2009-12-10 23:53:22 +02:00
# @param string $test (optional) Test title
2010-01-30 14:56:39 +02:00
# @param string $failcmd (optional, default "unresolved") Failure command
2009-06-09 22:49:53 +02:00
# @see assert_bash_exec()
2010-01-30 14:56:39 +02:00
proc assert_exec {cmd {stdout ''} {test ''} {failcmd "unresolved"}} {
2010-01-14 20:19:35 +02:00
if {$test == ""} {set test "$cmd should execute successfully"}
2009-06-09 22:49:53 +02:00
upvar $stdout results
set status [catch {eval exec $cmd} results]
if {$status == 0} {
set result true
} else {
set result false
# Command not found (POSIX/ENOENT = no such file or directory)?
if {[lindex $::errorCode 0] == "POSIX" && [lindex $::errorCode 1] == "ENOENT"} {
# Yes, command not found;
# Indicate test is unsupported
unsupported "$test"
} else {
2010-01-30 14:56:39 +02:00
$failcmd "$test"
2010-06-18 17:21:38 +02:00
}
}
2009-06-09 22:49:53 +02:00
return $result
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
2010-01-29 23:23:30 +01:00
# Check that no completion is attempted on a certain command.
# Params:
# @cmd The command to attempt to complete.
# @test Optional parameter with test name.
proc assert_no_complete {{cmd} {test ""}} {
if {[string length $test] == 0} {
set test "$cmd shouldn't complete"
2010-06-18 17:21:38 +02:00
}
2010-01-29 23:23:30 +01:00
send "$cmd\t"
expect -ex "$cmd"
# We can't anchor on $, simulate typing a magical string instead.
set endguard "Magic End Guard"
send "$endguard"
expect {
-re "^$endguard$" { pass "$test" }
default { fail "$test" }
timeout { fail "$test" }
2010-06-18 17:21:38 +02:00
}
}
2010-01-29 23:23:30 +01:00
2010-02-07 15:18:58 +01:00
# Check that no output is generated on a certain command.
# @param string $cmd The command to attempt to complete.
# @param string $test Optional parameter with test name.
# @param string $prompt (optional) Bash prompt. Default is "/@"
proc assert_no_output {{cmd} {test ""} {prompt /@}} {
if {[string length $test] == 0} {
set test "$cmd shouldn't generate output"
}
send "$cmd\r"
expect -ex "$cmd"
expect {
-re "^\r\n$prompt$" { pass "$test" }
default { fail "$test" }
timeout { fail "$test" }
}
}
2010-02-05 14:35:45 +01:00
# Source/run file with additional tests if completion for the specified command
# is installed in bash.
# @param string $command Command to check completion availability for.
# @param string $file (optional) File to source/run. Default is
# "lib/completions/$cmd.exp".
proc assert_source_completions {command {file ""}} {
2010-04-22 22:22:50 +02:00
if {[is_bash_completion_installed_for $command]} {
2010-02-05 14:35:45 +01:00
if {[string length $file] == 0} {
2010-11-12 23:35:36 +01:00
set file "$::srcdir/lib/completions/$command.exp"
2010-02-05 14:35:45 +01:00
}
source $file
} else {
untested $command
}
2010-04-22 22:22:50 +02:00
}
2010-02-05 14:35:45 +01:00
2009-12-30 14:48:17 +01:00
# Sort list.
# `exec sort' is used instead of `lsort' to achieve exactly the
# same sort order as in bash.
# @param list $items
# @return list Sort list
proc bash_sort {items} {
return [split [exec sort << [join $items "\n"]] "\n"]
2010-04-22 22:22:50 +02:00
}
2009-12-30 14:48:17 +01:00
2010-01-24 10:32:41 +01:00
# Get 'known' hostnames. Looks also in ssh's 'known_hosts' files.
2010-02-05 08:45:44 +01:00
# @param string cword (optional) Word, hosts should start with.
2010-01-24 10:32:41 +01:00
# @return list Hostnames
# @see get_hosts()
2010-02-05 08:45:44 +01:00
proc get_known_hosts {{cword ''}} {
assert_bash_exec "_known_hosts_real '$cword'; echo_array COMPREPLY" \
{} /@ result
2010-01-24 10:32:41 +01:00
return $result
2010-06-18 17:21:38 +02:00
}
2010-01-24 10:32:41 +01:00
2009-07-19 14:48:55 +02:00
# Get hostnames
2011-03-27 23:12:14 +02:00
# @param list $args Options:
2011-03-29 23:27:51 +03:00
# -unsorted Do not sort unique. Default is sort unique.
2009-06-19 14:56:36 +02:00
# @return list Hostnames
2010-01-24 10:32:41 +01:00
# @see get_known_hosts()
2011-03-27 23:12:14 +02:00
proc get_hosts {{args {}}} {
array set arg [::cmdline::getoptions args {
{unsorted "do not sort unique"}
}]
set sort "| sort -u"
if {$arg(unsorted)} {set sort ""}
set hosts [exec bash -c "compgen -A hostname $sort"]
2009-08-18 21:28:43 +02:00
# NOTE: Circumventing var `avahi_hosts' and appending directly to `hosts'
2009-07-17 23:15:15 +02:00
# causes an empty element to be inserted in `hosts'.
# -- FVu, Fri Jul 17 23:11:46 CEST 2009
2009-08-18 21:28:43 +02:00
set avahi_hosts [get_hosts_avahi]
2009-07-31 12:20:51 +02:00
if {[llength $avahi_hosts] > 0} {
2009-07-17 23:15:15 +02:00
lappend hosts $avahi_hosts
2010-06-18 17:21:38 +02:00
}
2009-11-25 22:31:29 +01:00
return $hosts
2010-06-18 17:21:38 +02:00
}
2009-06-19 14:56:36 +02:00
2009-08-18 21:28:43 +02:00
# Get hostnames according to avahi
# @return list Hostnames
proc get_hosts_avahi {} {
2009-09-13 18:05:58 +02:00
# Retrieving hosts is successful?
if { [catch {exec bash -c {
2009-11-25 21:47:58 +01:00
type avahi-browse >&/dev/null \
2009-12-15 23:48:10 +02:00
&& avahi-browse -cpr _workstation._tcp 2>/dev/null | command grep ^= | cut -d\; -f7 | sort -u
2009-09-13 18:05:58 +02:00
}} hosts] } {
# No, retrieving hosts yields error;
# Reset hosts
set hosts {}
2010-06-18 17:21:38 +02:00
}
2009-08-18 21:28:43 +02:00
return $hosts
2010-06-18 17:21:38 +02:00
}
2009-08-18 21:28:43 +02:00
2009-07-19 14:48:55 +02:00
# Get signals
# This function is written in analogy to the bash function `_signals()' in
# `bash_completion'.
2011-10-26 21:01:18 +03:00
# @param prefix
2009-07-19 14:48:55 +02:00
# @return list Signals starting with `SIG', but with the `SIG' prefix removed.
2011-10-26 21:01:18 +03:00
proc get_signals {{prefix ""}} {
2009-07-19 14:48:55 +02:00
set signals {}
foreach signal [exec bash -c {compgen -A signal}] {
# Does signal start with `SIG'?
if {[string range $signal 0 [expr [string length "SIG"] - 1]] == "SIG"} {
# Remove `SIG' prefix
set signal [string range $signal 3 end]
# Add signal (with dash (-) prefix) to list
2011-10-26 21:01:18 +03:00
lappend signals $prefix$signal
2010-06-18 17:21:38 +02:00
}
}
2009-07-19 14:48:55 +02:00
return $signals
2010-06-18 17:21:38 +02:00
}
2009-07-19 14:48:55 +02:00
2009-12-05 14:53:47 +01:00
# Initialize tcl globals with bash variables
proc init_tcl_bash_globals {} {
2010-10-18 22:29:59 +02:00
global BASH_VERSINFO BASH_VERSION COMP_WORDBREAKS LC_CTYPE
2009-12-05 14:53:47 +01:00
assert_bash_exec {printf "%s" "$COMP_WORDBREAKS"} {} /@ COMP_WORDBREAKS
assert_bash_exec {printf "%s " "${BASH_VERSINFO[@]}"} "" /@ BASH_VERSINFO
set BASH_VERSINFO [eval list $BASH_VERSINFO]
assert_bash_exec {printf "%s" "$BASH_VERSION"} "" /@ BASH_VERSION
assert_bash_exec {printf "%s" "$TESTDIR"} "" /@ TESTDIR
2010-10-18 22:29:59 +02:00
assert_bash_exec {eval $(locale); printf "%s" "$LC_CTYPE"} "" /@ LC_CTYPE
2010-06-18 17:21:38 +02:00
}
2009-12-05 14:53:47 +01:00
2010-02-05 14:35:45 +01:00
# Check whether completion is installed for the specified command by executing
# `complete -p ...' in bash.
# @param string $command Command to check completion availability for.
# @return boolean True (1) if completion is installed, False (0) if not.
proc is_bash_completion_installed_for {command} {
set test "$command should have completion installed in bash"
set cmd "complete -p $command &> /dev/null && echo -n 0 || echo -n 1"
send "$cmd\r"
expect "$cmd\r\n"
expect {
-ex 0 { set result true }
-ex 1 { set result false }
}
expect "/@"
return $result
2010-06-18 17:21:38 +02:00
}
2010-02-05 14:35:45 +01:00
2009-12-30 14:48:17 +01:00
# Detect if test suite is running under Cygwin/Windows
proc is_cygwin {} {
expr {[string first [string tolower [exec uname -s]] cygwin] >= 0}
2010-06-18 17:21:38 +02:00
}
2009-12-30 14:48:17 +01:00
2010-10-31 17:51:14 +01:00
# Expect items, a limited number (20) at a time.
2009-06-09 22:49:53 +02:00
# Break items into chunks because `expect' seems to have a limited buffer size
2010-10-31 17:51:14 +01:00
# @param list $items Expected list items
# @param list $args Options:
# -bash-sort Compare list bash-sorted. Default is
# unsorted
# -prompt PROMPT Bash prompt. Default is `/@'
# -chunk-size CHUNK-SIZE Compare list CHUNK-SIZE items at
# a time. Default is 20.
2010-11-17 23:36:58 +01:00
# -end-newline Expect newline after last item.
# Default is not.
# -end-prompt Expect prompt after last item.
# Default is not.
# -end-space Expect single space after last item.
# Default is not. Valid only if
# `end-newline' not set.
2009-06-09 22:49:53 +02:00
# @result boolean True if successful, False if not
2010-10-31 17:51:14 +01:00
proc match_items {items {args {}}} {
array set arg [::cmdline::getoptions args {
{bash-sort "compare list sorted"}
{prompt.arg "/@" "bash prompt"}
{chunk-size.arg 20 "compare N list items at a time"}
2010-11-17 23:36:58 +01:00
{end-newline "expect newline after last item"}
{end-prompt "expect prompt after last item"}
{end-space "expect space ater last item"}
2010-10-31 17:51:14 +01:00
}]
set prompt $arg(prompt)
set size $arg(chunk-size)
if {$arg(bash-sort)} {set items [bash_sort $items]}
2009-06-09 22:49:53 +02:00
set result false
for {set i 0} {$i < [llength $items]} {set i [expr {$i + $size}]} {
2010-01-16 12:56:29 +02:00
# For chunks > 1, allow leading whitespace
if {$i > $size} { set expected "\\s*" } else { set expected "" }
2009-06-09 22:49:53 +02:00
for {set j 0} {$j < $size && $i + $j < [llength $items]} {incr j} {
set item "[lindex $items [expr {$i + $j}]]"
2010-01-29 23:23:30 +01:00
_escape_regexp_chars item
2009-06-14 12:18:24 +02:00
append expected $item
2010-11-16 23:06:13 +01:00
if {[llength $items] > 1} {append expected {\s+}}
2010-06-18 17:21:38 +02:00
}
2009-06-19 14:23:57 +02:00
if {[llength $items] == 1} {
2010-11-17 23:36:58 +01:00
if {$arg(end-prompt)} {set end $prompt} {set end ""}
# Both trailing space and newline are specified?
if {$arg(end-newline) && $arg(end-space)} {
# Indicate both trailing space or newline are ok
set expected2 "|^$expected $end$"; # Include space
append expected "\r\n$end"; # Include newline
} else {
if {$arg(end-newline)} {append expected "\r\n$end"}
if {$arg(end-space)} {append expected " $end"}
set expected2 ""
}
2009-06-19 14:23:57 +02:00
expect {
2010-11-17 23:36:58 +01:00
-re "^$expected$$expected2" { set result true }
2009-12-24 09:41:22 +01:00
-re "^$prompt$" {set result false; break }
2009-06-19 14:23:57 +02:00
default { set result false; break }
timeout { set result false; break }
2010-06-18 17:21:38 +02:00
}
2009-06-19 14:23:57 +02:00
} else {
2010-11-22 22:57:00 +01:00
set end ""
if {$arg(end-prompt) && $i + $j == [llength $items]} {
set end "$prompt"
_escape_regexp_chars end
# \$ matches real end of expect_out buffer
set end "$end\$"
}
2009-06-19 14:23:57 +02:00
expect {
2010-11-22 22:57:00 +01:00
-re "^$expected$end" { set result true }
2009-06-19 14:23:57 +02:00
default { set result false; break }
timeout { set result false; break }
2010-06-18 17:21:38 +02:00
}
}
}
2009-06-09 22:49:53 +02:00
return $result
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Get real command.
# - arg: $1 Command
# - return: Command found, empty string if not found
proc realcommand {cmd} {
set result ""
if [string length [set path [auto_execok $cmd]]] {
if {[string length [auto_execok realpath]]} {
set result [exec realpath $path]
} elseif {[string length [auto_execok readlink]]} {
set result [exec readlink -f $path]
} else {
set result $path
2010-06-18 17:21:38 +02:00
}
}
2009-06-09 22:49:53 +02:00
return $result
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Generate filename to save environment to.
# @param string $file File-basename to save environment to. If the file has a
# `.exp' suffix, it is removed. E.g.:
# - "file.exp" becomes "file.env1~"
# - "" becomes "env.env1~"
# - "filename" becomes "filename.env1~"
# The file will be stored in the $TESTDIR/tmp directory.
# @param integer $seq Sequence number. Must be either 1 or 2.
proc gen_env_filename {{file ""} {seq 1}} {
if {[string length $file] == 0} {
set file "env"
} else {
# Remove possible directories
set file [file tail $file]
# Remove possible '.exp' suffix from filename
if {[string last ".exp" $file] == [string length $file] - [string length ".exp"]} {
set file [string range $file 0 [expr [string last ".exp" $file] - 1]]
2010-06-18 17:21:38 +02:00
}
}
2009-06-09 22:49:53 +02:00
return "\$TESTDIR/tmp/$file.env$seq~"
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Save the environment for later comparison
# @param string $file Filename to generate environment save file from. See
# `gen_env_filename()'.
proc save_env {{file ""}} {
_save_env [gen_env_filename $file 1]
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Save the environment for later comparison
# @param string File to save the environment to. Default is "$TESTDIR/tmp/env1~".
# @see assert_env_unmodified()
proc _save_env {{file ""}} {
2010-02-02 11:16:29 +02:00
assert_bash_exec "{ set; declare -F; shopt -p; } > \"$file\""
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
2009-12-05 14:53:47 +01:00
# Source bash_completion package
proc source_bash_completion {} {
2011-05-02 18:41:36 +02:00
assert_bash_exec {BASH_COMPLETION_COMPAT_DIR=$(cd "$SRCDIR/.."; pwd)/completions}
2011-05-03 09:33:18 +03:00
assert_bash_exec {source $(cd "$SRCDIR/.."; pwd)/bash_completion}
2010-06-18 17:21:38 +02:00
}
2009-12-05 14:53:47 +01:00
2010-01-29 23:23:30 +01:00
# Split line into words, disregarding backslash escapes (e.g. \b (backspace),
# \g (bell)), but taking backslashed spaces into account.
# Aimed for simulating bash word splitting.
# Example usage:
#
# % set a {f cd\ \be}
# % split_words $a
# f {cd\ \be}
#
# @param string Line to split
# @return list Words
proc split_words_bash {line} {
set words {}
set glue false
foreach part [split $line] {
set glue_next false
# Does `part' end with a backslash (\)?
if {[string last "\\" $part] == [string length $part] - [string length "\\"]} {
# Remove end backslash
set part [string range $part 0 [expr [string length $part] - [string length "\\"] - 1]]
# Indicate glue on next run
set glue_next true
2010-06-18 17:21:38 +02:00
}
2010-01-29 23:23:30 +01:00
# Must `part' be appended to latest word (= glue)?
if {[llength $words] > 0 && [string is true $glue]} {
# Yes, join `part' to latest word;
set zz [lindex $words [expr [llength $words] - 1]]
# Separate glue with backslash-space (\ );
lset words [expr [llength $words] - 1] "$zz\\ $part"
} else {
# No, don't append word to latest word;
# Append `part' as separate word
lappend words $part
2010-06-18 17:21:38 +02:00
}
2010-01-29 23:23:30 +01:00
set glue $glue_next
2010-06-18 17:21:38 +02:00
}
2010-01-29 23:23:30 +01:00
return $words
2010-06-18 17:21:38 +02:00
}
2010-01-29 23:23:30 +01:00
2010-02-03 14:08:55 +02:00
# Given a list of items this proc finds a (part, full) pair so that when
# completing from $part $full will be the only option.
#
# Arguments:
# list The list of full completions.
# partName Output parameter for the partial string.
# fullName Output parameter for the full string, member of item.
#
# Results:
# 1, or 0 if no suitable result was found.
proc find_unique_completion_pair {{list} {partName} {fullName}} {
upvar $partName part
upvar $fullName full
set bestscore 0
set list [lsort $list]
set n [llength $list]
for {set i 0} {$i < $n} {incr i} {
set cur [lindex $list $i]
set curlen [string length $cur]
set prev [lindex $list [expr {$i - 1}]]
set next [lindex $list [expr {$i + 1}]]
set diffprev [expr {$prev == ""}]
set diffnext [expr {$next == ""}]
# Analyse each item of the list and look for the minimum length of the
# partial prefix which is distinct from both $next and $prev. The list
# is sorted so the prefix will be unique in the entire list.
#
# In the worst case we analyse every character in the list 3 times.
# That's actually very fast, sorting could take more.
for {set j 0} {$j < $curlen} {incr j} {
set curchar [string index $cur $j]
if {!$diffprev && [string index $prev $j] != $curchar} {
set diffprev 1
}
if {!$diffnext && [string index $next $j] != $curchar} {
set diffnext 1
}
if {$diffnext && $diffprev} {
break
}
}
# At the end of the loop $j is the index of last character of
# the unique partial prefix. The length is one plus that.
set parlen [expr {$j + 1}]
if {$parlen >= $curlen} {
continue
}
# Try to find the most "readable pair"; look for a long pair where
# $part is about half of $full.
if {$parlen < $curlen / 2} {
set parlen [expr {$curlen / 2}]
}
set score [expr {$curlen - $parlen}]
if {$score > $bestscore} {
set bestscore $score
set part [string range $cur 0 [expr {$parlen - 1}]]
set full $cur
}
}
return [expr {$bestscore != 0}]
}
2009-12-05 14:53:47 +01:00
# Start bash running as test environment.
proc start_bash {} {
2010-11-12 23:35:36 +01:00
global TESTDIR TOOL_EXECUTABLE spawn_id env srcdirabs
2009-12-05 14:53:47 +01:00
set TESTDIR [pwd]
2010-11-12 23:35:36 +01:00
set srcdirabs [file normalize $::srcdir]; # Absolute srcdir
2009-12-05 14:53:47 +01:00
# If `--tool_exec' option not specified, use "bash"
if {! [info exists TOOL_EXECUTABLE]} {set TOOL_EXECUTABLE bash}
2010-11-12 23:35:36 +01:00
set env(SRCDIR) $::srcdir
set env(SRCDIRABS) $::srcdirabs
2010-11-05 21:28:48 +01:00
exp_spawn $TOOL_EXECUTABLE --rcfile $::srcdir/config/bashrc
assert_bash_exec {} "$TOOL_EXECUTABLE --rcfile $::srcdir/config/bashrc"
2010-06-18 17:21:38 +02:00
}
2009-12-05 14:53:47 +01:00
2010-02-20 22:53:30 +02:00
# Redirect xtrace output to a file.
#
# 'set -x' can be very useful for debugging but by default it writes to
2011-04-21 12:20:59 +03:00
# stderr.
2010-02-20 22:53:30 +02:00
#
# This function uses file descriptor 6. This will break if any completion
# tries to use the same descriptor.
proc init_bash_xtrace {{fname xtrace.log}} {
verbose "Enabling bash xtrace output to '$fname'"
assert_bash_exec "exec 6>'$fname'"
assert_bash_exec "BASH_XTRACEFD=6"
assert_bash_exec "set -o xtrace"
}
# Setup test environment
#
# Common initialization for unit and completion tests.
proc start_interactive_test {} {
start_bash
source_bash_completion
init_tcl_bash_globals
global OPT_BASH_XTRACE
if {[info exists OPT_BASH_XTRACE]} {
init_bash_xtrace
}
2010-02-21 00:53:18 +02:00
global OPT_TIMEOUT
if {[info exists OPT_TIMEOUT]} {
global timeout
verbose "Changing default expect timeout from $timeout to $OPT_TIMEOUT"
set timeout $OPT_TIMEOUT
}
2010-02-20 22:53:30 +02:00
}
2009-06-09 22:49:53 +02:00
# Interrupt completion and sync with prompt.
# Send signals QUIT & INT.
2009-07-17 16:15:02 +02:00
# @param string $prompt (optional) Bash prompt. Default is "/@"
proc sync_after_int {{prompt /@}} {
2009-06-09 22:49:53 +02:00
set test "Sync after INT"
sleep .1
send \031\003; # QUIT/INT
2010-02-11 23:09:46 +01:00
# Wait to allow bash to become ready
# See also: http://lists.alioth.debian.org/pipermail/bash-completion-devel/
# 2010-February/002566.html
sleep .1
2010-02-11 23:12:04 +01:00
# NOTE: Regexp `.*' causes `expect' to discard previous unknown output.
# This is necessary if a completion doesn't match expectations.
# For instance with `filetype_xspec' completion (e.g. `kdvi') if
# one expects `.txt' as a completion (wrong, because it isn't
# there), the unmatched completions need to be cleaned up.
2009-07-19 14:48:55 +02:00
expect -re ".*$prompt$"
2010-02-11 23:12:04 +01:00
}
2009-06-09 22:49:53 +02:00
proc sync_after_tab {} {
# NOTE: Wait in case completion returns nothing - because `units' isn't
# installed, so that "^$cdm.*$" doesn't match too early - before
# comp_install has finished
sleep .4
2010-06-18 17:21:38 +02:00
}
2009-06-09 22:49:53 +02:00
# Return current working directory with `TESTDIR' stripped
# @return string Working directory. E.g. /, or /fixtures/
proc wd {} {
global TESTDIR
# Remove `$TESTDIR' prefix from current working directory
set wd [string replace [pwd] 0 [expr [string length $TESTDIR] - 1]]/
2010-06-18 17:21:38 +02:00
}