| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | require "tempfile" | 
					
						
							|  |  |  | require "utils/shell" | 
					
						
							| 
									
										
										
										
											2020-03-13 11:39:27 +11:00
										 |  |  | require "hardware" | 
					
						
							| 
									
										
										
										
											2019-01-01 15:51:32 -08:00
										 |  |  | require "os/linux/glibc" | 
					
						
							|  |  |  | require "os/linux/kernel" | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  | module OS | 
					
						
							|  |  |  |   module Linux | 
					
						
							|  |  |  |     module Diagnostic | 
					
						
							| 
									
										
										
										
											2025-08-12 11:00:14 +01:00
										 |  |  |       # Linux-specific diagnostic checks for Homebrew. | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |       module Checks | 
					
						
							|  |  |  |         extend T::Helpers | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         requires_ancestor { Homebrew::Diagnostic::Checks } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def fatal_preinstall_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_access_directories | 
					
						
							|  |  |  |             check_linuxbrew_core | 
					
						
							|  |  |  |             check_linuxbrew_bottle_domain | 
					
						
							|  |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-10-20 11:01:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def supported_configuration_checks | 
					
						
							|  |  |  |           %w[
 | 
					
						
							|  |  |  |             check_glibc_minimum_version | 
					
						
							|  |  |  |             check_kernel_minimum_version | 
					
						
							|  |  |  |             check_supported_architecture | 
					
						
							| 
									
										
										
										
											2025-08-12 11:00:14 +01:00
										 |  |  |             check_for_symlinked_home | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           ].freeze | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-01-21 12:39:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_tmpdir_sticky_bit | 
					
						
							| 
									
										
										
										
											2025-06-13 16:59:10 +01:00
										 |  |  |           message = super | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           return if message.nil? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           message + <<~EOS | 
					
						
							|  |  |  |             If you don't have administrative privileges on this machine, | 
					
						
							| 
									
										
										
										
											2025-08-02 23:27:59 -04:00
										 |  |  |             create a directory and set the `$HOMEBREW_TEMP` environment variable, | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             for example: | 
					
						
							|  |  |  |               install -d -m 1755 ~/tmp
 | 
					
						
							|  |  |  |               #{Utils::Shell.set_variable_in_profile("HOMEBREW_TEMP", "~/tmp")} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_tmpdir_executable | 
					
						
							|  |  |  |           f = Tempfile.new(%w[homebrew_check_tmpdir_executable .sh], HOMEBREW_TEMP) | 
					
						
							|  |  |  |           f.write "#!/bin/sh\n" | 
					
						
							|  |  |  |           f.chmod 0700
 | 
					
						
							|  |  |  |           f.close | 
					
						
							|  |  |  |           return if system T.must(f.path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             The directory #{HOMEBREW_TEMP} does not permit executing | 
					
						
							| 
									
										
										
										
											2025-08-02 23:27:59 -04:00
										 |  |  |             programs. It is likely mounted as "noexec". Please set `$HOMEBREW_TEMP` | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             in your #{Utils::Shell.profile} to a different directory, for example: | 
					
						
							|  |  |  |               export HOMEBREW_TEMP=~/tmp
 | 
					
						
							|  |  |  |               echo 'export HOMEBREW_TEMP=~/tmp' >> #{Utils::Shell.profile} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         ensure | 
					
						
							|  |  |  |           f&.unlink | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2018-08-14 11:51:18 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_umask_not_zero | 
					
						
							|  |  |  |           return unless File.umask.zero? | 
					
						
							| 
									
										
										
										
											2018-11-02 15:03:54 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             umask is currently set to 000. Directories created by Homebrew cannot | 
					
						
							|  |  |  |             be world-writable. This issue can be resolved by adding "umask 002" to | 
					
						
							|  |  |  |             your #{Utils::Shell.profile}: | 
					
						
							|  |  |  |               echo 'umask 002' >> #{Utils::Shell.profile} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-01-01 15:51:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_supported_architecture | 
					
						
							| 
									
										
										
										
											2025-06-13 16:59:10 +01:00
										 |  |  |           return if ::Hardware::CPU.intel? | 
					
						
							|  |  |  |           return if Homebrew::EnvConfig.developer? && ENV["HOMEBREW_ARM64_TESTING"].present? && ::Hardware::CPU.arm? | 
					
						
							| 
									
										
										
										
											2020-03-13 11:39:27 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							| 
									
										
										
										
											2025-06-13 16:59:10 +01:00
										 |  |  |             Your CPU architecture (#{::Hardware::CPU.arch}) is not supported. We only support | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             x86_64 CPU architectures. You will be unable to use binary packages (bottles). | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2020-03-13 11:39:27 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_glibc_minimum_version | 
					
						
							|  |  |  |           return unless OS::Linux::Glibc.below_minimum_version? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your system glibc #{OS::Linux::Glibc.system_version} is too old. | 
					
						
							|  |  |  |             We only support glibc #{OS::Linux::Glibc.minimum_version} or later. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             We recommend updating to a newer version via your distribution's | 
					
						
							|  |  |  |             package manager, upgrading your distribution to the latest version, | 
					
						
							|  |  |  |             or changing distributions. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: :unsupported)} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def check_glibc_version | 
					
						
							|  |  |  |           return unless OS::Linux::Glibc.below_ci_version? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           # We want to bypass this check in some tests. | 
					
						
							|  |  |  |           return if ENV["HOMEBREW_GLIBC_TESTING"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your system glibc #{OS::Linux::Glibc.system_version} is too old. | 
					
						
							|  |  |  |             We will need to automatically install a newer version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             We recommend updating to a newer version via your distribution's | 
					
						
							|  |  |  |             package manager, upgrading your distribution to the latest version, | 
					
						
							|  |  |  |             or changing distributions. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2019-01-01 15:51:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_kernel_minimum_version | 
					
						
							|  |  |  |           return unless OS::Linux::Kernel.below_minimum_version? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Linux kernel #{OS.kernel_version} is too old. | 
					
						
							|  |  |  |             We only support kernel #{OS::Linux::Kernel.minimum_version} or later. | 
					
						
							|  |  |  |             You will be unable to use binary packages (bottles). | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             We recommend updating to a newer version via your distribution's | 
					
						
							|  |  |  |             package manager, upgrading your distribution to the latest version, | 
					
						
							|  |  |  |             or changing distributions. | 
					
						
							| 
									
										
										
										
											2025-03-31 09:53:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 3)} | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-10-20 11:01:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_linuxbrew_core | 
					
						
							|  |  |  |           return unless Homebrew::EnvConfig.no_install_from_api? | 
					
						
							|  |  |  |           return unless CoreTap.instance.linuxbrew_core? | 
					
						
							| 
									
										
										
										
											2021-10-20 11:01:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your Linux core repository is still linuxbrew-core. | 
					
						
							|  |  |  |             You must `brew update` to update to homebrew-core. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-10-21 15:01:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_linuxbrew_bottle_domain | 
					
						
							|  |  |  |           return unless Homebrew::EnvConfig.bottle_domain.include?("linuxbrew") | 
					
						
							| 
									
										
										
										
											2021-10-21 15:01:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           <<~EOS | 
					
						
							| 
									
										
										
										
											2025-08-02 23:27:59 -04:00
										 |  |  |             Your `$HOMEBREW_BOTTLE_DOMAIN` still contains "linuxbrew". | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             You must unset it (or adjust it to not contain linuxbrew | 
					
						
							|  |  |  |             e.g. by using homebrew instead). | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-12 11:00:14 +01:00
										 |  |  |         def check_for_symlinked_home | 
					
						
							|  |  |  |           return unless File.symlink?("/home") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           <<~EOS | 
					
						
							|  |  |  |             Your /home directory is a symlink.
 | 
					
						
							|  |  |  |             This is known to cause issues with formula linking, particularly when installing | 
					
						
							|  |  |  |             multiple formulae that create symlinks in shared directories. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             While this may be a standard directory structure in some distributions | 
					
						
							|  |  |  |             (e.g. Fedora Silverblue) there are known issues as-is. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             If you encounter linking issues, you may need to manually create conflicting | 
					
						
							|  |  |  |             directories or use `brew link --overwrite` as a workaround. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             We'd welcome a PR to fix this functionality. | 
					
						
							|  |  |  |             See https://github.com/Homebrew/brew/issues/18036 for more context. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             #{support_tier_message(tier: 2)} | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |         def check_gcc_dependent_linkage | 
					
						
							|  |  |  |           gcc_dependents = ::Formula.installed.select do |formula| | 
					
						
							|  |  |  |             next false unless formula.tap&.core_tap? | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             # FIXME: This includes formulae that have no runtime dependency on GCC. | 
					
						
							|  |  |  |             formula.recursive_dependencies.map(&:name).include? "gcc" | 
					
						
							|  |  |  |           rescue TapFormulaUnavailableError | 
					
						
							|  |  |  |             false | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           return if gcc_dependents.empty? | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           badly_linked = gcc_dependents.select do |dependent| | 
					
						
							| 
									
										
										
										
											2025-03-17 15:44:16 +01:00
										 |  |  |             dependent_prefix = dependent.any_installed_prefix | 
					
						
							|  |  |  |             # Keg.new() may raise an error if it is not a directory. | 
					
						
							|  |  |  |             # As the result `brew doctor` may display `Error: <keg> is not a directory` | 
					
						
							|  |  |  |             # instead of proper `doctor` information. | 
					
						
							|  |  |  |             # There are other checks that test that, we can skip broken kegs. | 
					
						
							|  |  |  |             next if dependent_prefix.nil? || !dependent_prefix.exist? || !dependent_prefix.directory? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             keg = Keg.new(dependent_prefix) | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |             keg.binary_executable_or_library_files.any? do |binary| | 
					
						
							|  |  |  |               paths = binary.rpaths | 
					
						
							|  |  |  |               versioned_linkage = paths.any? { |path| path.match?(%r{lib/gcc/\d+$}) } | 
					
						
							|  |  |  |               unversioned_linkage = paths.any? { |path| path.match?(%r{lib/gcc/current$}) } | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |               versioned_linkage && !unversioned_linkage | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           return if badly_linked.empty? | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  |           inject_file_list badly_linked, <<~EOS | 
					
						
							|  |  |  |             Formulae which link to GCC through a versioned path were found. These formulae | 
					
						
							|  |  |  |             are prone to breaking when GCC is updated. You should `brew reinstall` these formulae: | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2025-06-21 17:38:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def check_cask_software_versions | 
					
						
							|  |  |  |           super | 
					
						
							|  |  |  |           add_info "Linux", OS::Linux.os_version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           nil | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2022-08-03 20:54:16 +08:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2018-05-12 11:47:12 -05:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							| 
									
										
										
										
											2024-09-21 12:24:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | Homebrew::Diagnostic::Checks.prepend(OS::Linux::Diagnostic::Checks) |