| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | require "download_strategy" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe AbstractDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version, **specs) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 10:53:49 +00:00
										 |  |  |   let(:specs) { {} } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:name) { "foo" } | 
					
						
							|  |  |  |   let(:url) { "http://example.com/foo.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:args) { %w[foo bar baz] } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   specify "#source_modified_time" do | 
					
						
							| 
									
										
										
										
											2018-07-13 14:42:49 +01:00
										 |  |  |     Mktemp.new("mtime") do | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |       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 | 
					
						
							| 
									
										
										
										
											2018-03-24 10:53:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |       expect(subject).not_to be_a_kind_of(AbstractDownloadStrategy::Pourable) | 
					
						
							| 
									
										
										
										
											2018-03-24 10:53:49 +00:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe VCSDownloadStrategy do | 
					
						
							|  |  |  |   let(:url) { "http://example.com/bar" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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") | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |       downloader = described_class.new(url, "baz", version) | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |       expect(downloader.cached_location).to eq(HOMEBREW_CACHE/"baz--foo") | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe GitHubPrivateRepositoryDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, "foo", version) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:url) { "https://github.com/owner/repo/archive/1.1.5.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |   before do | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     ENV["HOMEBREW_GITHUB_API_TOKEN"] = "token" | 
					
						
							|  |  |  |     allow(GitHub).to receive(:repository).and_return({}) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it "sets the @github_token instance variable" do | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@github_token)).to eq("token") | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it "parses the URL and sets the corresponding instance variables" do | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@owner)).to eq("owner") | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@repo)).to eq("repo") | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@filepath)).to eq("archive/1.1.5.tar.gz") | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   its(:download_url) { is_expected.to eq("https://token@github.com/owner/repo/archive/1.1.5.tar.gz") } | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe GitHubPrivateRepositoryReleaseDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, "foo", version) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:url) { "https://github.com/owner/repo/releases/download/tag/foo_v0.1.0_darwin_amd64.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |   before do | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     ENV["HOMEBREW_GITHUB_API_TOKEN"] = "token" | 
					
						
							|  |  |  |     allow(GitHub).to receive(:repository).and_return({}) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it "parses the URL and sets the corresponding instance variables" do | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@owner)).to eq("owner") | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@repo)).to eq("repo") | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@tag)).to eq("tag") | 
					
						
							|  |  |  |     expect(subject.instance_variable_get(:@filename)).to eq("foo_v0.1.0_darwin_amd64.tar.gz") | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#download_url" do | 
					
						
							|  |  |  |     it "returns the download URL for a given resource" do | 
					
						
							|  |  |  |       allow(subject).to receive(:resolve_asset_id).and_return(456) | 
					
						
							|  |  |  |       expect(subject.download_url).to eq("https://token@api.github.com/repos/owner/repo/releases/assets/456") | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   specify "#resolve_asset_id" do | 
					
						
							|  |  |  |     release_metadata = { | 
					
						
							|  |  |  |       "assets" => [ | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           "id" => 123, | 
					
						
							|  |  |  |           "name" => "foo_v0.1.0_linux_amd64.tar.gz", | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           "id" => 456, | 
					
						
							|  |  |  |           "name" => "foo_v0.1.0_darwin_amd64.tar.gz", | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     allow(subject).to receive(:fetch_release_metadata).and_return(release_metadata) | 
					
						
							|  |  |  |     expect(subject.send(:resolve_asset_id)).to eq(456) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#fetch_release_metadata" do | 
					
						
							|  |  |  |     it "fetches release metadata from GitHub" do | 
					
						
							|  |  |  |       expected_release_url = "https://api.github.com/repos/owner/repo/releases/tags/tag" | 
					
						
							| 
									
										
										
										
											2018-03-07 16:14:55 +00:00
										 |  |  |       expect(GitHub).to receive(:open_api).with(expected_release_url).and_return({}) | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |       subject.send(:fetch_release_metadata) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe GitHubGitDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:name) { "brew" } | 
					
						
							|  |  |  |   let(:url) { "https://github.com/homebrew/brew.git" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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 | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:name) { "baz" } | 
					
						
							|  |  |  |   let(:url) { "https://github.com/homebrew/foo" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   let(:cached_location) { subject.cached_location } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |   before do | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     @commit_id = 1
 | 
					
						
							|  |  |  |     FileUtils.mkpath cached_location | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def git_commit_all | 
					
						
							| 
									
										
										
										
											2017-07-29 19:55:05 +02:00
										 |  |  |     system "git", "add", "--all" | 
					
						
							|  |  |  |     system "git", "commit", "-m", "commit number #{@commit_id}" | 
					
						
							|  |  |  |     @commit_id += 1
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def setup_git_repo | 
					
						
							| 
									
										
										
										
											2017-07-29 19:55:05 +02:00
										 |  |  |     system "git", "init" | 
					
						
							|  |  |  |     system "git", "remote", "add", "origin", "https://github.com/Homebrew/homebrew-foo" | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     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" } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |     before { remote_repo.mkpath } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |     after { FileUtils.rm_rf remote_repo } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  | describe S3DownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |   let(:name) { "foo" } | 
					
						
							|  |  |  |   let(:url) { "http://bucket.s3.amazonaws.com/foo.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   describe "#_fetch" do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |     subject { described_class.new(url, name, version)._fetch } | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     context "when given Bad S3 URL" do | 
					
						
							|  |  |  |       let(:url) { "http://example.com/foo.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       it "raises Bad S3 URL error" do | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |         expect { | 
					
						
							|  |  |  |           subject._fetch | 
					
						
							|  |  |  |         }.to raise_error(RuntimeError) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 11:13:25 -06:00
										 |  |  | describe CurlDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version, **specs) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 11:13:25 -06:00
										 |  |  |   let(:name) { "foo" } | 
					
						
							|  |  |  |   let(:url) { "http://example.com/foo.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-06 00:16:57 +02:00
										 |  |  |   let(:version) { "1.2.3" } | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   let(:specs) { { user: "download:123456" } } | 
					
						
							| 
									
										
										
										
											2017-08-21 11:13:25 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   it "parses the opts and sets the corresponding args" do | 
					
						
							| 
									
										
										
										
											2018-08-02 11:16:36 +02:00
										 |  |  |     expect(subject.send(:_curl_args)).to eq(["--user", "download:123456"]) | 
					
						
							| 
									
										
										
										
											2017-08-21 11:13:25 -06:00
										 |  |  |   end | 
					
						
							| 
									
										
										
										
											2018-01-09 19:56:54 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 10:51:01 +02:00
										 |  |  |   describe "#cached_location" do | 
					
						
							|  |  |  |     subject { described_class.new(url, name, version, **specs).cached_location } | 
					
						
							| 
									
										
										
										
											2018-01-09 19:56:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     context "when URL ends with file" do | 
					
						
							| 
									
										
										
										
											2018-08-06 00:16:57 +02:00
										 |  |  |       it { is_expected.to eq(HOMEBREW_CACHE/"foo--1.2.3.tar.gz") } | 
					
						
							| 
									
										
										
										
											2018-01-09 19:56:54 -06:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when URL file is in middle" do | 
					
						
							|  |  |  |       let(:url) { "http://example.com/foo.tar.gz/from/this/mirror" } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 00:16:57 +02:00
										 |  |  |       it { is_expected.to eq(HOMEBREW_CACHE/"foo--1.2.3.tar.gz") } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#fetch" do | 
					
						
							|  |  |  |     before(:each) do | 
					
						
							|  |  |  |       FileUtils.touch subject.temporary_path | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it "calls curl with default arguments" do | 
					
						
							|  |  |  |       expect(subject).to receive(:curl).with( | 
					
						
							|  |  |  |         "--location", | 
					
						
							|  |  |  |         "--remote-time", | 
					
						
							|  |  |  |         "--continue-at", "-", | 
					
						
							|  |  |  |         "--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: "kie", | 
					
						
							|  |  |  |             mon: "ster", | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it "adds the appropriate curl args" do | 
					
						
							|  |  |  |         expect(subject).to receive(:system_command!) { |*, args:, **| | 
					
						
							|  |  |  |           expect(args.each_cons(2)).to include(["-b", "coo=kie;mon=ster"]) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "with referer set" do | 
					
						
							|  |  |  |       let(:specs) { { referer: "http://somehost/also" } } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it "adds the appropriate curl args" do | 
					
						
							|  |  |  |         expect(subject).to receive(:system_command!) { |*, args:, **| | 
					
						
							|  |  |  |           expect(args.each_cons(2)).to include(["-e", "http://somehost/also"]) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#cached_location" do | 
					
						
							|  |  |  |     context "with a file name trailing the URL path" do | 
					
						
							|  |  |  |       let(:url) { "http://example.com/cask.dmg" } | 
					
						
							|  |  |  |       its("cached_location.extname") { is_expected.to eq(".dmg") } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "with no discernible file name in it" do | 
					
						
							|  |  |  |       let(:url) { "http://example.com/download" } | 
					
						
							|  |  |  |       its("cached_location.basename.to_path") { is_expected.to eq("foo--1.2.3") } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "with a file name trailing the first query parameter" do | 
					
						
							|  |  |  |       let(:url) { "http://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) { "http://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) { "http://example.com/foo.tar.gz" } | 
					
						
							|  |  |  |   let(:version) { "1.2.3" } | 
					
						
							|  |  |  |   let(:specs) { {} } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#fetch" do | 
					
						
							|  |  |  |     before(:each) do | 
					
						
							|  |  |  |       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 | 
					
						
							| 
									
										
										
										
											2018-01-09 19:56:54 -06:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2017-08-21 11:13:25 -06:00
										 |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  | describe ScpDownloadStrategy do | 
					
						
							| 
									
										
										
										
											2018-08-02 10:29:40 +02:00
										 |  |  |   subject { described_class.new(url, name, version) } | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |   let(:name) { "foo" } | 
					
						
							|  |  |  |   let(:url) { "scp://example.com/foo.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-08-01 05:18:00 +02:00
										 |  |  |   let(:version) { nil } | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   describe "#initialize" do | 
					
						
							|  |  |  |     invalid_urls = %w[
 | 
					
						
							|  |  |  |       http://example.com/foo.tar.gz | 
					
						
							|  |  |  |       scp://@example.com/foo.tar.gz | 
					
						
							|  |  |  |       scp://example.com:/foo.tar.gz | 
					
						
							|  |  |  |       scp://example.com | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     invalid_urls.each do |invalid_url| | 
					
						
							|  |  |  |       context "with invalid URL #{invalid_url}" do | 
					
						
							| 
									
										
										
										
											2018-08-02 09:59:22 +02:00
										 |  |  |         let(:url) { invalid_url } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |         it "raises ScpDownloadStrategyError" do | 
					
						
							| 
									
										
										
										
											2018-08-02 09:59:22 +02:00
										 |  |  |           expect { subject }.to raise_error(ScpDownloadStrategyError) | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe "#fetch" do | 
					
						
							|  |  |  |     before do | 
					
						
							|  |  |  |       expect(subject.temporary_path).to receive(:rename).and_return(true) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when given a valid URL" do | 
					
						
							|  |  |  |       let(:url) { "scp://example.com/foo.tar.gz" } | 
					
						
							|  |  |  |       it "copies the file via scp" do | 
					
						
							|  |  |  |         expect(subject) | 
					
						
							| 
									
										
										
										
											2018-08-04 11:32:36 +02:00
										 |  |  |           .to receive(:system_command!) | 
					
						
							|  |  |  |           .with("scp", args: ["example.com:/foo.tar.gz", anything]) | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |           .and_return(true) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when given a URL with a username" do | 
					
						
							|  |  |  |       let(:url) { "scp://user@example.com/foo.tar.gz" } | 
					
						
							|  |  |  |       it "copies the file via scp" do | 
					
						
							|  |  |  |         expect(subject) | 
					
						
							| 
									
										
										
										
											2018-08-04 11:32:36 +02:00
										 |  |  |           .to receive(:system_command!) | 
					
						
							|  |  |  |           .with("scp", args: ["user@example.com:/foo.tar.gz", anything]) | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |           .and_return(true) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when given a URL with a port" do | 
					
						
							|  |  |  |       let(:url) { "scp://example.com:1234/foo.tar.gz" } | 
					
						
							|  |  |  |       it "copies the file via scp" do | 
					
						
							|  |  |  |         expect(subject) | 
					
						
							| 
									
										
										
										
											2018-08-04 11:32:36 +02:00
										 |  |  |           .to receive(:system_command!) | 
					
						
							|  |  |  |           .with("scp", args: ["-P 1234 example.com:/foo.tar.gz", anything]) | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |           .and_return(true) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when given a URL with /~/" do | 
					
						
							|  |  |  |       let(:url) { "scp://example.com/~/foo.tar.gz" } | 
					
						
							|  |  |  |       it "treats the path as relative to the home directory" do | 
					
						
							|  |  |  |         expect(subject) | 
					
						
							| 
									
										
										
										
											2018-08-04 11:32:36 +02:00
										 |  |  |           .to receive(:system_command!) | 
					
						
							|  |  |  |           .with("scp", args: ["example.com:~/foo.tar.gz", anything]) | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |           .and_return(true) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subject.fetch | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 00:16:57 +02:00
										 |  |  | describe SubversionDownloadStrategy do | 
					
						
							|  |  |  |   subject { described_class.new(url, name, version, **specs) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let(:name) { "foo" } | 
					
						
							|  |  |  |   let(:url) { "http://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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | describe DownloadStrategyDetector do | 
					
						
							|  |  |  |   describe "::detect" do | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |     subject { described_class.detect(url, strategy) } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     let(:url) { Object.new } | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |     let(:strategy) { nil } | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     context "when given Git URL" do | 
					
						
							|  |  |  |       let(:url) { "git://example.com/foo.git" } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |       it { is_expected.to eq(GitDownloadStrategy) } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     context "when given a GitHub Git URL" do | 
					
						
							|  |  |  |       let(:url) { "https://github.com/homebrew/brew.git" } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |       it { is_expected.to eq(GitHubGitDownloadStrategy) } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |     context "when given an S3 URL" do | 
					
						
							|  |  |  |       let(:url) { "s3://bucket/homebrew/brew.tar.gz" } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |       it "returns S3DownloadStrategy" do | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |         allow(described_class).to receive(:require_aws_sdk).and_return(true) | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |         is_expected.to eq(S3DownloadStrategy) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |     context "when given strategy = S3DownloadStrategy" do | 
					
						
							|  |  |  |       let(:url) { "https://bkt.s3.amazonaws.com/key.tar.gz" } | 
					
						
							|  |  |  |       let(:strategy) { S3DownloadStrategy } | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |       it "requires aws-sdk-s3" do | 
					
						
							| 
									
										
										
										
											2018-03-25 13:30:37 +01:00
										 |  |  |         allow(described_class).to receive(:require_aws_sdk).and_return(true) | 
					
						
							| 
									
										
										
										
											2018-03-20 16:46:00 -04:00
										 |  |  |         is_expected.to eq(S3DownloadStrategy) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 11:38:39 -07:00
										 |  |  |     context "when given an scp URL" do | 
					
						
							|  |  |  |       let(:url) { "scp://example.com/brew.tar.gz" } | 
					
						
							|  |  |  |       it { is_expected.to eq(ScpDownloadStrategy) } | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 23:53:06 +01:00
										 |  |  |     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 |