Merge pull request #20425 from Homebrew/internal-api-helper
Create `Homebrew::API::Internal` for working with internal API
This commit is contained in:
commit
b8c82b44b8
177
Library/Homebrew/api/internal.rb
Normal file
177
Library/Homebrew/api/internal.rb
Normal file
@ -0,0 +1,177 @@
|
||||
# typed: strict
|
||||
# frozen_string_literal: true
|
||||
|
||||
require "cachable"
|
||||
require "api"
|
||||
require "api/source_download"
|
||||
require "download_queue"
|
||||
require "formula_stub"
|
||||
|
||||
module Homebrew
|
||||
module API
|
||||
# Helper functions for using the JSON internal API.
|
||||
module Internal
|
||||
extend Cachable
|
||||
|
||||
private_class_method :cache
|
||||
|
||||
sig { returns(String) }
|
||||
def self.formula_endpoint
|
||||
"internal/formula.#{SimulateSystem.current_tag}.jws.json"
|
||||
end
|
||||
|
||||
sig { returns(String) }
|
||||
def self.cask_endpoint
|
||||
"internal/cask.#{SimulateSystem.current_tag}.jws.json"
|
||||
end
|
||||
|
||||
sig { params(name: String).returns(Homebrew::FormulaStub) }
|
||||
def self.formula_stub(name)
|
||||
return cache["formula_stubs"][name] if cache.key?("formula_stubs") && cache["formula_stubs"].key?(name)
|
||||
|
||||
stub_array = formula_arrays[name]
|
||||
raise "No formula stub found for #{name}" unless stub_array
|
||||
|
||||
stub = Homebrew::FormulaStub.new(
|
||||
name: name,
|
||||
pkg_version: PkgVersion.parse(stub_array[0]),
|
||||
rebuild: stub_array[1],
|
||||
sha256: stub_array[2],
|
||||
)
|
||||
|
||||
cache["formula_stubs"] ||= {}
|
||||
cache["formula_stubs"][name] = stub
|
||||
|
||||
stub
|
||||
end
|
||||
|
||||
sig {
|
||||
params(download_queue: T.nilable(Homebrew::DownloadQueue), stale_seconds: Integer)
|
||||
.returns([T::Hash[String, T.untyped], T::Boolean])
|
||||
}
|
||||
def self.fetch_formula_api!(download_queue: nil, stale_seconds: Homebrew::EnvConfig.api_auto_update_secs.to_i)
|
||||
json_contents, updated = (Homebrew::API.fetch_json_api_file formula_endpoint, stale_seconds:, download_queue:)
|
||||
[T.cast(json_contents, T::Hash[String, T.untyped]), updated]
|
||||
end
|
||||
|
||||
sig {
|
||||
params(download_queue: T.nilable(Homebrew::DownloadQueue), stale_seconds: Integer)
|
||||
.returns([T::Hash[String, T.untyped], T::Boolean])
|
||||
}
|
||||
def self.fetch_cask_api!(download_queue: nil, stale_seconds: Homebrew::EnvConfig.api_auto_update_secs.to_i)
|
||||
json_contents, updated = (Homebrew::API.fetch_json_api_file cask_endpoint, stale_seconds:, download_queue:)
|
||||
[T.cast(json_contents, T::Hash[String, T.untyped]), updated]
|
||||
end
|
||||
|
||||
sig { returns(T::Boolean) }
|
||||
def self.download_and_cache_formula_data!
|
||||
json_contents, updated = fetch_formula_api!
|
||||
cache["formula_stubs"] = {}
|
||||
cache["formula_aliases"] = json_contents["aliases"]
|
||||
cache["formula_renames"] = json_contents["renames"]
|
||||
cache["formula_tap_migrations"] = json_contents["tap_migrations"]
|
||||
cache["formula_arrays"] = json_contents["formulae"]
|
||||
|
||||
updated
|
||||
end
|
||||
private_class_method :download_and_cache_formula_data!
|
||||
|
||||
sig { returns(T::Boolean) }
|
||||
def self.download_and_cache_cask_data!
|
||||
json_contents, updated = fetch_cask_api!
|
||||
cache["cask_stubs"] = {}
|
||||
cache["cask_renames"] = json_contents["renames"]
|
||||
cache["cask_tap_migrations"] = json_contents["tap_migrations"]
|
||||
cache["cask_hashes"] = json_contents["casks"]
|
||||
|
||||
updated
|
||||
end
|
||||
private_class_method :download_and_cache_cask_data!
|
||||
|
||||
sig { params(regenerate: T::Boolean).void }
|
||||
def self.write_formula_names_and_aliases(regenerate: false)
|
||||
download_and_cache_formula_data! unless cache.key?("formula_arrays")
|
||||
|
||||
Homebrew::API.write_names_file!(formula_arrays.keys, "formula", regenerate:)
|
||||
Homebrew::API.write_aliases_file!(formula_aliases, "formula", regenerate:)
|
||||
end
|
||||
|
||||
sig { params(regenerate: T::Boolean).void }
|
||||
def self.write_cask_names(regenerate: false)
|
||||
download_and_cache_cask_data! unless cache.key?("cask_hashes")
|
||||
|
||||
Homebrew::API.write_names_file!(cask_hashes.keys, "cask", regenerate:)
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, [String, Integer, T.nilable(String)]]) }
|
||||
def self.formula_arrays
|
||||
unless cache.key?("formula_arrays")
|
||||
updated = download_and_cache_formula_data!
|
||||
write_formula_names_and_aliases(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["formula_arrays"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, String]) }
|
||||
def self.formula_aliases
|
||||
unless cache.key?("formula_aliases")
|
||||
updated = download_and_cache_formula_data!
|
||||
write_formula_names_and_aliases(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["formula_aliases"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, String]) }
|
||||
def self.formula_renames
|
||||
unless cache.key?("formula_renames")
|
||||
updated = download_and_cache_formula_data!
|
||||
write_formula_names_and_aliases(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["formula_renames"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, String]) }
|
||||
def self.formula_tap_migrations
|
||||
unless cache.key?("formula_tap_migrations")
|
||||
updated = download_and_cache_formula_data!
|
||||
write_formula_names_and_aliases(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["formula_tap_migrations"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, T::Hash[String, T.untyped]]) }
|
||||
def self.cask_hashes
|
||||
unless cache.key?("cask_hashes")
|
||||
updated = download_and_cache_cask_data!
|
||||
write_cask_names(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["cask_hashes"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, String]) }
|
||||
def self.cask_renames
|
||||
unless cache.key?("cask_renames")
|
||||
updated = download_and_cache_cask_data!
|
||||
write_cask_names(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["cask_renames"]
|
||||
end
|
||||
|
||||
sig { returns(T::Hash[String, String]) }
|
||||
def self.cask_tap_migrations
|
||||
unless cache.key?("cask_tap_migrations")
|
||||
updated = download_and_cache_cask_data!
|
||||
write_cask_names(regenerate: updated)
|
||||
end
|
||||
|
||||
cache["cask_tap_migrations"]
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
34
Library/Homebrew/formula_stub.rb
Normal file
34
Library/Homebrew/formula_stub.rb
Normal file
@ -0,0 +1,34 @@
|
||||
# typed: strict
|
||||
# frozen_string_literal: true
|
||||
|
||||
require "pkg_version"
|
||||
|
||||
module Homebrew
|
||||
# A stub for a formula, with only the information needed to fetch the bottle manifest.
|
||||
class FormulaStub < T::Struct
|
||||
const :name, String
|
||||
const :pkg_version, PkgVersion
|
||||
const :rebuild, Integer, default: 0
|
||||
const :sha256, T.nilable(String)
|
||||
|
||||
sig { returns(Version) }
|
||||
def version
|
||||
pkg_version.version
|
||||
end
|
||||
|
||||
sig { returns(Integer) }
|
||||
def revision
|
||||
pkg_version.revision
|
||||
end
|
||||
|
||||
sig { params(other: T.anything).returns(T::Boolean) }
|
||||
def ==(other)
|
||||
case other
|
||||
when FormulaStub
|
||||
name == other.name && pkg_version == other.pkg_version && rebuild == other.rebuild && sha256 == other.sha256
|
||||
else
|
||||
false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
179
Library/Homebrew/test/api/internal_spec.rb
Normal file
179
Library/Homebrew/test/api/internal_spec.rb
Normal file
@ -0,0 +1,179 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require "api/internal"
|
||||
|
||||
RSpec.describe Homebrew::API::Internal do
|
||||
let(:cache_dir) { mktmpdir }
|
||||
|
||||
before do
|
||||
FileUtils.mkdir_p(cache_dir/"internal")
|
||||
stub_const("Homebrew::API::HOMEBREW_CACHE_API", cache_dir)
|
||||
end
|
||||
|
||||
def mock_curl_download(stdout:)
|
||||
allow(Utils::Curl).to receive(:curl_download) do |*_args, **kwargs|
|
||||
kwargs[:to].write stdout
|
||||
end
|
||||
allow(Homebrew::API).to receive(:verify_and_parse_jws) do |json_data|
|
||||
[true, json_data]
|
||||
end
|
||||
end
|
||||
|
||||
context "for formulae" do
|
||||
let(:formula_json) do
|
||||
<<~JSON
|
||||
{
|
||||
"formulae": {
|
||||
"foo": ["1.0.0", 0, "09f88b61e36045188ddb1b1ba8e402b9f3debee1770cc4ca91355eeccb5f4a38"],
|
||||
"bar": ["0.4.0_5", 0, "bb6e3408f39a404770529cfce548dc2666e861077acd173825cb3138c27c205a"],
|
||||
"baz": ["10.4.5_2", 2, "404c97537d65ca0b75c389e7d439dcefb9b56f34d3b98017669eda0d0501add7"]
|
||||
},
|
||||
"aliases": {
|
||||
"foo-alias1": "foo",
|
||||
"foo-alias2": "foo",
|
||||
"bar-alias": "bar"
|
||||
},
|
||||
"renames": {
|
||||
"foo-old": "foo",
|
||||
"bar-old": "bar",
|
||||
"baz-old": "baz"
|
||||
},
|
||||
"tap_migrations": {
|
||||
"abc": "some/tap",
|
||||
"def": "another/tap"
|
||||
}
|
||||
}
|
||||
JSON
|
||||
end
|
||||
let(:formula_arrays) do
|
||||
{
|
||||
"foo" => ["1.0.0", 0, "09f88b61e36045188ddb1b1ba8e402b9f3debee1770cc4ca91355eeccb5f4a38"],
|
||||
"bar" => ["0.4.0_5", 0, "bb6e3408f39a404770529cfce548dc2666e861077acd173825cb3138c27c205a"],
|
||||
"baz" => ["10.4.5_2", 2, "404c97537d65ca0b75c389e7d439dcefb9b56f34d3b98017669eda0d0501add7"],
|
||||
}
|
||||
end
|
||||
let(:formula_stubs) do
|
||||
formula_arrays.to_h do |name, (pkg_version, rebuild, sha256)|
|
||||
stub = Homebrew::FormulaStub.new(
|
||||
name: name,
|
||||
pkg_version: PkgVersion.parse(pkg_version),
|
||||
rebuild: rebuild,
|
||||
sha256: sha256,
|
||||
)
|
||||
[name, stub]
|
||||
end
|
||||
end
|
||||
let(:formulae_aliases) do
|
||||
{
|
||||
"foo-alias1" => "foo",
|
||||
"foo-alias2" => "foo",
|
||||
"bar-alias" => "bar",
|
||||
}
|
||||
end
|
||||
let(:formulae_renames) do
|
||||
{
|
||||
"foo-old" => "foo",
|
||||
"bar-old" => "bar",
|
||||
"baz-old" => "baz",
|
||||
}
|
||||
end
|
||||
let(:formula_tap_migrations) do
|
||||
{
|
||||
"abc" => "some/tap",
|
||||
"def" => "another/tap",
|
||||
}
|
||||
end
|
||||
|
||||
it "returns the expected formula stubs" do
|
||||
mock_curl_download stdout: formula_json
|
||||
formula_stubs.each do |name, stub|
|
||||
expect(described_class.formula_stub(name)).to eq stub
|
||||
end
|
||||
end
|
||||
|
||||
it "returns the expected formula arrays" do
|
||||
mock_curl_download stdout: formula_json
|
||||
formula_arrays_output = described_class.formula_arrays
|
||||
expect(formula_arrays_output).to eq formula_arrays
|
||||
end
|
||||
|
||||
it "returns the expected formula alias list" do
|
||||
mock_curl_download stdout: formula_json
|
||||
formula_aliases_output = described_class.formula_aliases
|
||||
expect(formula_aliases_output).to eq formulae_aliases
|
||||
end
|
||||
|
||||
it "returns the expected formula rename list" do
|
||||
mock_curl_download stdout: formula_json
|
||||
formula_renames_output = described_class.formula_renames
|
||||
expect(formula_renames_output).to eq formulae_renames
|
||||
end
|
||||
|
||||
it "returns the expected formula tap migrations list" do
|
||||
mock_curl_download stdout: formula_json
|
||||
formula_tap_migrations_output = described_class.formula_tap_migrations
|
||||
expect(formula_tap_migrations_output).to eq formula_tap_migrations
|
||||
end
|
||||
end
|
||||
|
||||
context "for casks" do
|
||||
let(:cask_json) do
|
||||
<<~JSON
|
||||
{
|
||||
"casks": {
|
||||
"foo": { "version": "1.0.0" },
|
||||
"bar": { "version": "0.4.0" },
|
||||
"baz": { "version": "10.4.5" }
|
||||
},
|
||||
"renames": {
|
||||
"foo-old": "foo",
|
||||
"bar-old": "bar",
|
||||
"baz-old": "baz"
|
||||
},
|
||||
"tap_migrations": {
|
||||
"abc": "some/tap",
|
||||
"def": "another/tap"
|
||||
}
|
||||
}
|
||||
JSON
|
||||
end
|
||||
let(:cask_hashes) do
|
||||
{
|
||||
"foo" => { "version" => "1.0.0" },
|
||||
"bar" => { "version" => "0.4.0" },
|
||||
"baz" => { "version" => "10.4.5" },
|
||||
}
|
||||
end
|
||||
let(:cask_renames) do
|
||||
{
|
||||
"foo-old" => "foo",
|
||||
"bar-old" => "bar",
|
||||
"baz-old" => "baz",
|
||||
}
|
||||
end
|
||||
let(:cask_tap_migrations) do
|
||||
{
|
||||
"abc" => "some/tap",
|
||||
"def" => "another/tap",
|
||||
}
|
||||
end
|
||||
|
||||
it "returns the expected cask hashes" do
|
||||
mock_curl_download stdout: cask_json
|
||||
cask_hashes_output = described_class.cask_hashes
|
||||
expect(cask_hashes_output).to eq cask_hashes
|
||||
end
|
||||
|
||||
it "returns the expected cask rename list" do
|
||||
mock_curl_download stdout: cask_json
|
||||
cask_renames_output = described_class.cask_renames
|
||||
expect(cask_renames_output).to eq cask_renames
|
||||
end
|
||||
|
||||
it "returns the expected cask tap migrations list" do
|
||||
mock_curl_download stdout: cask_json
|
||||
cask_tap_migrations_output = described_class.cask_tap_migrations
|
||||
expect(cask_tap_migrations_output).to eq cask_tap_migrations
|
||||
end
|
||||
end
|
||||
end
|
||||
Loading…
x
Reference in New Issue
Block a user