mirror of
https://github.com/ethauvin/sdkman-for-fish.git
synced 2025-04-25 13:27:10 -07:00
Migrate wrapper test to Cucumber.
This commit is contained in:
parent
479fa1e541
commit
56bc601b17
9 changed files with 144 additions and 94 deletions
|
@ -38,7 +38,6 @@ install:
|
||||||
script:
|
script:
|
||||||
- (cd test && cucumber)
|
- (cd test && cucumber)
|
||||||
# TODO: Migrate these to Cucumber:
|
# TODO: Migrate these to Cucumber:
|
||||||
- bash test/prepare_tests.sh && fish test/wrapper.fish
|
|
||||||
- fish test/reinitialize.fish
|
- fish test/reinitialize.fish
|
||||||
- fish -c "sdk install crash 1.3.0; and sdk uninstall crash 1.3.0" > /dev/null && fish test/check_for_path_zombies.fish
|
- fish -c "sdk install crash 1.3.0; and sdk uninstall crash 1.3.0" > /dev/null && fish test/check_for_path_zombies.fish
|
||||||
- bash test/remove_sdkman.sh > /dev/null && fish -c "echo 'y' | sdk" > /dev/null && fish -c "sdk version"
|
- bash test/remove_sdkman.sh > /dev/null && fish -c "echo 'y' | sdk" > /dev/null && fish -c "sdk version"
|
||||||
|
|
|
@ -23,9 +23,9 @@ RUN curl -s "https://get.sdkman.io" | bash
|
||||||
|
|
||||||
# "Install" sdkman-for-fish
|
# "Install" sdkman-for-fish
|
||||||
RUN mkdir -p $TEST_HOME/.config/fish/
|
RUN mkdir -p $TEST_HOME/.config/fish/
|
||||||
COPY completions $TEST_HOME/.config/fish/completions/
|
COPY --chown=test:test completions $TEST_HOME/.config/fish/completions/
|
||||||
COPY conf.d $TEST_HOME/.config/fish/conf.d/
|
COPY --chown=test:test conf.d $TEST_HOME/.config/fish/conf.d/
|
||||||
COPY completions $TEST_HOME/.config/fish/functions/
|
COPY --chown=test:test functions $TEST_HOME/.config/fish/functions/
|
||||||
RUN ls -R $TEST_HOME/.config/fish/
|
RUN ls -R $TEST_HOME/.config/fish/
|
||||||
|
|
||||||
# Run tests
|
# Run tests
|
||||||
|
|
|
@ -4,10 +4,9 @@ require 'open3'
|
||||||
|
|
||||||
module CompletionHelper
|
module CompletionHelper
|
||||||
def complete(cmd)
|
def complete(cmd)
|
||||||
completions = run_fish_command("complete -C\"sdk #{cmd}\"")
|
completions = run_fish_command("complete -C\"sdk #{cmd}\"")[:stdout]
|
||||||
|
|
||||||
completions.split("\n") \
|
completions.map { |line| line.split(/\s+/)[0].strip }
|
||||||
.map { |line| line.split(/\s+/)[0].strip }
|
|
||||||
# TODO: Why do we get duplicates in the Docker container?
|
# TODO: Why do we get duplicates in the Docker container?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
39
test/features/step_definitions/wrapper.rb
Normal file
39
test/features/step_definitions/wrapper.rb
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module WrapperHelper
|
||||||
|
def reject_then_select(lines, exclude, select)
|
||||||
|
lines.select do |e|
|
||||||
|
(exclude.nil? || e !~ exclude) && e =~ select
|
||||||
|
end.sort
|
||||||
|
end
|
||||||
|
|
||||||
|
def compare_env(exclude, include)
|
||||||
|
env_bash = reject_then_select(@response_bash[:env], exclude, include)
|
||||||
|
env_fish = reject_then_select(@response_fish[:env], exclude, include)
|
||||||
|
expect(env_fish).to eq(env_bash)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
World WrapperHelper
|
||||||
|
|
||||||
|
When('we run {string} in Bash and Fish') do |command|
|
||||||
|
@response_bash = run_bash_command(command)
|
||||||
|
@response_fish = run_fish_command(command)
|
||||||
|
end
|
||||||
|
|
||||||
|
Then('the exit code is the same') do
|
||||||
|
expect(@response_fish[:status]).to eq(@response_bash[:status])
|
||||||
|
end
|
||||||
|
|
||||||
|
Then('the output is the same') do
|
||||||
|
%i[stdout stderr].each do |out|
|
||||||
|
expect(@response_fish[out]).to eq(@response_bash[out])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
Then('environment variable(s) {env_glob} is/are the same') do |pattern|
|
||||||
|
compare_env(nil, pattern)
|
||||||
|
end
|
||||||
|
|
||||||
|
Then('environment variable(s) {env_glob} is/are the same except for {env_glob}') do |pattern, exclude_pattern|
|
||||||
|
compare_env(exclude_pattern, pattern)
|
||||||
|
end
|
|
@ -1,5 +1,8 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'fileutils'
|
||||||
|
require 'tmpdir'
|
||||||
|
|
||||||
def list_installed_candidates
|
def list_installed_candidates
|
||||||
candidates = {}
|
candidates = {}
|
||||||
|
|
||||||
|
@ -17,26 +20,70 @@ end
|
||||||
|
|
||||||
def installed?(candidate, version = nil)
|
def installed?(candidate, version = nil)
|
||||||
candidates = list_installed_candidates
|
candidates = list_installed_candidates
|
||||||
candidates.key?(candidate) && (version.nil? || candidates[candidate].include?(version))
|
candidates.key?(candidate) \
|
||||||
|
&& (version.nil? || candidates[candidate].include?(version))
|
||||||
end
|
end
|
||||||
|
|
||||||
def run_bash_command(cmd)
|
def run_bash_command(cmd)
|
||||||
stdout, stderr, status = Open3.capture3("bash -c 'source \"$HOME/.sdkman/bin/sdkman-init.sh\"; #{cmd}'")
|
Dir.mktmpdir(%w[sdkman-for-fish-test_ _fish]) do |tmp_dir|
|
||||||
unless status.success?
|
files = %i[status stdout stderr env].map { |s|
|
||||||
warn(stderr)
|
[s, FileUtils.touch("#{tmp_dir}/#{s}")[0]]
|
||||||
raise "Bash command failed: #{stderr}"
|
}.to_h
|
||||||
end
|
|
||||||
|
|
||||||
stdout
|
out, status = Open3.capture2e(<<~BASH
|
||||||
|
bash -c 'source "#{ENV['HOME']}/.sdkman/bin/sdkman-init.sh" && \
|
||||||
|
#{cmd} > #{files[:stdout]} 2> #{files[:stderr]}; \
|
||||||
|
echo "$?" > #{files[:status]}; \
|
||||||
|
env > #{files[:env]}; \
|
||||||
|
'
|
||||||
|
BASH
|
||||||
|
)
|
||||||
|
|
||||||
|
unless status.success?
|
||||||
|
warn(out)
|
||||||
|
raise "Bash command failed: #{out}"
|
||||||
|
end
|
||||||
|
|
||||||
|
{
|
||||||
|
status: File.read(files[:status]).to_i,
|
||||||
|
stdout: File.readlines(files[:stdout]),
|
||||||
|
stderr: File.readlines(files[:stderr]),
|
||||||
|
env: File.readlines(files[:env])
|
||||||
|
}
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# # For nicer diffs: one entry per line, sorted
|
||||||
|
# string split ":" (cat path_bash) | sort > path_bash
|
||||||
|
# string split ":" (cat path_fish) \
|
||||||
|
# | string split " " \
|
||||||
|
# | sort > path_fish
|
||||||
|
# # split by spaces for fish 2.*
|
||||||
|
|
||||||
def run_fish_command(cmd)
|
def run_fish_command(cmd)
|
||||||
# NB: Fish errors out if we don't set terminal dimensions
|
Dir.mktmpdir(%w[sdkman-for-fish-test_ _fish]) do |tmp_dir|
|
||||||
stdout, stderr, status = Open3.capture3("fish -c 'stty rows 80 columns 80; #{cmd}'")
|
files = %i[status stdout stderr env].map { |s|
|
||||||
unless status.success?
|
[s, FileUtils.touch("#{tmp_dir}/#{s}")[0]]
|
||||||
warn(stderr)
|
}.to_h
|
||||||
raise 'Fish command failed'
|
|
||||||
end
|
|
||||||
|
|
||||||
stdout
|
out, status = Open3.capture2e(<<~FISH
|
||||||
|
fish -c '#{cmd} > #{files[:stdout]} ^ #{files[:stderr]}; \
|
||||||
|
echo $status > #{files[:status]}; \
|
||||||
|
env > #{files[:env]}; \
|
||||||
|
'
|
||||||
|
FISH
|
||||||
|
)
|
||||||
|
|
||||||
|
unless status.success?
|
||||||
|
warn(out)
|
||||||
|
raise "Fish command failed: #{out}"
|
||||||
|
end
|
||||||
|
|
||||||
|
{
|
||||||
|
status: File.read(files[:status]).to_i,
|
||||||
|
stdout: File.readlines(files[:stdout]),
|
||||||
|
stderr: File.readlines(files[:stderr]),
|
||||||
|
env: File.readlines(files[:env])
|
||||||
|
}
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -19,3 +19,12 @@ ParameterType(
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
)
|
)
|
||||||
|
|
||||||
|
ParameterType(
|
||||||
|
name: 'env_glob',
|
||||||
|
regexp: /[A-Z_*]+/,
|
||||||
|
type: Regexp,
|
||||||
|
transformer: lambda do |glob|
|
||||||
|
/^#{glob.gsub('*', '[A-Z_]*')}=/
|
||||||
|
end
|
||||||
|
)
|
||||||
|
|
30
test/features/wrapper.feature
Normal file
30
test/features/wrapper.feature
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
Feature: Wrapping of Bash
|
||||||
|
All calls to sdk are performed through Bash;
|
||||||
|
we need to wrap those calls in such a way that
|
||||||
|
the effect sdk has on the Bash environment carries
|
||||||
|
over the current Fish session.
|
||||||
|
|
||||||
|
We verifiy equality of (standard) output, exit code, and environment variables.
|
||||||
|
|
||||||
|
Background:
|
||||||
|
Given SDKMAN! candidate list is up to date
|
||||||
|
And candidate ant is installed at version 1.9.9
|
||||||
|
And candidate ant is installed at version 1.10.1
|
||||||
|
|
||||||
|
Scenario Outline:
|
||||||
|
When we run "<command>" in Bash and Fish
|
||||||
|
Then the exit code is the same
|
||||||
|
And the output is the same
|
||||||
|
And environment variable PATH is the same
|
||||||
|
And environment variables *_HOME are the same
|
||||||
|
And environment variables SDKMAN_* are the same except for SDKMAN_OFFLINE_MODE
|
||||||
|
# NB: SDKMAN_OFFLINE_MODE is not an environment variable in bash, so ignore it here.
|
||||||
|
Examples:
|
||||||
|
| command |
|
||||||
|
| sdk |
|
||||||
|
| sdk version |
|
||||||
|
| sdk list java |
|
||||||
|
| sdk update |
|
||||||
|
| sdk use ant 1.9.9 |
|
||||||
|
| sdk offline enable > /dev/null; sdk install ant foo |
|
||||||
|
| sdk use ant 1.9.9 > /dev/null; sdk broadcast |
|
|
@ -1,9 +0,0 @@
|
||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
source "${HOME}"/.sdkman/bin/sdkman-init.sh
|
|
||||||
|
|
||||||
# Set up an SDK with two installed versions
|
|
||||||
# --> test of `sdk use` in wrapper.fish
|
|
||||||
sdk install ant 1.9.9
|
|
||||||
echo "y" | sdk install ant 1.10.1
|
|
||||||
sdk default ant 1.10.1
|
|
|
@ -1,64 +0,0 @@
|
||||||
# Test that a couple of commands have the same effect when run through
|
|
||||||
# the fish wrapper and directly.
|
|
||||||
# Verifies equality of (standard) output, exit code, and PATH.
|
|
||||||
|
|
||||||
set test_commands \
|
|
||||||
"sdk" \
|
|
||||||
"sdk version" \
|
|
||||||
"sdk list java" \
|
|
||||||
"sdk update" \
|
|
||||||
"sdk use ant 1.9.9" \
|
|
||||||
"sdk offline enable > /dev/null; sdk install ant foo" \
|
|
||||||
"sdk use ant 1.9.9 > /dev/null; sdk broadcast"
|
|
||||||
set test_count (count $test_commands)
|
|
||||||
set check_count (math "3 * $test_count")
|
|
||||||
|
|
||||||
set sdk_init "$HOME/.sdkman/bin/sdkman-init.sh"
|
|
||||||
|
|
||||||
if [ (uname) = "Linux" ]
|
|
||||||
function checksum -a file
|
|
||||||
sha256sum $file | cut -d " " -f 1
|
|
||||||
end
|
|
||||||
else # assume macOS
|
|
||||||
function checksum -a file
|
|
||||||
shasum -a 256 $file | cut -d " " -f 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
echo "Testing the sdk wrapper"
|
|
||||||
set failures 0
|
|
||||||
for sdk_cmd in $test_commands
|
|
||||||
echo " Testing '$sdk_cmd'"
|
|
||||||
bash -c "source \"$sdk_init\" && $sdk_cmd > sout_bash;
|
|
||||||
echo \"\$?\" > status_bash;
|
|
||||||
echo \"\$PATH\" > path_bash;
|
|
||||||
echo \"\$ANT_HOME\" > anthome_bash"
|
|
||||||
fish -c "$sdk_cmd > sout_fish;
|
|
||||||
echo \"\$status\" > status_fish;
|
|
||||||
echo \"\$PATH\" > path_fish;
|
|
||||||
echo \"\$ANT_HOME\" > anthome_fish"
|
|
||||||
|
|
||||||
# For nicer diffs: one entry per line, sorted
|
|
||||||
string split ":" (cat path_bash) | sort > path_bash
|
|
||||||
string split ":" (cat path_fish) \
|
|
||||||
| string split " " \
|
|
||||||
| sort > path_fish
|
|
||||||
# split by spaces for fish 2.*
|
|
||||||
|
|
||||||
for out in sout status path anthome
|
|
||||||
if [ (checksum "$out"_bash) != (checksum "$out"_fish) ]
|
|
||||||
echo " - $out bad:"
|
|
||||||
diff "$out"_bash "$out"_fish | sed -e 's/^/ /'
|
|
||||||
set failures (math $failures + 1)
|
|
||||||
else
|
|
||||||
echo " - $out ok!"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
echo ""
|
|
||||||
end
|
|
||||||
|
|
||||||
rm {sout,status,path}_{bash,fish}
|
|
||||||
|
|
||||||
echo "Ran $test_count commands with 3 checks each."
|
|
||||||
echo "$failures/$check_count checks failed."
|
|
||||||
exit $failures
|
|
Loading…
Add table
Add a link
Reference in a new issue