| 
									
										
										
										
											2020-10-10 14:16:11 +02:00
										 |  |  | # typed: false | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "os/mac/version" | 
					
						
							|  |  |  | require "os/mac/xcode" | 
					
						
							|  |  |  | require "os/mac/xquartz" | 
					
						
							| 
									
										
										
										
											2015-12-04 13:58:22 -08:00
										 |  |  | require "os/mac/sdk" | 
					
						
							| 
									
										
										
										
											2017-09-27 02:08:23 -07:00
										 |  |  | require "os/mac/keg" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | module OS | 
					
						
							| 
									
										
										
										
											2020-08-26 03:06:19 +02:00
										 |  |  |   # Helper module for querying system information on macOS. | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |   module Mac | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |     extend T::Sig | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 20:11:54 +02:00
										 |  |  |     module_function | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 21:46:20 +09:00
										 |  |  |     # rubocop:disable Naming/ConstantName | 
					
						
							|  |  |  |     # rubocop:disable Style/MutableConstant | 
					
						
							| 
									
										
										
										
											2020-06-22 16:19:11 +05:30
										 |  |  |     ::MacOS = OS::Mac | 
					
						
							| 
									
										
										
										
											2019-04-19 21:46:20 +09:00
										 |  |  |     # rubocop:enable Naming/ConstantName | 
					
						
							|  |  |  |     # rubocop:enable Style/MutableConstant | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-25 17:53:43 +01:00
										 |  |  |     raise "Loaded OS::Mac on generic OS!" if ENV["HOMEBREW_TEST_GENERIC_OS"] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     # This can be compared to numerics, strings, or symbols | 
					
						
							|  |  |  |     # using the standard Ruby Comparable methods. | 
					
						
							|  |  |  |     def version | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       @version ||= Version.from_symbol(full_version.to_sym) | 
					
						
							| 
									
										
										
										
											2015-10-16 16:41:14 +08:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # This can be compared to numerics, strings, or symbols | 
					
						
							|  |  |  |     # using the standard Ruby Comparable methods. | 
					
						
							|  |  |  |     def full_version | 
					
						
							| 
									
										
										
										
											2016-09-19 09:17:35 +01:00
										 |  |  |       @full_version ||= Version.new((ENV["HOMEBREW_MACOS_VERSION"] || ENV["HOMEBREW_OSX_VERSION"]).chomp) | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 09:00:53 +01:00
										 |  |  |     def full_version=(version) | 
					
						
							|  |  |  |       @full_version = Version.new(version.chomp) | 
					
						
							|  |  |  |       @version = nil | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |     sig { returns(::Version) } | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  |     def latest_sdk_version | 
					
						
							|  |  |  |       # TODO: bump version when new Xcode macOS SDK is released | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       ::Version.new("11.0") | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     private :latest_sdk_version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sig { returns(::Version) } | 
					
						
							|  |  |  |     def sdk_version | 
					
						
							|  |  |  |       full_version.major_minor | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-09 10:21:25 +00:00
										 |  |  |     def outdated_release? | 
					
						
							| 
									
										
										
										
											2018-09-03 17:00:02 +01:00
										 |  |  |       # TODO: bump version when new macOS is released and also update | 
					
						
							| 
									
										
										
										
											2018-09-06 11:48:13 +01:00
										 |  |  |       # references in docs/Installation.md and | 
					
						
							| 
									
										
										
										
											2020-11-03 12:39:26 -05:00
										 |  |  |       # https://github.com/Homebrew/install/blob/HEAD/install.sh | 
					
						
							| 
									
										
										
										
											2020-11-05 23:38:25 +01:00
										 |  |  |       version < "10.14" | 
					
						
							| 
									
										
										
										
											2016-01-09 10:21:25 +00:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  |     def prerelease? | 
					
						
							| 
									
										
										
										
											2020-11-20 16:42:52 +00:00
										 |  |  |       # TODO: bump version when new macOS is released or announced | 
					
						
							|  |  |  |       # and also update references in docs/Installation.md and | 
					
						
							|  |  |  |       # https://github.com/Homebrew/install/blob/HEAD/install.sh | 
					
						
							| 
									
										
										
										
											2020-11-23 10:14:18 +01:00
										 |  |  |       version >= "12" | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 04:24:55 +02:00
										 |  |  |     def languages | 
					
						
							| 
									
										
										
										
											2020-07-08 11:03:56 +01:00
										 |  |  |       return @languages if @languages | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       os_langs = Utils.popen_read("defaults", "read", "-g", "AppleLanguages") | 
					
						
							|  |  |  |       if os_langs.blank? | 
					
						
							|  |  |  |         # User settings don't exist so check the system-wide one. | 
					
						
							|  |  |  |         os_langs = Utils.popen_read("defaults", "read", "/Library/Preferences/.GlobalPreferences", "AppleLanguages") | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       os_langs = os_langs.scan(/[^ \n"(),]+/) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-21 21:28:58 +02:00
										 |  |  |       @languages = os_langs | 
					
						
							| 
									
										
										
										
											2016-09-10 04:24:55 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 00:11:43 +02:00
										 |  |  |     def language | 
					
						
							| 
									
										
										
										
											2016-09-10 04:24:55 +02:00
										 |  |  |       languages.first | 
					
						
							| 
									
										
										
										
											2016-09-09 00:11:43 +02:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 18:36:46 -05:00
										 |  |  |     def active_developer_dir | 
					
						
							| 
									
										
										
										
											2014-12-16 15:27:36 -05:00
										 |  |  |       @active_developer_dir ||= Utils.popen_read("/usr/bin/xcode-select", "-print-path").strip | 
					
						
							| 
									
										
										
										
											2014-05-01 18:36:46 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 12:03:48 +02:00
										 |  |  |     sig { returns(T::Boolean) } | 
					
						
							| 
									
										
										
										
											2020-04-07 16:43:59 +01:00
										 |  |  |     def sdk_root_needed? | 
					
						
							|  |  |  |       if MacOS::CLT.installed? | 
					
						
							|  |  |  |         # If there's no CLT SDK, return false | 
					
						
							|  |  |  |         return false unless MacOS::CLT.provides_sdk? | 
					
						
							|  |  |  |         # If the CLT is installed and headers are provided by the system, return false | 
					
						
							|  |  |  |         return false unless MacOS::CLT.separate_header_package? | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       true | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-18 21:42:43 -04:00
										 |  |  |     # If a specific SDK is requested: | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     #   1. The requested SDK is returned, if it's installed. | 
					
						
							|  |  |  |     #   2. If the requested SDK is not installed, the newest SDK (if any SDKs | 
					
						
							| 
									
										
										
										
											2016-03-30 05:13:41 -07:00
										 |  |  |     #      are available) is returned. | 
					
						
							| 
									
										
										
										
											2018-10-18 21:42:43 -04:00
										 |  |  |     #   3. If no SDKs are available, nil is returned. | 
					
						
							|  |  |  |     # | 
					
						
							| 
									
										
										
										
											2020-03-08 20:08:33 +00:00
										 |  |  |     # If no specific SDK is requested, the SDK matching the OS version is returned, | 
					
						
							|  |  |  |     # if available. Otherwise, the latest SDK is returned. | 
					
						
							| 
									
										
										
										
											2016-03-30 05:13:41 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  |     def sdk_locator | 
					
						
							|  |  |  |       if CLT.installed? && CLT.provides_sdk? | 
					
						
							|  |  |  |         CLT.sdk_locator | 
					
						
							| 
									
										
										
										
											2020-03-08 20:09:06 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  |         Xcode.sdk_locator | 
					
						
							| 
									
										
										
										
											2018-06-12 14:55:31 -07:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-06-12 14:55:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  |     def sdk(v = nil) | 
					
						
							|  |  |  |       sdk_locator.sdk_if_applicable(v) | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:43:59 +01:00
										 |  |  |     def sdk_for_formula(f, v = nil) | 
					
						
							|  |  |  |       # If the formula requires Xcode, don't return the CLT SDK | 
					
						
							|  |  |  |       return Xcode.sdk if f.requirements.any? { |req| req.is_a? XcodeRequirement } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       sdk(v) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-05 17:17:03 -05:00
										 |  |  |     # Returns the path to an SDK or nil, following the rules set by {sdk}. | 
					
						
							| 
									
										
										
										
											2016-03-23 04:34:21 -07:00
										 |  |  |     def sdk_path(v = nil) | 
					
						
							| 
									
										
										
										
											2015-12-04 13:58:22 -08:00
										 |  |  |       s = sdk(v) | 
					
						
							| 
									
										
										
										
											2017-09-24 19:24:46 +01:00
										 |  |  |       s&.path | 
					
						
							| 
									
										
										
										
											2015-12-04 13:58:22 -08:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-24 09:32:55 -07:00
										 |  |  |     def sdk_path_if_needed(v = nil) | 
					
						
							| 
									
										
										
										
											2020-03-08 20:09:06 +00:00
										 |  |  |       # Prefer CLT SDK when both Xcode and the CLT are installed. | 
					
						
							| 
									
										
										
										
											2018-08-07 17:56:42 -07:00
										 |  |  |       # Expected results: | 
					
						
							|  |  |  |       # 1. On Xcode-only systems, return the Xcode SDK. | 
					
						
							|  |  |  |       # 2. On Xcode-and-CLT systems where headers are provided by the system, return nil. | 
					
						
							|  |  |  |       # 3. On CLT-only systems with no CLT SDK, return nil. | 
					
						
							|  |  |  |       # 4. On CLT-only systems with a CLT SDK, where headers are provided by the system, return nil. | 
					
						
							|  |  |  |       # 5. On CLT-only systems with a CLT SDK, where headers are not provided by the system, return the CLT SDK. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:43:59 +01:00
										 |  |  |       return unless sdk_root_needed? | 
					
						
							| 
									
										
										
										
											2018-08-07 17:56:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-24 09:32:55 -07:00
										 |  |  |       sdk_path(v) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     # See these issues for some history: | 
					
						
							| 
									
										
										
										
											2018-10-18 21:42:43 -04:00
										 |  |  |     # | 
					
						
							| 
									
										
										
										
											2020-11-03 12:39:26 -05:00
										 |  |  |     # - {https://github.com/Homebrew/legacy-homebrew/issues/13} | 
					
						
							|  |  |  |     # - {https://github.com/Homebrew/legacy-homebrew/issues/41} | 
					
						
							|  |  |  |     # - {https://github.com/Homebrew/legacy-homebrew/issues/48} | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     def macports_or_fink | 
					
						
							|  |  |  |       paths = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       # First look in the path because MacPorts is relocatable and Fink | 
					
						
							|  |  |  |       # may become relocatable in the future. | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |       %w[port fink].each do |ponk| | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         path = which(ponk) | 
					
						
							|  |  |  |         paths << path unless path.nil? | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       # Look in the standard locations, because even if port or fink are | 
					
						
							|  |  |  |       # not in the path they can still break builds if the build scripts | 
					
						
							|  |  |  |       # have these paths baked in. | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |       %w[/sw/bin/fink /opt/local/bin/port].each do |ponk| | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         path = Pathname.new(ponk) | 
					
						
							|  |  |  |         paths << path if path.exist? | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-10 16:31:56 +00:00
										 |  |  |       # Finally, some users make their MacPorts or Fink directories | 
					
						
							| 
									
										
										
										
											2017-09-10 16:32:09 +00:00
										 |  |  |       # read-only in order to try out Homebrew, but this doesn't work as | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       # some build scripts error out when trying to read from these now | 
					
						
							|  |  |  |       # unreadable paths. | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |       %w[/sw /opt/local].map { |p| Pathname.new(p) }.each do |path| | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         paths << path if path.exist? && !path.readable? | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       paths.uniq | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-01 20:47:26 -05:00
										 |  |  |     def app_with_bundle_id(*ids) | 
					
						
							| 
									
										
										
										
											2017-10-05 21:47:26 +01:00
										 |  |  |       path = mdfind(*ids) | 
					
						
							|  |  |  |              .reject { |p| p.include?("/Backups.backupdb/") } | 
					
						
							|  |  |  |              .first | 
					
						
							| 
									
										
										
										
											2020-12-01 17:04:59 +00:00
										 |  |  |       Pathname.new(path) if path.present? | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-01 20:47:26 -05:00
										 |  |  |     def mdfind(*ids) | 
					
						
							|  |  |  |       (@mdfind ||= {}).fetch(ids) do | 
					
						
							| 
									
										
										
										
											2014-07-11 15:51:19 -05:00
										 |  |  |         @mdfind[ids] = Utils.popen_read("/usr/bin/mdfind", mdfind_query(*ids)).split("\n") | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-01 20:47:26 -05:00
										 |  |  |     def pkgutil_info(id) | 
					
						
							| 
									
										
										
										
											2014-04-01 20:49:11 -05:00
										 |  |  |       (@pkginfo ||= {}).fetch(id) do |key| | 
					
						
							| 
									
										
										
										
											2014-07-11 15:51:19 -05:00
										 |  |  |         @pkginfo[key] = Utils.popen_read("/usr/sbin/pkgutil", "--pkg-info", key).strip | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2014-04-01 20:47:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def mdfind_query(*ids) | 
					
						
							|  |  |  |       ids.map! { |id| "kMDItemCFBundleIdentifier == #{id}" }.join(" || ") | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | end |