From 6467fbadee2f268345eb2d76482c0e2ce955ba5b Mon Sep 17 00:00:00 2001
From: Mike McQuaid <mike@mikemcquaid.com>
Date: Wed, 9 Dec 2020 11:55:27 +0000
Subject: [PATCH] MacOS::Version: add (and use) architecture.

---
 .../Homebrew/dev-cmd/dispatch-build-bottle.rb | 12 ++----
 .../Homebrew/extend/os/mac/utils/bottles.rb   |  5 ++-
 Library/Homebrew/os/mac/version.rb            | 39 +++++++++++++-----
 Library/Homebrew/test/os/mac/version_spec.rb  | 40 +++++++++++++++++--
 4 files changed, 73 insertions(+), 23 deletions(-)

diff --git a/Library/Homebrew/dev-cmd/dispatch-build-bottle.rb b/Library/Homebrew/dev-cmd/dispatch-build-bottle.rb
index 69e6ee2a9f..9732e9bce6 100644
--- a/Library/Homebrew/dev-cmd/dispatch-build-bottle.rb
+++ b/Library/Homebrew/dev-cmd/dispatch-build-bottle.rb
@@ -36,13 +36,8 @@ module Homebrew
     args = dispatch_build_bottle_args.parse
 
     # Fixup version for ARM/Apple Silicon
-    arm_regex = Regexp.union(/^arm64_/, /-arm$/)
-    arm_label = if arm_regex.match?(args.macos)
-      args.macos&.gsub!(arm_regex, "")
-      true
-    else
-      false
-    end
+    # TODO: fix label name to be 11-arm64 instead and remove this.
+    args.macos&.gsub!(/^11-arm$/, "11-arm64")
 
     macos = args.macos&.yield_self do |s|
       MacOS::Version.from_symbol(s.to_sym)
@@ -53,7 +48,8 @@ module Homebrew
     raise UsageError, "Must specify --macos option" if macos.blank?
 
     # Fixup label for ARM/Apple Silicon
-    macos_label = if arm_label
+    macos_label = if macos.arch == :arm64
+      # TODO: fix label name to be 11-arm64 instead.
       "#{macos}-arm"
     else
       macos.to_s
diff --git a/Library/Homebrew/extend/os/mac/utils/bottles.rb b/Library/Homebrew/extend/os/mac/utils/bottles.rb
index 6bd47aa1a0..75023c1473 100644
--- a/Library/Homebrew/extend/os/mac/utils/bottles.rb
+++ b/Library/Homebrew/extend/os/mac/utils/bottles.rb
@@ -42,7 +42,10 @@ module Utils
         return if tag_version.blank?
 
         keys.find do |key|
-          MacOS::Version.from_symbol(key) <= tag_version
+          key_version = MacOS::Version.from_symbol(key)
+          next if key_version.arch != tag_version.arch
+
+          key_version <= tag_version
         rescue MacOSVersionError
           false
         end
diff --git a/Library/Homebrew/os/mac/version.rb b/Library/Homebrew/os/mac/version.rb
index 75893a1058..3950ca1b41 100644
--- a/Library/Homebrew/os/mac/version.rb
+++ b/Library/Homebrew/os/mac/version.rb
@@ -13,6 +13,8 @@ module OS
     class Version < ::Version
       extend T::Sig
 
+      attr_reader :arch
+
       SYMBOLS = {
         big_sur:     "11",
         catalina:    "10.15",
@@ -25,20 +27,37 @@ module OS
 
       sig { params(sym: Symbol).returns(T.attached_class) }
       def self.from_symbol(sym)
-        @all_archs_regex ||= /^#{Regexp.union(Hardware::CPU::ALL_ARCHS.map(&:to_s))}_/
-        sym_without_arch = sym.to_s
-                              .sub(@all_archs_regex, "")
-                              .to_sym
-        str = SYMBOLS.fetch(sym_without_arch) { raise MacOSVersionError, sym }
-        new(str)
+        version, arch = version_arch(sym)
+        version ||= sym
+        str = SYMBOLS.fetch(version.to_sym) { raise MacOSVersionError, sym }
+        new(str, arch: arch)
+      end
+
+      sig { params(value: T.any(String, Symbol)).returns(T::Array[String]) }
+      def self.version_arch(value)
+        @all_archs_regex ||= begin
+          all_archs = Hardware::CPU::ALL_ARCHS.map(&:to_s)
+          /^((#{Regexp.union(all_archs)})_)?([\w.]+)(-(#{Regexp.union(all_archs)}))?$/
+        end
+        match = @all_archs_regex.match(value)
+        return [] unless match
+
+        version = match[3]
+        arch = match[2] || match[5]
+        [version, arch]
       end
 
-      sig { params(value: T.nilable(String)).void }
-      def initialize(value)
-        raise MacOSVersionError, value unless /\A1\d+(?:\.\d+){0,2}\Z/.match?(value)
+      sig { params(value: T.nilable(String), arch: T.nilable(String)).void }
+      def initialize(value, arch: nil)
+        version, arch = Version.version_arch(value) if value.present? && arch.nil?
+        version ||= value
+        arch    ||= "intel"
+
+        raise MacOSVersionError, version unless /\A1\d+(?:\.\d+){0,2}\Z/.match?(version)
 
-        super(value)
+        super(version)
 
+        @arch = arch.to_sym
         @comparison_cache = {}
       end
 
diff --git a/Library/Homebrew/test/os/mac/version_spec.rb b/Library/Homebrew/test/os/mac/version_spec.rb
index 9b3f3ed15e..7c835fbdf5 100644
--- a/Library/Homebrew/test/os/mac/version_spec.rb
+++ b/Library/Homebrew/test/os/mac/version_spec.rb
@@ -62,12 +62,44 @@ describe OS::Mac::Version do
         described_class.new("1.2")
       }.to raise_error(MacOSVersionError, 'unknown or unsupported macOS version: "1.2"')
     end
+
+    it "creates a new version from a valid macOS version" do
+      string_version = described_class.new("11")
+      expect(string_version).to eq(:big_sur)
+      expect(string_version.arch).to eq(:intel)
+    end
+
+    it "creates a new version from a valid macOS version with architecture" do
+      string_version = described_class.new("11-arm64")
+      expect(string_version).to eq(:big_sur)
+      expect(string_version.arch).to eq(:arm64)
+    end
+
+    it "creates a new version from a valid macOS version and architecture" do
+      string_version = described_class.new("11", arch: "arm64")
+      expect(string_version).to eq(:big_sur)
+      expect(string_version.arch).to eq(:arm64)
+    end
   end
 
-  specify "#from_symbol" do
-    expect(described_class.from_symbol(:mojave)).to eq(version)
-    expect { described_class.from_symbol(:foo) }
-      .to raise_error(MacOSVersionError, "unknown or unsupported macOS version: :foo")
+  describe "#from_symbol" do
+    it "raises an error if the symbol is not a valid macOS version" do
+      expect {
+        described_class.from_symbol(:foo)
+      }.to raise_error(MacOSVersionError, "unknown or unsupported macOS version: :foo")
+    end
+
+    it "creates a new version from a valid macOS version" do
+      symbol_version = described_class.from_symbol(:mojave)
+      expect(symbol_version).to eq(version)
+      expect(symbol_version.arch).to eq(:intel)
+    end
+
+    it "creates a new version from a valid macOS version with architecture" do
+      symbol_version = described_class.from_symbol(:arm64_big_sur)
+      expect(symbol_version).to eq(:big_sur)
+      expect(symbol_version.arch).to eq(:arm64)
+    end
   end
 
   specify "#pretty_name" do
-- 
GitLab