Merge pull request #10973 from reitermarkus/nested-url-do
Implement nested `url do` API.
This commit is contained in:
commit
2bd0b6ee61
@ -308,7 +308,7 @@ module Cask
|
|||||||
LIVECHECK_REFERENCE_URL = "https://github.com/Homebrew/homebrew-cask/blob/HEAD/doc/cask_language_reference/stanzas/livecheck.md"
|
LIVECHECK_REFERENCE_URL = "https://github.com/Homebrew/homebrew-cask/blob/HEAD/doc/cask_language_reference/stanzas/livecheck.md"
|
||||||
|
|
||||||
def check_hosting_with_livecheck(livecheck_result:)
|
def check_hosting_with_livecheck(livecheck_result:)
|
||||||
return if cask.appcast || cask.livecheckable?
|
return if block_url_offline? || cask.appcast || cask.livecheckable?
|
||||||
return if livecheck_result == :auto_detected
|
return if livecheck_result == :auto_detected
|
||||||
|
|
||||||
add_livecheck = "please add a livecheck. See #{Formatter.url(LIVECHECK_REFERENCE_URL)}"
|
add_livecheck = "please add a livecheck. See #{Formatter.url(LIVECHECK_REFERENCE_URL)}"
|
||||||
@ -419,9 +419,16 @@ module Cask
|
|||||||
URI(cask.url.to_s).scheme == "file"
|
URI(cask.url.to_s).scheme == "file"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def block_url_offline?
|
||||||
|
return false if online?
|
||||||
|
|
||||||
|
cask.url.from_block?
|
||||||
|
end
|
||||||
|
|
||||||
VERIFIED_URL_REFERENCE_URL = "https://github.com/Homebrew/homebrew-cask/blob/master/doc/cask_language_reference/stanzas/url.md#when-url-and-homepage-hostnames-differ-add-verified"
|
VERIFIED_URL_REFERENCE_URL = "https://github.com/Homebrew/homebrew-cask/blob/master/doc/cask_language_reference/stanzas/url.md#when-url-and-homepage-hostnames-differ-add-verified"
|
||||||
|
|
||||||
def check_unnecessary_verified
|
def check_unnecessary_verified
|
||||||
|
return if block_url_offline?
|
||||||
return unless verified_present?
|
return unless verified_present?
|
||||||
return unless url_match_homepage?
|
return unless url_match_homepage?
|
||||||
return unless verified_matches_url?
|
return unless verified_matches_url?
|
||||||
@ -432,7 +439,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
|
|
||||||
def check_missing_verified
|
def check_missing_verified
|
||||||
return if cask.url.from_block?
|
return if block_url_offline?
|
||||||
return if file_url?
|
return if file_url?
|
||||||
return if url_match_homepage?
|
return if url_match_homepage?
|
||||||
return if verified_present?
|
return if verified_present?
|
||||||
@ -443,6 +450,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
|
|
||||||
def check_no_match
|
def check_no_match
|
||||||
|
return if block_url_offline?
|
||||||
return unless verified_present?
|
return unless verified_present?
|
||||||
return if verified_matches_url?
|
return if verified_matches_url?
|
||||||
|
|
||||||
@ -592,7 +600,7 @@ module Cask
|
|||||||
return if cask.tap == "homebrew/cask-versions"
|
return if cask.tap == "homebrew/cask-versions"
|
||||||
|
|
||||||
odebug "Auditing GitHub prerelease"
|
odebug "Auditing GitHub prerelease"
|
||||||
user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if @online
|
user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if online?
|
||||||
return if user.nil?
|
return if user.nil?
|
||||||
|
|
||||||
tag = SharedAudits.github_tag_from_url(cask.url)
|
tag = SharedAudits.github_tag_from_url(cask.url)
|
||||||
@ -604,7 +612,7 @@ module Cask
|
|||||||
def check_gitlab_prerelease_version
|
def check_gitlab_prerelease_version
|
||||||
return if cask.tap == "homebrew/cask-versions"
|
return if cask.tap == "homebrew/cask-versions"
|
||||||
|
|
||||||
user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if @online
|
user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if online?
|
||||||
return if user.nil?
|
return if user.nil?
|
||||||
|
|
||||||
odebug "Auditing GitLab prerelease"
|
odebug "Auditing GitLab prerelease"
|
||||||
@ -616,7 +624,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
|
|
||||||
def check_github_repository_archived
|
def check_github_repository_archived
|
||||||
user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if @online
|
user, repo = get_repo_data(%r{https?://github\.com/([^/]+)/([^/]+)/?.*}) if online?
|
||||||
return if user.nil?
|
return if user.nil?
|
||||||
|
|
||||||
odebug "Auditing GitHub repo archived"
|
odebug "Auditing GitHub repo archived"
|
||||||
@ -636,7 +644,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
|
|
||||||
def check_gitlab_repository_archived
|
def check_gitlab_repository_archived
|
||||||
user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if @online
|
user, repo = get_repo_data(%r{https?://gitlab\.com/([^/]+)/([^/]+)/?.*}) if online?
|
||||||
return if user.nil?
|
return if user.nil?
|
||||||
|
|
||||||
odebug "Auditing GitLab repo archived"
|
odebug "Auditing GitLab repo archived"
|
||||||
|
|||||||
@ -96,6 +96,7 @@ module Cask
|
|||||||
@token = cask.token
|
@token = cask.token
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def name(*args)
|
def name(*args)
|
||||||
@name ||= []
|
@name ||= []
|
||||||
return @name if args.empty?
|
return @name if args.empty?
|
||||||
@ -103,6 +104,7 @@ module Cask
|
|||||||
@name.concat(args.flatten)
|
@name.concat(args.flatten)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def desc(description = nil)
|
def desc(description = nil)
|
||||||
set_unique_stanza(:desc, description.nil?) { description }
|
set_unique_stanza(:desc, description.nil?) { description }
|
||||||
end
|
end
|
||||||
@ -121,6 +123,7 @@ module Cask
|
|||||||
raise CaskInvalidError.new(cask, "'#{stanza}' stanza failed with: #{e}")
|
raise CaskInvalidError.new(cask, "'#{stanza}' stanza failed with: #{e}")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def homepage(homepage = nil)
|
def homepage(homepage = nil)
|
||||||
set_unique_stanza(:homepage, homepage.nil?) { homepage }
|
set_unique_stanza(:homepage, homepage.nil?) { homepage }
|
||||||
end
|
end
|
||||||
@ -176,32 +179,32 @@ module Cask
|
|||||||
@language_blocks.keys.flatten
|
@language_blocks.keys.flatten
|
||||||
end
|
end
|
||||||
|
|
||||||
def url(*args, **options)
|
# @api public
|
||||||
|
def url(*args, **options, &block)
|
||||||
caller_location = caller_locations[0]
|
caller_location = caller_locations[0]
|
||||||
|
|
||||||
set_unique_stanza(:url, args.empty? && options.empty? && !block_given?) do
|
set_unique_stanza(:url, args.empty? && options.empty? && !block) do
|
||||||
if block_given?
|
if block
|
||||||
LazyObject.new do
|
URL.new(*args, **options, caller_location: caller_location, dsl: self, &block)
|
||||||
*args = yield
|
|
||||||
options = args.last.is_a?(Hash) ? args.pop : {}
|
|
||||||
URL.new(*args, **options, from_block: true, caller_location: caller_location)
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
URL.new(*args, **options, caller_location: caller_location)
|
URL.new(*args, **options, caller_location: caller_location)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def appcast(*args)
|
def appcast(*args)
|
||||||
set_unique_stanza(:appcast, args.empty?) { DSL::Appcast.new(*args) }
|
set_unique_stanza(:appcast, args.empty?) { DSL::Appcast.new(*args) }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def container(*args)
|
def container(*args)
|
||||||
set_unique_stanza(:container, args.empty?) do
|
set_unique_stanza(:container, args.empty?) do
|
||||||
DSL::Container.new(*args)
|
DSL::Container.new(*args)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def version(arg = nil)
|
def version(arg = nil)
|
||||||
set_unique_stanza(:version, arg.nil?) do
|
set_unique_stanza(:version, arg.nil?) do
|
||||||
if !arg.is_a?(String) && arg != :latest
|
if !arg.is_a?(String) && arg != :latest
|
||||||
@ -212,6 +215,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def sha256(arg = nil)
|
def sha256(arg = nil)
|
||||||
set_unique_stanza(:sha256, arg.nil?) do
|
set_unique_stanza(:sha256, arg.nil?) do
|
||||||
case arg
|
case arg
|
||||||
@ -226,6 +230,7 @@ module Cask
|
|||||||
end
|
end
|
||||||
|
|
||||||
# `depends_on` uses a load method so that multiple stanzas can be merged.
|
# `depends_on` uses a load method so that multiple stanzas can be merged.
|
||||||
|
# @api public
|
||||||
def depends_on(*args)
|
def depends_on(*args)
|
||||||
@depends_on ||= DSL::DependsOn.new
|
@depends_on ||= DSL::DependsOn.new
|
||||||
return @depends_on if args.empty?
|
return @depends_on if args.empty?
|
||||||
@ -238,6 +243,7 @@ module Cask
|
|||||||
@depends_on
|
@depends_on
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def conflicts_with(*args)
|
def conflicts_with(*args)
|
||||||
# TODO: remove this constraint, and instead merge multiple conflicts_with stanzas
|
# TODO: remove this constraint, and instead merge multiple conflicts_with stanzas
|
||||||
set_unique_stanza(:conflicts_with, args.empty?) { DSL::ConflictsWith.new(*args) }
|
set_unique_stanza(:conflicts_with, args.empty?) { DSL::ConflictsWith.new(*args) }
|
||||||
@ -251,6 +257,7 @@ module Cask
|
|||||||
cask.caskroom_path
|
cask.caskroom_path
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def staged_path
|
def staged_path
|
||||||
return @staged_path if @staged_path
|
return @staged_path if @staged_path
|
||||||
|
|
||||||
@ -258,6 +265,7 @@ module Cask
|
|||||||
@staged_path = caskroom_path.join(cask_version.to_s)
|
@staged_path = caskroom_path.join(cask_version.to_s)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def caveats(*strings, &block)
|
def caveats(*strings, &block)
|
||||||
@caveats ||= DSL::Caveats.new(cask)
|
@caveats ||= DSL::Caveats.new(cask)
|
||||||
if block
|
if block
|
||||||
@ -276,10 +284,12 @@ module Cask
|
|||||||
@caveats&.discontinued?
|
@caveats&.discontinued?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def auto_updates(auto_updates = nil)
|
def auto_updates(auto_updates = nil)
|
||||||
set_unique_stanza(:auto_updates, auto_updates.nil?) { auto_updates }
|
set_unique_stanza(:auto_updates, auto_updates.nil?) { auto_updates }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def livecheck(&block)
|
def livecheck(&block)
|
||||||
@livecheck ||= Livecheck.new(self)
|
@livecheck ||= Livecheck.new(self)
|
||||||
return @livecheck unless block
|
return @livecheck unless block
|
||||||
@ -317,8 +327,6 @@ module Cask
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# No need to define it as it's the default/superclass implementation.
|
|
||||||
# rubocop:disable Style/MissingRespondToMissing
|
|
||||||
def method_missing(method, *)
|
def method_missing(method, *)
|
||||||
if method
|
if method
|
||||||
Utils.method_missing_message(method, token)
|
Utils.method_missing_message(method, token)
|
||||||
@ -327,8 +335,12 @@ module Cask
|
|||||||
super
|
super
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
# rubocop:enable Style/MissingRespondToMissing
|
|
||||||
|
|
||||||
|
def respond_to_missing?(*)
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def appdir
|
def appdir
|
||||||
cask.config.appdir
|
cask.config.appdir
|
||||||
end
|
end
|
||||||
|
|||||||
@ -4,7 +4,11 @@
|
|||||||
# Class corresponding to the `url` stanza.
|
# Class corresponding to the `url` stanza.
|
||||||
#
|
#
|
||||||
# @api private
|
# @api private
|
||||||
class URL
|
class URL < Delegator
|
||||||
|
extend T::Sig
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
class DSL
|
||||||
extend T::Sig
|
extend T::Sig
|
||||||
|
|
||||||
attr_reader :uri, :specs,
|
attr_reader :uri, :specs,
|
||||||
@ -16,6 +20,7 @@ class URL
|
|||||||
extend Forwardable
|
extend Forwardable
|
||||||
def_delegators :uri, :path, :scheme, :to_s
|
def_delegators :uri, :path, :scheme, :to_s
|
||||||
|
|
||||||
|
# @api public
|
||||||
sig {
|
sig {
|
||||||
params(
|
params(
|
||||||
uri: T.any(URI::Generic, String),
|
uri: T.any(URI::Generic, String),
|
||||||
@ -31,9 +36,7 @@ class URL
|
|||||||
header: T.nilable(String),
|
header: T.nilable(String),
|
||||||
user_agent: T.nilable(T.any(Symbol, String)),
|
user_agent: T.nilable(T.any(Symbol, String)),
|
||||||
data: T.nilable(T::Hash[String, String]),
|
data: T.nilable(T::Hash[String, String]),
|
||||||
from_block: T::Boolean,
|
).void
|
||||||
caller_location: Thread::Backtrace::Location,
|
|
||||||
).returns(T.untyped)
|
|
||||||
}
|
}
|
||||||
def initialize(
|
def initialize(
|
||||||
uri,
|
uri,
|
||||||
@ -48,9 +51,7 @@ class URL
|
|||||||
referer: nil,
|
referer: nil,
|
||||||
header: nil,
|
header: nil,
|
||||||
user_agent: nil,
|
user_agent: nil,
|
||||||
data: nil,
|
data: nil
|
||||||
from_block: false,
|
|
||||||
caller_location: T.must(caller_locations).fetch(0)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
@uri = URI(uri)
|
@uri = URI(uri)
|
||||||
@ -70,11 +71,153 @@ class URL
|
|||||||
specs[:data] = @data = data
|
specs[:data] = @data = data
|
||||||
|
|
||||||
@specs = specs.compact
|
@specs = specs.compact
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
@from_block = from_block
|
# @api private
|
||||||
|
class BlockDSL
|
||||||
|
extend T::Sig
|
||||||
|
|
||||||
|
module PageWithURL
|
||||||
|
extend T::Sig
|
||||||
|
|
||||||
|
# @api public
|
||||||
|
sig { returns(URI::Generic) }
|
||||||
|
attr_accessor :url
|
||||||
|
end
|
||||||
|
|
||||||
|
sig {
|
||||||
|
params(
|
||||||
|
uri: T.nilable(T.any(URI::Generic, String)),
|
||||||
|
dsl: T.nilable(Cask::DSL),
|
||||||
|
block: T.proc.params(arg0: T.all(String, PageWithURL)).returns(T.untyped),
|
||||||
|
).void
|
||||||
|
}
|
||||||
|
def initialize(uri, dsl: nil, &block)
|
||||||
|
@uri = uri
|
||||||
|
@dsl = dsl
|
||||||
|
@block = block
|
||||||
|
end
|
||||||
|
|
||||||
|
sig { returns(T.untyped) }
|
||||||
|
def call
|
||||||
|
if @uri
|
||||||
|
result = curl_output("--fail", "--silent", "--location", @uri)
|
||||||
|
result.assert_success!
|
||||||
|
|
||||||
|
page = result.stdout
|
||||||
|
page.extend PageWithURL
|
||||||
|
page.url = URI(@uri)
|
||||||
|
|
||||||
|
instance_exec(page, &@block)
|
||||||
|
else
|
||||||
|
instance_exec(&@block)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
|
sig {
|
||||||
|
params(
|
||||||
|
uri: T.any(URI::Generic, String),
|
||||||
|
block: T.proc.params(arg0: T.all(String, PageWithURL)).returns(T.untyped),
|
||||||
|
).void
|
||||||
|
}
|
||||||
|
def url(uri, &block)
|
||||||
|
self.class.new(uri, &block).call
|
||||||
|
end
|
||||||
|
private :url
|
||||||
|
|
||||||
|
# @api public
|
||||||
|
def method_missing(method, *args, **options, &block)
|
||||||
|
if @dsl.respond_to?(method)
|
||||||
|
T.unsafe(@dsl).public_send(method, *args, **options, &block)
|
||||||
|
else
|
||||||
|
super
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def respond_to_missing?(method, include_all)
|
||||||
|
@dsl.respond_to?(method, include_all) || super
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
sig {
|
||||||
|
params(
|
||||||
|
uri: T.nilable(T.any(URI::Generic, String)),
|
||||||
|
verified: T.nilable(String),
|
||||||
|
using: T.nilable(Symbol),
|
||||||
|
tag: T.nilable(String),
|
||||||
|
branch: T.nilable(String),
|
||||||
|
revisions: T.nilable(T::Array[String]),
|
||||||
|
revision: T.nilable(String),
|
||||||
|
trust_cert: T.nilable(T::Boolean),
|
||||||
|
cookies: T.nilable(T::Hash[String, String]),
|
||||||
|
referer: T.nilable(T.any(URI::Generic, String)),
|
||||||
|
header: T.nilable(String),
|
||||||
|
user_agent: T.nilable(T.any(Symbol, String)),
|
||||||
|
data: T.nilable(T::Hash[String, String]),
|
||||||
|
caller_location: Thread::Backtrace::Location,
|
||||||
|
dsl: T.nilable(Cask::DSL),
|
||||||
|
block: T.nilable(T.proc.params(arg0: T.all(String, BlockDSL::PageWithURL)).returns(T.untyped)),
|
||||||
|
).void
|
||||||
|
}
|
||||||
|
def initialize(
|
||||||
|
uri = nil,
|
||||||
|
verified: nil,
|
||||||
|
using: nil,
|
||||||
|
tag: nil,
|
||||||
|
branch: nil,
|
||||||
|
revisions: nil,
|
||||||
|
revision: nil,
|
||||||
|
trust_cert: nil,
|
||||||
|
cookies: nil,
|
||||||
|
referer: nil,
|
||||||
|
header: nil,
|
||||||
|
user_agent: nil,
|
||||||
|
data: nil,
|
||||||
|
caller_location: T.must(caller_locations).fetch(0),
|
||||||
|
dsl: nil,
|
||||||
|
&block
|
||||||
|
)
|
||||||
|
super(
|
||||||
|
if block
|
||||||
|
LazyObject.new do
|
||||||
|
*args = BlockDSL.new(uri, dsl: dsl, &block).call
|
||||||
|
options = args.last.is_a?(Hash) ? args.pop : {}
|
||||||
|
uri = T.let(args.first, T.any(URI::Generic, String))
|
||||||
|
DSL.new(uri, **options)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
DSL.new(
|
||||||
|
T.must(uri),
|
||||||
|
verified: verified,
|
||||||
|
using: using,
|
||||||
|
tag: tag,
|
||||||
|
branch: branch,
|
||||||
|
revisions: revisions,
|
||||||
|
revision: revision,
|
||||||
|
trust_cert: trust_cert,
|
||||||
|
cookies: cookies,
|
||||||
|
referer: referer,
|
||||||
|
header: header,
|
||||||
|
user_agent: user_agent,
|
||||||
|
data: data,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
)
|
||||||
|
|
||||||
|
@from_block = !block.nil?
|
||||||
@caller_location = caller_location
|
@caller_location = caller_location
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def __getobj__
|
||||||
|
@dsl
|
||||||
|
end
|
||||||
|
|
||||||
|
def __setobj__(dsl)
|
||||||
|
@dsl = dsl
|
||||||
|
end
|
||||||
|
|
||||||
sig { returns(T.nilable(String)) }
|
sig { returns(T.nilable(String)) }
|
||||||
def raw_interpolated_url
|
def raw_interpolated_url
|
||||||
return @raw_interpolated_url if defined?(@raw_interpolated_url)
|
return @raw_interpolated_url if defined?(@raw_interpolated_url)
|
||||||
|
|||||||
6
Library/Homebrew/cask/url.rbi
Normal file
6
Library/Homebrew/cask/url.rbi
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
# typed: strict
|
||||||
|
# typed: false
|
||||||
|
|
||||||
|
class URL
|
||||||
|
include Kernel
|
||||||
|
end
|
||||||
@ -795,7 +795,17 @@ describe Cask::Audit, :cask do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when doing the audit" do
|
context "when doing an offline audit" do
|
||||||
|
let(:online) { false }
|
||||||
|
|
||||||
|
it "does not evaluate the block" do
|
||||||
|
expect(run).not_to pass
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when doing and online audit" do
|
||||||
|
let(:online) { true }
|
||||||
|
|
||||||
it "evaluates the block" do
|
it "evaluates the block" do
|
||||||
expect(run).to fail_with(/Boom/)
|
expect(run).to fail_with(/Boom/)
|
||||||
end
|
end
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user