193 lines
5.9 KiB
Ruby
Raw Normal View History

require "open3"
2017-08-08 18:10:13 +02:00
def curl_executable
@curl ||= [
ENV["HOMEBREW_CURL"],
which("curl"),
"/usr/bin/curl",
2018-03-17 16:46:44 -07:00
].compact.map { |c| Pathname(c) }.find(&:executable?)
raise "no executable curl was found" unless @curl
2018-09-17 02:45:00 +02:00
@curl
2017-08-08 18:10:13 +02:00
end
2017-08-08 18:10:13 +02:00
def curl_args(*extra_args, show_output: false, user_agent: :default)
2018-07-30 10:11:00 +02:00
args = []
2018-04-08 15:51:58 -07:00
# do not load .curlrc unless requested (must be the first argument)
args << "-q" unless ENV["HOMEBREW_CURLRC"]
2018-04-08 15:51:58 -07:00
args << "--show-error"
2017-08-08 18:10:13 +02:00
args << "--user-agent" << case user_agent
when :browser, :fake
HOMEBREW_USER_AGENT_FAKE_SAFARI
when :default
HOMEBREW_USER_AGENT_CURL
else
2017-08-08 18:10:13 +02:00
user_agent
end
2017-08-08 18:10:13 +02:00
unless show_output
args << "--fail"
args << "--progress-bar" unless ARGV.verbose?
args << "--verbose" if ENV["HOMEBREW_CURL_VERBOSE"]
args << "--silent" if !$stdout.tty? || ENV["HOMEBREW_TRAVIS_CI"]
end
2017-08-08 18:10:13 +02:00
args + extra_args
end
def curl(*args)
2017-11-03 18:58:59 +00:00
# SSL_CERT_FILE can be incorrectly set by users or portable-ruby and screw
# with SSL downloads so unset it here.
2018-07-30 10:11:00 +02:00
system_command! curl_executable,
args: curl_args(*args),
2018-08-01 11:15:42 +02:00
print_stdout: true,
2018-07-30 10:11:00 +02:00
env: { "SSL_CERT_FILE" => nil }
2017-08-04 16:24:29 +02:00
end
def curl_download(*args, to: nil, **options)
destination = Pathname(to)
destination.dirname.mkpath
continue_at = if destination.exist? &&
curl_output("--location", "--head", "--range", "0-1",
"--write-out", "%{http_code}",
"--output", "/dev/null", *args, **options).stdout.to_i == 206 # Partial Content
"-"
else
0
end
curl("--location", "--remote-time", "--continue-at", continue_at.to_s, "--output", destination, *args, **options)
2017-08-08 18:10:13 +02:00
end
def curl_output(*args, **options)
2018-07-30 10:11:00 +02:00
system_command(curl_executable,
args: curl_args(*args, show_output: true, **options),
print_stderr: false)
end
def curl_check_http_content(url, user_agents: [:default], check_content: false, strict: false, require_http: false)
return unless url.start_with? "http"
details = nil
user_agent = nil
hash_needed = url.start_with?("http:") && !require_http
user_agents.each do |ua|
details = curl_http_content_headers_and_checksum(url, hash_needed: hash_needed, user_agent: ua)
user_agent = ua
break if details[:status].to_s.start_with?("2")
end
unless details[:status]
# Hack around https://github.com/Homebrew/brew/issues/3199
return if MacOS.version == :el_capitan
2018-09-17 02:45:00 +02:00
return "The URL #{url} is not reachable"
end
unless details[:status].start_with? "2"
return "The URL #{url} is not reachable (HTTP status code #{details[:status]})"
end
return unless hash_needed
secure_url = url.sub "http", "https"
secure_details =
curl_http_content_headers_and_checksum(secure_url, hash_needed: true, user_agent: user_agent)
if !details[:status].to_s.start_with?("2") ||
!secure_details[:status].to_s.start_with?("2")
return
end
etag_match = details[:etag] &&
details[:etag] == secure_details[:etag]
content_length_match =
details[:content_length] &&
details[:content_length] == secure_details[:content_length]
file_match = details[:file_hash] == secure_details[:file_hash]
if etag_match || content_length_match || file_match
2018-10-10 21:36:06 +00:00
return curl_check_http_redirections(secure_url, original_url: url, user_agents: user_agents)
end
return unless check_content
no_protocol_file_contents = %r{https?:\\?/\\?/}
details[:file] = details[:file].gsub(no_protocol_file_contents, "/")
secure_details[:file] = secure_details[:file].gsub(no_protocol_file_contents, "/")
# Check for the same content after removing all protocols
if details[:file] == secure_details[:file]
2018-10-10 21:36:06 +00:00
return curl_check_http_redirections(secure_url, original_url: url, user_agents: user_agents)
end
return unless strict
# Same size, different content after normalization
# (typical causes: Generated ID, Timestamp, Unix time)
if details[:file].length == secure_details[:file].length
return "The URL #{url} may be able to use HTTPS rather than HTTP. Please verify it in a browser."
end
lenratio = (100 * secure_details[:file].length / details[:file].length).to_i
return unless (90..110).cover?(lenratio)
2018-09-17 02:45:00 +02:00
"The URL #{url} may be able to use HTTPS rather than HTTP. Please verify it in a browser."
end
def curl_http_content_headers_and_checksum(url, hash_needed: false, user_agent: :default)
max_time = hash_needed ? "600" : "25"
output, = curl_output(
2018-10-10 21:36:06 +00:00
"--connect-timeout", "15", "--include", "--max-time", max_time, "--location", url, "--head",
user_agent: user_agent
)
status_code = :unknown
while status_code == :unknown || status_code.to_s.start_with?("3")
headers, _, output = output.partition("\r\n\r\n")
status_code = headers[%r{HTTP\/.* (\d+)}, 1]
end
output_hash = Digest::SHA256.digest(output) if hash_needed
{
status: status_code,
etag: headers[%r{ETag: ([wW]\/)?"(([^"]|\\")*)"}, 2],
content_length: headers[/Content-Length: (\d+)/, 1],
file_hash: output_hash,
file: output,
}
end
2018-10-10 21:36:06 +00:00
def curl_check_http_redirections(url, original_url: nil, user_agents: [:default])
out, _, status= curl_output("--location", "--silent", "--head", url.to_s)
lines = status.success? ? out.lines.map(&:chomp) : []
locations = lines.map { |line| line[/^Location:\s*(.*)$/i, 1] }
.compact
redirect_url = locations.reduce(url) do |current_url, location|
if location.start_with?("/")
uri = URI(current_url)
"#{uri.scheme}://#{uri.host}#{location}"
else
location
end
end
if original_url.start_with?("https://")
unless redirect_url.start_with?("https://")
return "The URL #{original_url} redirects back to HTTP"
end
elsif url.start_with?("https://")
if redirect_url.start_with?("https://")
return "The URL #{original_url} should use HTTPS rather than HTTP"
end
end
end