| 
									
										
										
										
											2020-10-10 14:16:11 +02:00
										 |  |  | # typed: true | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  | require "exceptions" | 
					
						
							| 
									
										
										
										
											2020-06-23 14:11:05 +01:00
										 |  |  | require "hardware" | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "version" | 
					
						
							| 
									
										
										
										
											2013-02-06 22:49:43 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | module OS | 
					
						
							|  |  |  |   module Mac | 
					
						
							| 
									
										
										
										
											2020-08-25 00:35:07 +02:00
										 |  |  |     # A macOS version. | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     # @api private | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     class Version < ::Version | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       extend T::Sig | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 10:32:34 -08:00
										 |  |  |       sig { returns(Symbol) } | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |       attr_reader :arch | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       SYMBOLS = { | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |         big_sur:     "11", | 
					
						
							| 
									
										
										
										
											2019-06-03 11:45:08 -07:00
										 |  |  |         catalina:    "10.15", | 
					
						
							| 
									
										
										
										
											2019-01-26 17:13:14 +00:00
										 |  |  |         mojave:      "10.14", | 
					
						
							|  |  |  |         high_sierra: "10.13", | 
					
						
							|  |  |  |         sierra:      "10.12", | 
					
						
							|  |  |  |         el_capitan:  "10.11", | 
					
						
							|  |  |  |         yosemite:    "10.10", | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |       }.freeze | 
					
						
							| 
									
										
										
										
											2013-06-15 19:39:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       sig { params(sym: Symbol).returns(T.attached_class) } | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def self.from_symbol(sym) | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         version, arch = version_arch(sym) | 
					
						
							|  |  |  |         version ||= sym | 
					
						
							|  |  |  |         str = SYMBOLS.fetch(version.to_sym) { raise MacOSVersionError, sym } | 
					
						
							|  |  |  |         new(str, arch: arch) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 10:32:34 -08:00
										 |  |  |       sig { params(value: T.any(String, Symbol)).returns(T.any([], [String, T.nilable(String)])) } | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |       def self.version_arch(value) | 
					
						
							|  |  |  |         @all_archs_regex ||= begin | 
					
						
							|  |  |  |           all_archs = Hardware::CPU::ALL_ARCHS.map(&:to_s) | 
					
						
							| 
									
										
										
										
											2021-01-18 10:26:31 -08:00
										 |  |  |           /
 | 
					
						
							|  |  |  |             ^((?<prefix_arch>#{Regexp.union(all_archs)})_)? | 
					
						
							|  |  |  |             (?<version>[\w.]+) | 
					
						
							|  |  |  |             (-(?<suffix_arch>#{Regexp.union(all_archs)}))?$ | 
					
						
							|  |  |  |           /x
 | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-01-20 10:32:34 -08:00
										 |  |  |         match = @all_archs_regex.match(value.to_s) | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         return [] unless match | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-18 10:26:31 -08:00
										 |  |  |         version = match[:version] | 
					
						
							|  |  |  |         arch = match[:prefix_arch] || match[:suffix_arch] | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         [version, arch] | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-06-15 19:39:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |       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) | 
					
						
							| 
									
										
										
										
											2020-10-05 14:46:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         super(version) | 
					
						
							| 
									
										
										
										
											2020-10-05 14:46:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 11:55:27 +00:00
										 |  |  |         @arch = arch.to_sym | 
					
						
							| 
									
										
										
										
											2014-04-02 20:29:20 -05:00
										 |  |  |         @comparison_cache = {} | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 10:32:34 -08:00
										 |  |  |       sig { override.params(other: T.untyped).returns(T.nilable(Integer)) } | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def <=>(other) | 
					
						
							| 
									
										
										
										
											2014-04-02 20:29:20 -05:00
										 |  |  |         @comparison_cache.fetch(other) do | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |           if SYMBOLS.key?(other) && to_sym == other | 
					
						
							|  |  |  |             0
 | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             v = SYMBOLS.fetch(other) { other.to_s } | 
					
						
							|  |  |  |             @comparison_cache[other] = super(::Version.new(v)) | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2014-04-02 20:29:20 -05:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-06-14 11:56:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-09 18:34:47 +00:00
										 |  |  |       sig { returns(T.self_type) } | 
					
						
							|  |  |  |       def strip_patch | 
					
						
							|  |  |  |         # Big Sur is 11.x but Catalina is 10.15.x. | 
					
						
							|  |  |  |         if major >= 11
 | 
					
						
							|  |  |  |           self.class.new(major.to_s) | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           major_minor | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       sig { returns(Symbol) } | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def to_sym | 
					
						
							| 
									
										
										
										
											2021-02-09 18:34:47 +00:00
										 |  |  |         @to_sym ||= SYMBOLS.invert.fetch(strip_patch.to_s, :dunno) | 
					
						
							| 
									
										
										
										
											2013-06-14 14:47:16 -07:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-06-15 19:40:42 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       sig { returns(String) } | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def pretty_name | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |         @pretty_name ||= to_sym.to_s.split("_").map(&:capitalize).join(" ").freeze | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-01-28 19:31:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |       # For {OS::Mac::Version} compatibility. | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       sig { returns(T::Boolean) } | 
					
						
							| 
									
										
										
										
											2019-01-28 19:31:21 +00:00
										 |  |  |       def requires_nehalem_cpu? | 
					
						
							| 
									
										
										
										
											2020-07-08 20:53:10 +02:00
										 |  |  |         unless Hardware::CPU.intel? | 
					
						
							|  |  |  |           raise "Unexpected architecture: #{Hardware::CPU.arch}. This only works with Intel architecture." | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-28 19:31:21 +00:00
										 |  |  |         Hardware.oldest_cpu(self) == :nehalem | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       # https://en.wikipedia.org/wiki/Nehalem_(microarchitecture) | 
					
						
							|  |  |  |       # Ensure any extra methods are also added to version/null.rb | 
					
						
							|  |  |  |       alias requires_sse4? requires_nehalem_cpu? | 
					
						
							|  |  |  |       alias requires_sse41? requires_nehalem_cpu? | 
					
						
							|  |  |  |       alias requires_sse42? requires_nehalem_cpu? | 
					
						
							|  |  |  |       alias requires_popcnt? requires_nehalem_cpu? | 
					
						
							| 
									
										
										
										
											2013-06-15 19:40:42 -05:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-02-06 22:49:43 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | end |