brew/Library/Homebrew/test/extend/kernel_spec.rb

302 lines
8.6 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
RSpec.describe "globally-scoped helper methods" do
2017-02-28 14:50:46 +01:00
let(:dir) { mktmpdir }
2017-02-16 21:17:46 +01:00
def esc(code)
/(\e\[\d+m)*\e\[#{code}m/
end
describe "#ofail" do
it "sets Homebrew.failed to true" do
expect do
ofail "foo"
end.to output("Error: foo\n").to_stderr
2017-02-16 21:17:46 +01:00
expect(Homebrew).to have_failed
2017-02-16 21:17:46 +01:00
end
end
describe "#odie" do
it "exits with 1" do
expect do
2017-02-16 21:17:46 +01:00
odie "foo"
2023-03-14 08:50:56 -07:00
end.to output("Error: foo\n").to_stderr.and raise_error SystemExit
2017-02-16 21:17:46 +01:00
end
end
describe "#pretty_installed" do
subject(:pretty_installed_output) { pretty_installed("foo") }
2017-02-16 21:17:46 +01:00
context "when $stdout is a TTY" do
before { allow($stdout).to receive(:tty?).and_return(true) }
2017-02-16 21:17:46 +01:00
context "with HOMEBREW_NO_EMOJI unset" do
it "returns a string with a colored checkmark" do
expect(pretty_installed_output)
2017-02-16 21:17:46 +01:00
.to match(/#{esc 1}foo #{esc 32}#{esc 0}/)
end
end
context "with HOMEBREW_NO_EMOJI set" do
before { ENV["HOMEBREW_NO_EMOJI"] = "1" }
2017-02-16 21:17:46 +01:00
it "returns a string with colored info" do
expect(pretty_installed_output)
2017-02-16 21:17:46 +01:00
.to match(/#{esc 1}foo \(installed\)#{esc 0}/)
end
end
end
context "when $stdout is not a TTY" do
before { allow($stdout).to receive(:tty?).and_return(false) }
2017-02-16 21:17:46 +01:00
it "returns plain text" do
expect(pretty_installed_output).to eq("foo")
2017-02-16 21:17:46 +01:00
end
end
end
describe "#pretty_uninstalled" do
subject(:pretty_uninstalled_output) { pretty_uninstalled("foo") }
2017-02-16 21:17:46 +01:00
context "when $stdout is a TTY" do
before { allow($stdout).to receive(:tty?).and_return(true) }
2017-02-16 21:17:46 +01:00
context "with HOMEBREW_NO_EMOJI unset" do
it "returns a string with a colored checkmark" do
expect(pretty_uninstalled_output)
2017-02-16 21:17:46 +01:00
.to match(/#{esc 1}foo #{esc 31}#{esc 0}/)
end
end
context "with HOMEBREW_NO_EMOJI set" do
before { ENV["HOMEBREW_NO_EMOJI"] = "1" }
2017-02-16 21:17:46 +01:00
it "returns a string with colored info" do
expect(pretty_uninstalled_output)
2017-02-16 21:17:46 +01:00
.to match(/#{esc 1}foo \(uninstalled\)#{esc 0}/)
end
end
end
context "when $stdout is not a TTY" do
before { allow($stdout).to receive(:tty?).and_return(false) }
2017-02-16 21:17:46 +01:00
it "returns plain text" do
expect(pretty_uninstalled_output).to eq("foo")
2017-02-16 21:17:46 +01:00
end
end
end
describe "#interactive_shell" do
let(:shell) { dir/"myshell" }
it "starts an interactive shell session" do
File.write shell, <<~SH
2017-02-16 21:17:46 +01:00
#!/bin/sh
echo called > "#{dir}/called"
2018-07-11 15:17:40 +02:00
SH
2017-02-16 21:17:46 +01:00
FileUtils.chmod 0755, shell
ENV["SHELL"] = shell
expect { interactive_shell }.not_to raise_error
expect(dir/"called").to exist
end
end
describe "#with_custom_locale" do
it "temporarily overrides the system locale" do
ENV["LC_ALL"] = "en_US.UTF-8"
with_custom_locale("C") do
2022-05-30 04:37:09 +01:00
expect(ENV.fetch("LC_ALL")).to eq("C")
2017-02-16 21:17:46 +01:00
end
2022-05-30 04:37:09 +01:00
expect(ENV.fetch("LC_ALL")).to eq("en_US.UTF-8")
2017-02-16 21:17:46 +01:00
end
end
describe "#which" do
let(:cmd) { dir/"foo" }
before { FileUtils.touch cmd }
2017-02-16 21:17:46 +01:00
it "returns the first executable that is found" do
cmd.chmod 0744
expect(which(File.basename(cmd), File.dirname(cmd))).to eq(cmd)
end
it "skips non-executables" do
expect(which(File.basename(cmd), File.dirname(cmd))).to be_nil
2017-02-16 21:17:46 +01:00
end
it "skips malformed path and doesn't fail" do
# 'which' should not fail if a path is malformed
# see https://github.com/Homebrew/legacy-homebrew/issues/32789 for an example
cmd.chmod 0744
# ~~ will fail because ~foo resolves to foo's home and there is no '~' user
path = ["~~", File.dirname(cmd)].join(File::PATH_SEPARATOR)
expect(which(File.basename(cmd), path)).to eq(cmd)
end
end
describe "#which_all" do
2023-04-20 12:14:45 +00:00
let(:cmd_foo) { dir/"foo" }
let(:cmd_foo_bar) { dir/"bar/foo" }
let(:cmd_bar_baz_foo) { dir/"bar/baz/foo" }
2017-02-16 21:17:46 +01:00
before do
2017-02-16 21:17:46 +01:00
(dir/"bar/baz").mkpath
2023-04-20 12:14:45 +00:00
FileUtils.touch cmd_foo_bar
2017-02-16 21:17:46 +01:00
2023-04-20 12:14:45 +00:00
[cmd_foo, cmd_bar_baz_foo].each do |cmd|
2017-02-16 21:17:46 +01:00
FileUtils.touch cmd
cmd.chmod 0744
end
end
it "returns an array of all executables that are found" do
path = [
"#{dir}/bar/baz",
"#{dir}/baz:#{dir}",
"~baduserpath",
].join(File::PATH_SEPARATOR)
2023-04-20 12:14:45 +00:00
expect(which_all("foo", path)).to eq([cmd_bar_baz_foo, cmd_foo])
2017-02-16 21:17:46 +01:00
end
end
specify "#which_editor" do
ENV["HOMEBREW_EDITOR"] = "vemate -w"
ENV["HOMEBREW_PATH"] = dir
editor = "#{dir}/vemate"
FileUtils.touch editor
FileUtils.chmod 0755, editor
expect(which_editor).to eq("vemate -w")
2017-02-16 21:17:46 +01:00
end
describe "#pretty_duration" do
it "converts seconds to a human-readable string" do
expect(pretty_duration(1)).to eq("1 second")
expect(pretty_duration(2.5)).to eq("2 seconds")
expect(pretty_duration(42)).to eq("42 seconds")
expect(pretty_duration(240)).to eq("4 minutes")
expect(pretty_duration(252.45)).to eq("4 minutes 12 seconds")
end
end
specify "#disk_usage_readable" do
expect(disk_usage_readable(1)).to eq("1B")
expect(disk_usage_readable(1000)).to eq("1000B")
expect(disk_usage_readable(1024)).to eq("1KB")
expect(disk_usage_readable(1025)).to eq("1KB")
expect(disk_usage_readable(4_404_020)).to eq("4.2MB")
expect(disk_usage_readable(4_509_715_660)).to eq("4.2GB")
2017-02-16 21:17:46 +01:00
end
describe "#number_readable" do
it "returns a string with thousands separators" do
expect(number_readable(1)).to eq("1")
expect(number_readable(1_000)).to eq("1,000")
expect(number_readable(1_000_000)).to eq("1,000,000")
end
end
specify "#truncate_text_to_approximate_size" do
glue = "\n[...snip...]\n" # hard-coded copy from truncate_text_to_approximate_size
n = 20
long_s = "x" * 40
s = truncate_text_to_approximate_size(long_s, n)
expect(s.length).to eq(n)
expect(s).to match(/^x+#{Regexp.escape(glue)}x+$/)
s = truncate_text_to_approximate_size(long_s, n, front_weight: 0.0)
expect(s).to eq(glue + ("x" * (n - glue.length)))
s = truncate_text_to_approximate_size(long_s, n, front_weight: 1.0)
expect(s).to eq(("x" * (n - glue.length)) + glue)
end
describe "#odeprecated" do
2018-08-15 12:13:21 +02:00
it "raises a MethodDeprecatedError when `disable` is true" do
2017-02-16 21:17:46 +01:00
ENV.delete("HOMEBREW_DEVELOPER")
expect do
2017-02-16 21:17:46 +01:00
odeprecated(
"method", "replacement",
2018-11-02 17:18:07 +00:00
caller: ["#{HOMEBREW_LIBRARY}/Taps/homebrew/homebrew-core/"],
2017-02-16 21:17:46 +01:00
disable: true
)
end.to raise_error(
MethodDeprecatedError,
2020-06-02 09:49:23 +01:00
%r{method.*replacement.*homebrew/core.*/Taps/homebrew/homebrew-core/}m,
)
2017-02-16 21:17:46 +01:00
end
end
describe "#with_env" do
it "sets environment variables within the block" do
expect(ENV.fetch("PATH")).not_to eq("/bin")
with_env(PATH: "/bin") do
expect(ENV.fetch("PATH", nil)).to eq("/bin")
end
end
it "restores ENV after the block" do
with_env(PATH: "/bin") do
expect(ENV.fetch("PATH", nil)).to eq("/bin")
end
path = ENV.fetch("PATH", nil)
expect(path).not_to be_nil
expect(path).not_to eq("/bin")
end
it "restores ENV if an exception is raised" do
expect do
with_env(PATH: "/bin") do
raise StandardError, "boom"
end
end.to raise_error(StandardError)
path = ENV.fetch("PATH", nil)
expect(path).not_to be_nil
expect(path).not_to eq("/bin")
end
end
describe "#tap_and_name_comparison" do
describe "both strings are only names" do
it "alphabetizes the strings" do
expect(%w[a b].sort(&tap_and_name_comparison)).to eq(%w[a b])
expect(%w[b a].sort(&tap_and_name_comparison)).to eq(%w[a b])
end
end
describe "both strings include tap" do
it "alphabetizes the strings" do
expect(%w[a/z/z b/z/z].sort(&tap_and_name_comparison)).to eq(%w[a/z/z b/z/z])
expect(%w[b/z/z a/z/z].sort(&tap_and_name_comparison)).to eq(%w[a/z/z b/z/z])
expect(%w[z/a/z z/b/z].sort(&tap_and_name_comparison)).to eq(%w[z/a/z z/b/z])
expect(%w[z/b/z z/a/z].sort(&tap_and_name_comparison)).to eq(%w[z/a/z z/b/z])
expect(%w[z/z/a z/z/b].sort(&tap_and_name_comparison)).to eq(%w[z/z/a z/z/b])
expect(%w[z/z/b z/z/a].sort(&tap_and_name_comparison)).to eq(%w[z/z/a z/z/b])
end
end
describe "only one string includes tap" do
it "prefers the string without tap" do
expect(%w[a/z/z z].sort(&tap_and_name_comparison)).to eq(%w[z a/z/z])
expect(%w[z a/z/z].sort(&tap_and_name_comparison)).to eq(%w[z a/z/z])
end
end
end
2017-02-16 21:17:46 +01:00
end