| 
									
										
										
										
											2024-08-12 10:30:59 +01:00
										 |  |  | # typed: true # rubocop:disable Sorbet/StrictSigil | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  | module OS | 
					
						
							|  |  |  |   module Mac | 
					
						
							|  |  |  |     module Diagnostic | 
					
						
							|  |  |  |       class Volumes | 
					
						
							|  |  |  |         def initialize | 
					
						
							|  |  |  |           @volumes = get_mounts | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def which(path) | 
					
						
							|  |  |  |           vols = get_mounts path | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # no volume found | 
					
						
							|  |  |  |           return -1 if vols.empty? | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           vol_index = @volumes.index(vols[0]) | 
					
						
							|  |  |  |           # volume not found in volume list | 
					
						
							|  |  |  |           return -1 if vol_index.nil? | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           vol_index | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def get_mounts(path = nil) | 
					
						
							|  |  |  |           vols = [] | 
					
						
							|  |  |  |           # get the volume of path, if path is nil returns all volumes | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           args = %w[/bin/df -P] | 
					
						
							|  |  |  |           args << path if path | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           Utils.popen_read(*args) do |io| | 
					
						
							|  |  |  |             io.each_line do |line| | 
					
						
							|  |  |  |               case line.chomp | 
					
						
							|  |  |  |                 # regex matches: /dev/disk0s2   489562928 440803616  48247312    91%    / | 
					
						
							|  |  |  |               when /^.+\s+[0-9]+\s+[0-9]+\s+[0-9]+\s+[0-9]{1,3}%\s+(.+)/ | 
					
						
							|  |  |  |                 vols << Regexp.last_match(1) | 
					
						
							|  |  |  |               end | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  |             end | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           vols | 
					
						
							| 
									
										
										
										
											2020-08-14 04:33:33 +02:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-02-05 12:26:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |       module Checks | 
					
						
							|  |  |  |         extend T::Helpers | 
					
						
							| 
									
										
										
										
											2021-02-05 12:26:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         requires_ancestor { Homebrew::Diagnostic::Checks } | 
					
						
							| 
									
										
										
										
											2021-02-05 12:26:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def fatal_preinstall_checks | 
					
						
							|  |  |  |           checks = %w[
 | 
					
						
							|  |  |  |             check_access_directories | 
					
						
							|  |  |  |           ] | 
					
						
							| 
									
										
										
										
											2019-01-21 19:23:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # We need the developer tools for `codesign`. | 
					
						
							|  |  |  |           checks << "check_for_installed_developer_tools" if ::Hardware::CPU.arm? | 
					
						
							| 
									
										
										
										
											2018-04-07 20:28:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           checks.freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-09-11 12:05:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def fatal_build_from_source_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_xcode_license_approved | 
					
						
							|  |  |  |             check_xcode_minimum_version | 
					
						
							|  |  |  |             check_clt_minimum_version | 
					
						
							|  |  |  |             check_if_xcode_needs_clt_installed | 
					
						
							|  |  |  |             check_if_supported_sdk_available | 
					
						
							|  |  |  |             check_broken_sdks | 
					
						
							|  |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-01-21 12:39:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def fatal_setup_build_environment_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_xcode_minimum_version | 
					
						
							|  |  |  |             check_clt_minimum_version | 
					
						
							|  |  |  |             check_if_supported_sdk_available | 
					
						
							|  |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-07-04 18:35:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def supported_configuration_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_for_unsupported_macos | 
					
						
							|  |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def build_from_source_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_for_installed_developer_tools | 
					
						
							|  |  |  |             check_xcode_up_to_date | 
					
						
							|  |  |  |             check_clt_up_to_date | 
					
						
							|  |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_for_non_prefixed_findutils | 
					
						
							|  |  |  |           findutils = ::Formula["findutils"] | 
					
						
							|  |  |  |           return unless findutils.any_version_installed? | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           gnubin = %W[#{findutils.opt_libexec}/gnubin #{findutils.libexec}/gnubin] | 
					
						
							|  |  |  |           default_names = Tab.for_name("findutils").with? "default-names" | 
					
						
							|  |  |  |           return if !default_names && !paths.intersect?(gnubin) | 
					
						
							| 
									
										
										
										
											2020-11-06 15:11:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Putting non-prefixed findutils in your path can cause python builds to fail. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         rescue FormulaUnavailableError | 
					
						
							|  |  |  |           nil | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-11-12 17:06:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_for_unsupported_macos | 
					
						
							|  |  |  |           return if Homebrew::EnvConfig.developer? | 
					
						
							|  |  |  |           return if ENV["HOMEBREW_INTEGRATION_TEST"] | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  |           tier = 2
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           who = +"We" | 
					
						
							|  |  |  |           what = if OS::Mac.version.prerelease? | 
					
						
							|  |  |  |             "pre-release version" | 
					
						
							|  |  |  |           elsif OS::Mac.version.outdated_release? | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  |             tier = 3
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             who << " (and Apple)" | 
					
						
							|  |  |  |             "old version" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           return if what.blank? | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           who.freeze | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             You are using macOS #{MacOS.version}. | 
					
						
							|  |  |  |             #{who} do not provide support for this #{what}. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier:)} | 
					
						
							| 
									
										
										
										
											2016-08-30 03:54:24 +01:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-13 09:43:16 +08:00
										 |  |  |         def check_for_opencore | 
					
						
							| 
									
										
										
										
											2024-11-13 22:48:45 +08:00
										 |  |  |           return if ::Hardware::CPU.physical_cpu_arm64? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # https://dortania.github.io/OpenCore-Legacy-Patcher/UPDATE.html#checking-oclp-and-opencore-versions | 
					
						
							|  |  |  |           begin | 
					
						
							|  |  |  |             opencore_version = Utils.safe_popen_read("/usr/sbin/nvram", | 
					
						
							| 
									
										
										
										
											2024-11-13 23:43:24 +08:00
										 |  |  |                                                      "4D1FDA02-38C7-4A6A-9CC6-4BCCA8B30102:opencore-version").split[1] | 
					
						
							| 
									
										
										
										
											2024-11-14 00:55:49 +08:00
										 |  |  |             oclp_version = Utils.safe_popen_read("/usr/sbin/nvram", | 
					
						
							| 
									
										
										
										
											2024-11-13 12:23:34 -05:00
										 |  |  |                                                  "4D1FDA02-38C7-4A6A-9CC6-4BCCA8B30102:OCLP-Version").split[1] | 
					
						
							| 
									
										
										
										
											2024-11-14 00:55:49 +08:00
										 |  |  |             return if opencore_version.blank? || oclp_version.blank? | 
					
						
							| 
									
										
										
										
											2024-11-13 22:48:45 +08:00
										 |  |  |           rescue ErrorDuringExecution | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-11-13 09:43:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-21 21:01:44 -04:00
										 |  |  |           oclp_support_tier = ::Hardware::CPU.features.include?(:pclmulqdq) ? 2 : 3
 | 
					
						
							| 
									
										
										
										
											2025-04-12 16:44:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-13 09:43:16 +08:00
										 |  |  |           <<~EOS | 
					
						
							| 
									
										
										
										
											2024-11-13 23:43:24 +08:00
										 |  |  |             You have booted macOS using OpenCore Legacy Patcher. | 
					
						
							| 
									
										
										
										
											2024-11-13 09:43:16 +08:00
										 |  |  |             We do not provide support for this configuration. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-12 16:44:06 -07:00
										 |  |  |             #{support_tier_message(tier: oclp_support_tier)} | 
					
						
							| 
									
										
										
										
											2024-11-13 09:43:16 +08:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_up_to_date | 
					
						
							|  |  |  |           return unless MacOS::Xcode.outdated? | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # avoid duplicate very similar messages | 
					
						
							|  |  |  |           return if MacOS::Xcode.below_minimum_version? | 
					
						
							| 
									
										
										
										
											2024-09-06 17:36:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # CI images are going to end up outdated so don't complain when | 
					
						
							|  |  |  |           # `brew test-bot` runs `brew doctor` in the CI for the Homebrew/brew | 
					
						
							|  |  |  |           # repository. This only needs to support whatever CI providers | 
					
						
							|  |  |  |           # Homebrew/brew is currently using. | 
					
						
							|  |  |  |           return if GitHub::Actions.env_set? | 
					
						
							| 
									
										
										
										
											2017-04-02 17:24:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # With fake El Capitan for Portable Ruby, we are intentionally not using Xcode 8. | 
					
						
							|  |  |  |           # This is because we are not using the CLT and Xcode 8 has the 10.12 SDK. | 
					
						
							|  |  |  |           return if ENV["HOMEBREW_FAKE_MACOS"] | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           message = <<~EOS | 
					
						
							|  |  |  |             Your Xcode (#{MacOS::Xcode.version}) is outdated. | 
					
						
							|  |  |  |             Please update to Xcode #{MacOS::Xcode.latest_version} (or delete it). | 
					
						
							|  |  |  |             #{MacOS::Xcode.update_instructions} | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							| 
									
										
										
										
											2019-02-19 13:12:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           if OS::Mac.version.prerelease? | 
					
						
							|  |  |  |             current_path = Utils.popen_read("/usr/bin/xcode-select", "-p") | 
					
						
							|  |  |  |             message += <<~EOS | 
					
						
							|  |  |  |               If #{MacOS::Xcode.latest_version} is installed, you may need to: | 
					
						
							|  |  |  |                 sudo xcode-select --switch /Applications/Xcode.app | 
					
						
							|  |  |  |               Current developer directory is: | 
					
						
							|  |  |  |                 #{current_path} | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           message | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-11-05 10:36:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_clt_up_to_date | 
					
						
							|  |  |  |           return unless MacOS::CLT.outdated? | 
					
						
							| 
									
										
										
										
											2016-11-05 10:36:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # avoid duplicate very similar messages | 
					
						
							|  |  |  |           return if MacOS::CLT.below_minimum_version? | 
					
						
							| 
									
										
										
										
											2016-11-05 10:36:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           # CI images are going to end up outdated so don't complain when | 
					
						
							|  |  |  |           # `brew test-bot` runs `brew doctor` in the CI for the Homebrew/brew | 
					
						
							|  |  |  |           # repository. This only needs to support whatever CI providers | 
					
						
							|  |  |  |           # Homebrew/brew is currently using. | 
					
						
							|  |  |  |           return if GitHub::Actions.env_set? | 
					
						
							| 
									
										
										
										
											2016-11-05 10:36:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             A newer Command Line Tools release is available. | 
					
						
							|  |  |  |             #{MacOS::CLT.update_instructions} | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_minimum_version | 
					
						
							|  |  |  |           return unless MacOS::Xcode.below_minimum_version? | 
					
						
							| 
									
										
										
										
											2017-11-17 19:53:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           xcode = MacOS::Xcode.version.to_s | 
					
						
							|  |  |  |           xcode += " => #{MacOS::Xcode.prefix}" unless MacOS::Xcode.default_prefix? | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Xcode (#{xcode}) at #{MacOS::Xcode.bundle_path} is too outdated. | 
					
						
							|  |  |  |             Please update to Xcode #{MacOS::Xcode.latest_version} (or delete it). | 
					
						
							|  |  |  |             #{MacOS::Xcode.update_instructions} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_clt_minimum_version | 
					
						
							|  |  |  |           return unless MacOS::CLT.below_minimum_version? | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Command Line Tools are too outdated. | 
					
						
							|  |  |  |             #{MacOS::CLT.update_instructions} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_if_xcode_needs_clt_installed | 
					
						
							|  |  |  |           return unless MacOS::Xcode.needs_clt_installed? | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Xcode alone is not sufficient on #{MacOS.version.pretty_name}. | 
					
						
							|  |  |  |             #{::DevelopmentTools.installation_instructions} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_prefix | 
					
						
							|  |  |  |           prefix = MacOS::Xcode.prefix | 
					
						
							|  |  |  |           return if prefix.nil? | 
					
						
							|  |  |  |           return unless prefix.to_s.include?(" ") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Xcode is installed to a directory with a space in the name. | 
					
						
							|  |  |  |             This will cause some formulae to fail to build. | 
					
						
							|  |  |  |           EOS | 
					
						
							| 
									
										
										
										
											2017-05-27 13:34:59 +01:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_prefix_exists | 
					
						
							|  |  |  |           prefix = MacOS::Xcode.prefix | 
					
						
							|  |  |  |           return if prefix.nil? || prefix.exist? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             The directory Xcode is reportedly installed to doesn't exist: | 
					
						
							|  |  |  |               #{prefix} | 
					
						
							|  |  |  |             You may need to `xcode-select` the proper path if you have moved Xcode. | 
					
						
							|  |  |  |           EOS | 
					
						
							| 
									
										
										
										
											2017-05-27 13:34:59 +01:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_select_path | 
					
						
							|  |  |  |           return if MacOS::CLT.installed? | 
					
						
							|  |  |  |           return unless MacOS::Xcode.installed? | 
					
						
							|  |  |  |           return if File.file?("#{MacOS.active_developer_dir}/usr/bin/xcodebuild") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           path = MacOS::Xcode.bundle_path | 
					
						
							|  |  |  |           path = "/Developer" if path.nil? || !path.directory? | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Xcode is configured with an invalid path. | 
					
						
							|  |  |  |             You should change it to the correct path: | 
					
						
							|  |  |  |               sudo xcode-select --switch #{path} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2017-05-27 13:34:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_xcode_license_approved | 
					
						
							|  |  |  |           # If the user installs Xcode-only, they have to approve the | 
					
						
							|  |  |  |           # license or no "xc*" tool will work. | 
					
						
							|  |  |  |           return unless `/usr/bin/xcrun clang 2>&1`.include?("license") | 
					
						
							|  |  |  |           return if $CHILD_STATUS.success? | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             You have not agreed to the Xcode license. | 
					
						
							|  |  |  |             Agree to the license by opening Xcode.app or running: | 
					
						
							|  |  |  |               sudo xcodebuild -license | 
					
						
							|  |  |  |           EOS | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_filesystem_case_sensitive | 
					
						
							|  |  |  |           dirs_to_check = [ | 
					
						
							|  |  |  |             HOMEBREW_PREFIX, | 
					
						
							|  |  |  |             HOMEBREW_REPOSITORY, | 
					
						
							|  |  |  |             HOMEBREW_CELLAR, | 
					
						
							|  |  |  |             HOMEBREW_TEMP, | 
					
						
							|  |  |  |           ] | 
					
						
							|  |  |  |           case_sensitive_dirs = dirs_to_check.select do |dir| | 
					
						
							|  |  |  |             # We select the dir as being case-sensitive if either the UPCASED or the | 
					
						
							|  |  |  |             # downcased variant is missing. | 
					
						
							|  |  |  |             # Of course, on a case-insensitive fs, both exist because the os reports so. | 
					
						
							|  |  |  |             # In the rare situation when the user has indeed a downcased and an upcased | 
					
						
							|  |  |  |             # dir (e.g. /TMP and /tmp) this check falsely thinks it is case-insensitive | 
					
						
							|  |  |  |             # but we don't care because: 1. there is more than one dir checked, 2. the | 
					
						
							|  |  |  |             # check is not vital and 3. we would have to touch files otherwise. | 
					
						
							|  |  |  |             upcased = Pathname.new(dir.to_s.upcase) | 
					
						
							|  |  |  |             downcased = Pathname.new(dir.to_s.downcase) | 
					
						
							|  |  |  |             dir.exist? && !(upcased.exist? && downcased.exist?) | 
					
						
							| 
									
										
										
										
											2021-06-15 11:51:29 +01:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           return if case_sensitive_dirs.empty? | 
					
						
							| 
									
										
										
										
											2021-06-15 11:51:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           volumes = Volumes.new | 
					
						
							|  |  |  |           case_sensitive_vols = case_sensitive_dirs.map do |case_sensitive_dir| | 
					
						
							|  |  |  |             volumes.get_mounts(case_sensitive_dir) | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           case_sensitive_vols.uniq! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             The filesystem on #{case_sensitive_vols.join(",")} appears to be case-sensitive. | 
					
						
							|  |  |  |             The default macOS filesystem is case-insensitive. Please report any apparent problems. | 
					
						
							|  |  |  |           EOS | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_for_gettext | 
					
						
							|  |  |  |           find_relative_paths("lib/libgettextlib.dylib", | 
					
						
							|  |  |  |                               "lib/libintl.dylib", | 
					
						
							|  |  |  |                               "include/libintl.h") | 
					
						
							|  |  |  |           return if @found.empty? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Our gettext formula will be caught by check_linked_keg_only_brews | 
					
						
							|  |  |  |           gettext = begin | 
					
						
							|  |  |  |             Formulary.factory("gettext") | 
					
						
							|  |  |  |           rescue | 
					
						
							|  |  |  |             nil | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           if gettext&.linked_keg&.directory? | 
					
						
							|  |  |  |             allowlist = ["#{HOMEBREW_CELLAR}/gettext"] | 
					
						
							|  |  |  |             if ::Hardware::CPU.physical_cpu_arm64? | 
					
						
							|  |  |  |               allowlist += %W[
 | 
					
						
							|  |  |  |                 #{HOMEBREW_MACOS_ARM_DEFAULT_PREFIX}/Cellar/gettext | 
					
						
							|  |  |  |                 #{HOMEBREW_DEFAULT_PREFIX}/Cellar/gettext | 
					
						
							|  |  |  |               ] | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             return if @found.all? do |path| | 
					
						
							|  |  |  |               realpath = Pathname.new(path).realpath.to_s | 
					
						
							|  |  |  |               allowlist.any? { |rack| realpath.start_with?(rack) } | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |           inject_file_list @found, <<~EOS | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             gettext files detected at a system prefix. | 
					
						
							|  |  |  |             These files can cause compilation and link failures, especially if they | 
					
						
							|  |  |  |             are compiled with improper architectures. Consider removing these files: | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_for_iconv | 
					
						
							|  |  |  |           find_relative_paths("lib/libiconv.dylib", "include/iconv.h") | 
					
						
							|  |  |  |           return if @found.empty? | 
					
						
							| 
									
										
										
										
											2018-09-17 02:45:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           libiconv = begin | 
					
						
							|  |  |  |             Formulary.factory("libiconv") | 
					
						
							|  |  |  |           rescue | 
					
						
							|  |  |  |             nil | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           if libiconv&.linked_keg&.directory? | 
					
						
							|  |  |  |             unless libiconv&.keg_only? | 
					
						
							|  |  |  |               <<~EOS | 
					
						
							|  |  |  |                 A libiconv formula is installed and linked. | 
					
						
							|  |  |  |                 This will break stuff. For serious. Unlink it. | 
					
						
							|  |  |  |               EOS | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             inject_file_list @found, <<~EOS | 
					
						
							|  |  |  |               libiconv files detected at a system prefix other than /usr.
 | 
					
						
							|  |  |  |               Homebrew doesn't provide a libiconv formula and expects to link against | 
					
						
							|  |  |  |               the system version in /usr. libiconv in other prefixes can cause
 | 
					
						
							|  |  |  |               compile or link failure, especially if compiled with improper | 
					
						
							|  |  |  |               architectures. macOS itself never installs anything to /usr/local so | 
					
						
							|  |  |  |               it was either installed by a user or some other third party software. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               tl;dr: delete these files: | 
					
						
							|  |  |  |             EOS | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_for_multiple_volumes | 
					
						
							|  |  |  |           return unless HOMEBREW_CELLAR.exist? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           volumes = Volumes.new | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # Find the volumes for the TMP folder & HOMEBREW_CELLAR | 
					
						
							|  |  |  |           real_cellar = HOMEBREW_CELLAR.realpath | 
					
						
							|  |  |  |           where_cellar = volumes.which real_cellar | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |           begin | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             tmp = Pathname.new(Dir.mktmpdir("doctor", HOMEBREW_TEMP)) | 
					
						
							|  |  |  |             begin | 
					
						
							|  |  |  |               real_tmp = tmp.realpath.parent | 
					
						
							|  |  |  |               where_tmp = volumes.which real_tmp | 
					
						
							|  |  |  |             ensure | 
					
						
							|  |  |  |               Dir.delete tmp.to_s | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           rescue | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           return if where_cellar == where_tmp | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Cellar and TEMP directories are on different volumes. | 
					
						
							|  |  |  |             macOS won't move relative symlinks across volumes unless the target file already | 
					
						
							| 
									
										
										
										
											2025-07-04 11:25:41 +01:00
										 |  |  |             exists. Formulae known to be affected by this are Git and Narwhal. | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             You should set the "HOMEBREW_TEMP" environment variable to a suitable | 
					
						
							|  |  |  |             directory on the same volume as your Cellar. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-05-08 09:18:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_if_supported_sdk_available | 
					
						
							|  |  |  |           return unless ::DevelopmentTools.installed? | 
					
						
							|  |  |  |           return unless MacOS.sdk_root_needed? | 
					
						
							|  |  |  |           return if MacOS.sdk | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           locator = MacOS.sdk_locator | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           source = if locator.source == :clt | 
					
						
							|  |  |  |             return if MacOS::CLT.below_minimum_version? # Handled by other diagnostics. | 
					
						
							| 
									
										
										
										
											2021-03-30 04:23:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             update_instructions = MacOS::CLT.update_instructions | 
					
						
							|  |  |  |             "Command Line Tools (CLT)" | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             return if MacOS::Xcode.below_minimum_version? # Handled by other diagnostics. | 
					
						
							| 
									
										
										
										
											2021-03-30 04:23:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             update_instructions = MacOS::Xcode.update_instructions | 
					
						
							|  |  |  |             "Xcode" | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2020-07-01 16:02:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your #{source} does not support macOS #{MacOS.version}. | 
					
						
							|  |  |  |             It is either outdated or was modified. | 
					
						
							|  |  |  |             Please update your #{source} or delete it if no updates are available. | 
					
						
							|  |  |  |             #{update_instructions} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         # The CLT 10.x -> 11.x upgrade process on 10.14 contained a bug which broke the SDKs. | 
					
						
							|  |  |  |         # Notably, MacOSX10.14.sdk would indirectly symlink to MacOSX10.15.sdk. | 
					
						
							|  |  |  |         # This diagnostic was introduced to check for this and recommend a full reinstall. | 
					
						
							|  |  |  |         def check_broken_sdks | 
					
						
							|  |  |  |           locator = MacOS.sdk_locator | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           return if locator.all_sdks.all? do |sdk| | 
					
						
							|  |  |  |             path_version = sdk.path.basename.to_s[MacOS::SDK::VERSIONED_SDK_REGEX, 1] | 
					
						
							|  |  |  |             next true if path_version.blank? | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             sdk.version == MacOSVersion.new(path_version).strip_patch | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           if locator.source == :clt | 
					
						
							|  |  |  |             source = "Command Line Tools (CLT)" | 
					
						
							|  |  |  |             path_to_remove = MacOS::CLT::PKG_PATH | 
					
						
							|  |  |  |             installation_instructions = MacOS::CLT.installation_instructions | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             source = "Xcode" | 
					
						
							|  |  |  |             path_to_remove = MacOS::Xcode.bundle_path | 
					
						
							|  |  |  |             installation_instructions = MacOS::Xcode.installation_instructions | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             The contents of the SDKs in your #{source} installation do not match the SDK folder names. | 
					
						
							|  |  |  |             A clean reinstall of #{source} should fix this. | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             Remove the broken installation before reinstalling: | 
					
						
							|  |  |  |               sudo rm -rf #{path_to_remove} | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             #{installation_instructions} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2025-06-21 17:38:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def check_cask_software_versions | 
					
						
							|  |  |  |           super | 
					
						
							|  |  |  |           add_info "macOS", MacOS.full_version | 
					
						
							|  |  |  |           add_info "SIP", begin | 
					
						
							|  |  |  |             csrutil = "/usr/bin/csrutil" | 
					
						
							|  |  |  |             if File.executable?(csrutil) | 
					
						
							|  |  |  |               Open3.capture2(csrutil, "status") | 
					
						
							|  |  |  |                    .first | 
					
						
							|  |  |  |                    .gsub("This is an unsupported configuration, likely to break in " \ | 
					
						
							|  |  |  |                          "the future and leave your machine in an unknown state.", "") | 
					
						
							|  |  |  |                    .gsub("System Integrity Protection status: ", "") | 
					
						
							|  |  |  |                    .delete("\t.") | 
					
						
							|  |  |  |                    .capitalize | 
					
						
							|  |  |  |                    .strip | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               "N/A" | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           nil | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-02-07 04:21:59 +00:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-04-25 17:58:50 +01:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | Homebrew::Diagnostic::Checks.prepend(OS::Mac::Diagnostic::Checks) |