mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
418 lines
11 KiB
Ruby
418 lines
11 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require "download_strategy"
|
|
|
|
describe AbstractDownloadStrategy do
|
|
subject { described_class.new(url, name, version, **specs) }
|
|
|
|
let(:specs) { {} }
|
|
let(:name) { "foo" }
|
|
let(:url) { "https://example.com/foo.tar.gz" }
|
|
let(:version) { nil }
|
|
let(:args) { %w[foo bar baz] }
|
|
|
|
specify "#source_modified_time" do
|
|
Mktemp.new("mtime") do
|
|
FileUtils.touch "foo", mtime: Time.now - 10
|
|
FileUtils.touch "bar", mtime: Time.now - 100
|
|
FileUtils.ln_s "not-exist", "baz"
|
|
expect(subject.source_modified_time).to eq(File.mtime("foo"))
|
|
end
|
|
end
|
|
|
|
context "when specs[:bottle]" do
|
|
let(:specs) { { bottle: true } }
|
|
|
|
it "extends Pourable" do
|
|
expect(subject).to be_a_kind_of(AbstractDownloadStrategy::Pourable)
|
|
end
|
|
end
|
|
|
|
context "without specs[:bottle]" do
|
|
it "is does not extend Pourable" do
|
|
expect(subject).not_to be_a_kind_of(AbstractDownloadStrategy::Pourable)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe VCSDownloadStrategy do
|
|
let(:url) { "https://example.com/bar" }
|
|
let(:version) { nil }
|
|
|
|
describe "#cached_location" do
|
|
it "returns the path of the cached resource" do
|
|
allow_any_instance_of(described_class).to receive(:cache_tag).and_return("foo")
|
|
downloader = described_class.new(url, "baz", version)
|
|
expect(downloader.cached_location).to eq(HOMEBREW_CACHE/"baz--foo")
|
|
end
|
|
end
|
|
end
|
|
|
|
describe GitHubGitDownloadStrategy do
|
|
subject { described_class.new(url, name, version) }
|
|
|
|
let(:name) { "brew" }
|
|
let(:url) { "https://github.com/homebrew/brew.git" }
|
|
let(:version) { nil }
|
|
|
|
it "parses the URL and sets the corresponding instance variables" do
|
|
expect(subject.instance_variable_get(:@user)).to eq("homebrew")
|
|
expect(subject.instance_variable_get(:@repo)).to eq("brew")
|
|
end
|
|
end
|
|
|
|
describe GitDownloadStrategy do
|
|
subject { described_class.new(url, name, version) }
|
|
|
|
let(:name) { "baz" }
|
|
let(:url) { "https://github.com/homebrew/foo" }
|
|
let(:version) { nil }
|
|
let(:cached_location) { subject.cached_location }
|
|
|
|
before do
|
|
@commit_id = 1
|
|
FileUtils.mkpath cached_location
|
|
end
|
|
|
|
def git_commit_all
|
|
system "git", "add", "--all"
|
|
system "git", "commit", "-m", "commit number #{@commit_id}"
|
|
@commit_id += 1
|
|
end
|
|
|
|
def setup_git_repo
|
|
system "git", "init"
|
|
system "git", "remote", "add", "origin", "https://github.com/Homebrew/homebrew-foo"
|
|
FileUtils.touch "README"
|
|
git_commit_all
|
|
end
|
|
|
|
describe "#source_modified_time" do
|
|
it "returns the right modification time" do
|
|
cached_location.cd do
|
|
setup_git_repo
|
|
end
|
|
expect(subject.source_modified_time.to_i).to eq(1_485_115_153)
|
|
end
|
|
end
|
|
|
|
specify "#last_commit" do
|
|
cached_location.cd do
|
|
setup_git_repo
|
|
FileUtils.touch "LICENSE"
|
|
git_commit_all
|
|
end
|
|
expect(subject.last_commit).to eq("f68266e")
|
|
end
|
|
|
|
describe "#fetch_last_commit" do
|
|
let(:url) { "file://#{remote_repo}" }
|
|
let(:version) { Version.create("HEAD") }
|
|
let(:remote_repo) { HOMEBREW_PREFIX/"remote_repo" }
|
|
|
|
before { remote_repo.mkpath }
|
|
|
|
after { FileUtils.rm_rf remote_repo }
|
|
|
|
it "fetches the hash of the last commit" do
|
|
remote_repo.cd do
|
|
setup_git_repo
|
|
FileUtils.touch "LICENSE"
|
|
git_commit_all
|
|
end
|
|
|
|
subject.shutup!
|
|
expect(subject.fetch_last_commit).to eq("f68266e")
|
|
end
|
|
end
|
|
end
|
|
|
|
describe CurlDownloadStrategy do
|
|
subject { described_class.new(url, name, version, **specs) }
|
|
|
|
let(:name) { "foo" }
|
|
let(:url) { "https://example.com/foo.tar.gz" }
|
|
let(:version) { "1.2.3" }
|
|
let(:specs) { { user: "download:123456" } }
|
|
|
|
it "parses the opts and sets the corresponding args" do
|
|
expect(subject.send(:_curl_args)).to eq(["--user", "download:123456"])
|
|
end
|
|
|
|
describe "#cached_location" do
|
|
subject { described_class.new(url, name, version, **specs).cached_location }
|
|
|
|
context "when URL ends with file" do
|
|
it {
|
|
expect(subject).to eq(
|
|
HOMEBREW_CACHE/"downloads/3d1c0ae7da22be9d83fb1eb774df96b7c4da71d3cf07e1cb28555cf9a5e5af70--foo.tar.gz",
|
|
)
|
|
}
|
|
end
|
|
|
|
context "when URL file is in middle" do
|
|
let(:url) { "https://example.com/foo.tar.gz/from/this/mirror" }
|
|
|
|
it {
|
|
expect(subject).to eq(
|
|
HOMEBREW_CACHE/"downloads/1ab61269ba52c83994510b1e28dd04167a2f2e8393a35a9c50c1f7d33fd8f619--foo.tar.gz",
|
|
)
|
|
}
|
|
end
|
|
end
|
|
|
|
describe "#fetch" do
|
|
before do
|
|
subject.temporary_path.dirname.mkpath
|
|
FileUtils.touch subject.temporary_path
|
|
end
|
|
|
|
it "calls curl with default arguments" do
|
|
expect(subject).to receive(:curl).with(
|
|
"--location",
|
|
"--remote-time",
|
|
"--continue-at", "0",
|
|
"--output", an_instance_of(Pathname),
|
|
url,
|
|
an_instance_of(Hash)
|
|
)
|
|
|
|
subject.fetch
|
|
end
|
|
|
|
context "with an explicit user agent" do
|
|
let(:specs) { { user_agent: "Mozilla/25.0.1" } }
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["--user-agent", "Mozilla/25.0.1"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with a generalized fake user agent" do
|
|
alias_matcher :a_string_matching, :match
|
|
|
|
let(:specs) { { user_agent: :fake } }
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2).to_a).to include(
|
|
[
|
|
"--user-agent",
|
|
a_string_matching(/Mozilla.*Mac OS X 10.*AppleWebKit/),
|
|
],
|
|
)
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with cookies set" do
|
|
let(:specs) {
|
|
{
|
|
cookies: {
|
|
coo: "k/e",
|
|
mon: "ster",
|
|
},
|
|
}
|
|
}
|
|
|
|
it "adds the appropriate curl args and does not URL-encode the cookies" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["-b", "coo=k/e;mon=ster"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with referer set" do
|
|
let(:specs) { { referer: "https://somehost/also" } }
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["-e", "https://somehost/also"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with headers set" do
|
|
alias_matcher :a_string_matching, :match
|
|
|
|
let(:specs) { { headers: ["foo", "bar"] } }
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2).to_a).to include(["--header", "foo"])
|
|
expect(args.each_cons(2).to_a).to include(["--header", "bar"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#cached_location" do
|
|
context "with a file name trailing the URL path" do
|
|
let(:url) { "https://example.com/cask.dmg" }
|
|
|
|
its("cached_location.extname") { is_expected.to eq(".dmg") }
|
|
end
|
|
|
|
context "with a file name trailing the first query parameter" do
|
|
let(:url) { "https://example.com/download?file=cask.zip&a=1" }
|
|
|
|
its("cached_location.extname") { is_expected.to eq(".zip") }
|
|
end
|
|
|
|
context "with a file name trailing the second query parameter" do
|
|
let(:url) { "https://example.com/dl?a=1&file=cask.zip&b=2" }
|
|
|
|
its("cached_location.extname") { is_expected.to eq(".zip") }
|
|
end
|
|
|
|
context "with an unusually long query string" do
|
|
let(:url) do
|
|
[
|
|
"https://node49152.ssl.fancycdn.example.com",
|
|
"/fancycdn/node/49152/file/upload/download",
|
|
"?cask_class=zf920df",
|
|
"&cask_group=2348779087242312",
|
|
"&cask_archive_file_name=cask.zip",
|
|
"&signature=CGmDulxL8pmutKTlCleNTUY%2FyO9Xyl5u9yVZUE0",
|
|
"uWrjadjuz67Jp7zx3H7NEOhSyOhu8nzicEHRBjr3uSoOJzwkLC8L",
|
|
"BLKnz%2B2X%2Biq5m6IdwSVFcLp2Q1Hr2kR7ETn3rF1DIq5o0lHC",
|
|
"yzMmyNe5giEKJNW8WF0KXriULhzLTWLSA3ZTLCIofAdRiiGje1kN",
|
|
"YY3C0SBqymQB8CG3ONn5kj7CIGbxrDOq5xI2ZSJdIyPysSX7SLvE",
|
|
"DBw2KdR24q9t1wfjS9LUzelf5TWk6ojj8p9%2FHjl%2Fi%2FVCXN",
|
|
"N4o1mW%2FMayy2tTY1qcC%2FTmqI1ulZS8SNuaSgr9Iys9oDF1%2",
|
|
"BPK%2B4Sg==",
|
|
].join
|
|
end
|
|
|
|
its("cached_location.extname") { is_expected.to eq(".zip") }
|
|
its("cached_location.to_path.length") { is_expected.to be_between(0, 255) }
|
|
end
|
|
end
|
|
end
|
|
|
|
describe CurlPostDownloadStrategy do
|
|
subject { described_class.new(url, name, version, **specs) }
|
|
|
|
let(:name) { "foo" }
|
|
let(:url) { "https://example.com/foo.tar.gz" }
|
|
let(:version) { "1.2.3" }
|
|
let(:specs) { {} }
|
|
|
|
describe "#fetch" do
|
|
before do
|
|
subject.temporary_path.dirname.mkpath
|
|
FileUtils.touch subject.temporary_path
|
|
end
|
|
|
|
context "with :using and :data specified" do
|
|
let(:specs) {
|
|
{
|
|
using: :post,
|
|
data: {
|
|
form: "data",
|
|
is: "good",
|
|
},
|
|
}
|
|
}
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["-d", "form=data"])
|
|
expect(args.each_cons(2)).to include(["-d", "is=good"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with :using but no :data" do
|
|
let(:specs) { { using: :post } }
|
|
|
|
it "adds the appropriate curl args" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["-X", "POST"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe SubversionDownloadStrategy do
|
|
subject { described_class.new(url, name, version, **specs) }
|
|
|
|
let(:name) { "foo" }
|
|
let(:url) { "https://example.com/foo.tar.gz" }
|
|
let(:version) { "1.2.3" }
|
|
let(:specs) { {} }
|
|
|
|
describe "#fetch" do
|
|
context "with :trust_cert set" do
|
|
let(:specs) { { trust_cert: true } }
|
|
|
|
it "adds the appropriate svn args" do
|
|
expect(subject).to receive(:system_command!)
|
|
.with("svn", args: array_including("--trust-server-cert",
|
|
"--non-interactive"))
|
|
subject.fetch
|
|
end
|
|
end
|
|
|
|
context "with :revision set" do
|
|
let(:specs) { { revision: "10" } }
|
|
|
|
it "adds svn arguments for :revision" do
|
|
expect(subject).to receive(:system_command!) { |*, args:, **|
|
|
expect(args.each_cons(2)).to include(["-r", "10"])
|
|
}
|
|
|
|
subject.fetch
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe DownloadStrategyDetector do
|
|
describe "::detect" do
|
|
subject { described_class.detect(url, strategy) }
|
|
|
|
let(:url) { Object.new }
|
|
let(:strategy) { nil }
|
|
|
|
context "when given Git URL" do
|
|
let(:url) { "git://example.com/foo.git" }
|
|
|
|
it { is_expected.to eq(GitDownloadStrategy) }
|
|
end
|
|
|
|
context "when given a GitHub Git URL" do
|
|
let(:url) { "https://github.com/homebrew/brew.git" }
|
|
|
|
it { is_expected.to eq(GitHubGitDownloadStrategy) }
|
|
end
|
|
|
|
it "defaults to cURL" do
|
|
expect(subject).to eq(CurlDownloadStrategy)
|
|
end
|
|
|
|
it "raises an error when passed an unrecognized strategy" do
|
|
expect {
|
|
described_class.detect("foo", Class.new)
|
|
}.to raise_error(TypeError)
|
|
end
|
|
end
|
|
end
|