2023-05-09 02:15:28 +02:00
|
|
|
# typed: true
|
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "version"
|
|
|
|
|
|
|
|
# A macOS version.
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
class MacOSVersion < Version
|
|
|
|
# Raised when a macOS version is unsupported.
|
|
|
|
class Error < RuntimeError
|
|
|
|
sig { returns(T.nilable(T.any(String, Symbol))) }
|
|
|
|
attr_reader :version
|
|
|
|
|
|
|
|
def initialize(version)
|
|
|
|
@version = version
|
|
|
|
super "unknown or unsupported macOS version: #{version.inspect}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# NOTE: When removing symbols here, ensure that they are added
|
|
|
|
# to `DEPRECATED_MACOS_VERSIONS` in `MacOSRequirement`.
|
|
|
|
SYMBOLS = {
|
2023-06-05 18:36:03 +01:00
|
|
|
sonoma: "14",
|
2023-05-09 02:15:28 +02:00
|
|
|
ventura: "13",
|
|
|
|
monterey: "12",
|
|
|
|
big_sur: "11",
|
|
|
|
catalina: "10.15",
|
|
|
|
mojave: "10.14",
|
|
|
|
high_sierra: "10.13",
|
|
|
|
sierra: "10.12",
|
|
|
|
el_capitan: "10.11",
|
|
|
|
}.freeze
|
|
|
|
|
|
|
|
sig { params(version: Symbol).returns(T.attached_class) }
|
|
|
|
def self.from_symbol(version)
|
|
|
|
str = SYMBOLS.fetch(version) { raise MacOSVersion::Error, version }
|
|
|
|
new(str)
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { params(version: T.nilable(String)).void }
|
|
|
|
def initialize(version)
|
|
|
|
raise MacOSVersion::Error, version unless /\A1\d+(?:\.\d+){0,2}\Z/.match?(version)
|
|
|
|
|
|
|
|
super(version)
|
|
|
|
|
|
|
|
@comparison_cache = {}
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { override.params(other: T.untyped).returns(T.nilable(Integer)) }
|
|
|
|
def <=>(other)
|
|
|
|
return @comparison_cache[other] if @comparison_cache.key?(other)
|
|
|
|
|
|
|
|
result = case other
|
|
|
|
when Symbol
|
|
|
|
if SYMBOLS.key?(other) && to_sym == other
|
|
|
|
0
|
|
|
|
else
|
|
|
|
v = SYMBOLS.fetch(other) { other.to_s }
|
|
|
|
super(v)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
super
|
|
|
|
end
|
|
|
|
|
|
|
|
@comparison_cache[other] = result unless frozen?
|
|
|
|
|
|
|
|
result
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(T.self_type) }
|
|
|
|
def strip_patch
|
|
|
|
return self if null?
|
|
|
|
|
|
|
|
# Big Sur is 11.x but Catalina is 10.15.x.
|
|
|
|
if T.must(major) >= 11
|
|
|
|
self.class.new(major.to_s)
|
|
|
|
else
|
|
|
|
major_minor
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(Symbol) }
|
|
|
|
def to_sym
|
|
|
|
return @sym if defined?(@sym)
|
|
|
|
|
|
|
|
sym = SYMBOLS.invert.fetch(strip_patch.to_s, :dunno)
|
|
|
|
|
|
|
|
@sym = sym unless frozen?
|
|
|
|
|
|
|
|
sym
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(String) }
|
|
|
|
def pretty_name
|
|
|
|
return @pretty_name if defined?(@pretty_name)
|
|
|
|
|
|
|
|
pretty_name = to_sym.to_s.split("_").map(&:capitalize).join(" ").freeze
|
|
|
|
|
|
|
|
@pretty_name = pretty_name unless frozen?
|
|
|
|
|
|
|
|
pretty_name
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def outdated_release?
|
|
|
|
self < HOMEBREW_MACOS_OLDEST_SUPPORTED
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def prerelease?
|
|
|
|
self >= HOMEBREW_MACOS_NEWEST_UNSUPPORTED
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def unsupported_release?
|
|
|
|
outdated_release? || prerelease?
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def requires_nehalem_cpu?
|
|
|
|
return false if null?
|
|
|
|
|
|
|
|
require "hardware"
|
|
|
|
|
|
|
|
return Hardware.oldest_cpu(self) == :nehalem if Hardware::CPU.intel?
|
|
|
|
|
|
|
|
raise ArgumentError, "Unexpected architecture: #{Hardware::CPU.arch}. This only works with Intel architecture."
|
|
|
|
end
|
|
|
|
# https://en.wikipedia.org/wiki/Nehalem_(microarchitecture)
|
|
|
|
alias requires_sse4? requires_nehalem_cpu?
|
|
|
|
alias requires_sse41? requires_nehalem_cpu?
|
|
|
|
alias requires_sse42? requires_nehalem_cpu?
|
|
|
|
alias requires_popcnt? requires_nehalem_cpu?
|
|
|
|
|
|
|
|
# Represents the absence of a version.
|
|
|
|
# NOTE: Constructor needs to called with an arbitrary macOS-like version which is then set to `nil`.
|
|
|
|
NULL = MacOSVersion.new("10.0").tap { |v| v.instance_variable_set(:@version, nil) }.freeze
|
|
|
|
end
|
|
|
|
|
|
|
|
require "lazy_object"
|
|
|
|
|
2023-05-23 21:43:33 +02:00
|
|
|
# `LazyObject` does not work for exceptions when used in `rescue` statements.
|
|
|
|
class Object
|
|
|
|
class << self
|
|
|
|
module MacOSVersionErrorCompat
|
|
|
|
def const_missing(name)
|
|
|
|
if name == :MacOSVersionError
|
|
|
|
# odeprecated "MacOSVersionError", "MacOSVersion::Error"
|
|
|
|
return MacOSVersion::Error
|
|
|
|
end
|
|
|
|
|
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
|
|
|
private_constant :MacOSVersionErrorCompat
|
|
|
|
|
|
|
|
prepend MacOSVersionErrorCompat
|
|
|
|
end
|
2023-05-09 02:15:28 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
module MacOSVersions
|
|
|
|
SYMBOLS = LazyObject.new do # rubocop:disable Style/MutableConstant
|
|
|
|
# odeprecated "MacOSVersions::SYMBOLS", "MacOSVersion::SYMBOLS"
|
|
|
|
MacOSVersion::SYMBOLS
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
module OS
|
|
|
|
module Mac
|
|
|
|
# TODO: Replace `::Version` with `Version` when this is removed.
|
|
|
|
Version = LazyObject.new do # rubocop:disable Style/MutableConstant
|
|
|
|
# odeprecated "OS::Mac::Version", "MacOSVersion"
|
|
|
|
MacOSVersion
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|