| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | module OS | 
					
						
							|  |  |  |   module Mac | 
					
						
							|  |  |  |     module Xcode | 
					
						
							| 
									
										
										
										
											2016-09-24 20:11:54 +02:00
										 |  |  |       module_function | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |       V4_BUNDLE_ID = "com.apple.dt.Xcode".freeze | 
					
						
							|  |  |  |       V3_BUNDLE_ID = "com.apple.Xcode".freeze | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       def latest_version | 
					
						
							|  |  |  |         case MacOS.version | 
					
						
							| 
									
										
										
										
											2014-08-09 22:00:06 -05:00
										 |  |  |         when "10.4"  then "2.5" | 
					
						
							|  |  |  |         when "10.5"  then "3.1.4" | 
					
						
							|  |  |  |         when "10.6"  then "3.2.6" | 
					
						
							|  |  |  |         when "10.7"  then "4.6.3" | 
					
						
							|  |  |  |         when "10.8"  then "5.1.1" | 
					
						
							| 
									
										
										
										
											2015-03-10 00:08:43 +00:00
										 |  |  |         when "10.9"  then "6.2" | 
					
						
							| 
									
										
										
										
											2016-02-07 19:30:51 +00:00
										 |  |  |         when "10.10" then "7.2.1" | 
					
						
							| 
									
										
										
										
											2016-10-28 23:34:12 +02:00
										 |  |  |         when "10.11" then "8.1" | 
					
						
							|  |  |  |         when "10.12" then "8.1" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2016-09-23 22:02:23 +02:00
										 |  |  |           raise "macOS '#{MacOS.version}' is invalid" unless OS::Mac.prerelease? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 19:57:19 +01:00
										 |  |  |           # Default to newest known version of Xcode for unreleased macOS versions. | 
					
						
							| 
									
										
										
										
											2016-10-28 23:34:12 +02:00
										 |  |  |           "8.1" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2013-06-04 13:58:08 +01:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-08-17 17:18:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-16 01:26:50 +01:00
										 |  |  |       def prerelease? | 
					
						
							| 
									
										
										
										
											2016-10-27 23:35:52 +02:00
										 |  |  |         # TODO: bump to version >= "8.3" after Xcode 8.2 is stable. | 
					
						
							|  |  |  |         version >= "8.2" | 
					
						
							| 
									
										
										
										
											2016-07-16 01:26:50 +01:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def outdated? | 
					
						
							|  |  |  |         version < latest_version | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-05-22 22:26:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def without_clt? | 
					
						
							|  |  |  |         installed? && version >= "4.3" && !MacOS::CLT.installed? | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-08-09 20:47:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-16 22:51:39 -07:00
										 |  |  |       # Returns a Pathname object corresponding to Xcode.app's Developer | 
					
						
							|  |  |  |       # directory or nil if Xcode.app is not installed | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def prefix | 
					
						
							| 
									
										
										
										
											2014-05-02 16:48:59 -05:00
										 |  |  |         @prefix ||= | 
					
						
							|  |  |  |           begin | 
					
						
							|  |  |  |             dir = MacOS.active_developer_dir | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if dir.empty? || dir == CLT::MAVERICKS_PKG_PATH || !File.directory?(dir) | 
					
						
							|  |  |  |               path = bundle_path | 
					
						
							|  |  |  |               path.join("Contents", "Developer") if path | 
					
						
							|  |  |  |             else | 
					
						
							| 
									
										
										
										
											2016-03-15 23:40:59 -07:00
										 |  |  |               # Use cleanpath to avoid pathological trailing slash | 
					
						
							|  |  |  |               Pathname.new(dir).cleanpath | 
					
						
							| 
									
										
										
										
											2014-05-02 16:48:59 -05:00
										 |  |  |             end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:18:02 -05:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 18:36:46 -05:00
										 |  |  |       def toolchain_path | 
					
						
							| 
									
										
										
										
											2016-03-16 22:09:41 -07:00
										 |  |  |         Pathname.new("#{prefix}/Toolchains/XcodeDefault.xctoolchain") if installed? && version >= "4.3" | 
					
						
							| 
									
										
										
										
											2014-05-01 18:36:46 -05:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       # Ask Spotlight where Xcode is. If the user didn't install the | 
					
						
							|  |  |  |       # helper tools and installed Xcode in a non-conventional place, this | 
					
						
							| 
									
										
										
										
											2015-01-04 05:02:27 +01:00
										 |  |  |       # is our only option. See: https://superuser.com/questions/390757 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def bundle_path | 
					
						
							| 
									
										
										
										
											2014-04-01 20:47:26 -05:00
										 |  |  |         MacOS.app_with_bundle_id(V4_BUNDLE_ID, V3_BUNDLE_ID) | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-07-21 16:41:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def installed? | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |         !prefix.nil? | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:18:02 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-16 21:01:34 +01:00
										 |  |  |       def update_instructions | 
					
						
							|  |  |  |         if MacOS.version >= "10.9" && !OS::Mac.prerelease? | 
					
						
							|  |  |  |           <<-EOS.undent
 | 
					
						
							|  |  |  |             Xcode can be updated from the App Store. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           <<-EOS.undent
 | 
					
						
							|  |  |  |             Xcode can be updated from | 
					
						
							|  |  |  |               https://developer.apple.com/xcode/downloads/ | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def version | 
					
						
							|  |  |  |         # may return a version string | 
					
						
							|  |  |  |         # that is guessed based on the compiler, so do not | 
					
						
							|  |  |  |         # use it in order to check if Xcode is installed. | 
					
						
							|  |  |  |         @version ||= uncached_version | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:18:02 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def uncached_version | 
					
						
							|  |  |  |         # This is a separate function as you can't cache the value out of a block | 
					
						
							|  |  |  |         # if return is used in the middle, which we do many times in here. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return "0" unless OS.mac? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 16:49:16 -04:00
										 |  |  |         return nil if !MacOS::Xcode.installed? && !MacOS::CLT.installed? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-04 20:44:19 +02:00
										 |  |  |         %W[
 | 
					
						
							|  |  |  |           #{prefix}/usr/bin/xcodebuild | 
					
						
							|  |  |  |           #{which("xcodebuild")} | 
					
						
							|  |  |  |         ].uniq.each do |xcodebuild_path| | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |           next unless File.executable? xcodebuild_path | 
					
						
							|  |  |  |           xcodebuild_output = Utils.popen_read(xcodebuild_path, "-version") | 
					
						
							|  |  |  |           next unless $?.success? | 
					
						
							| 
									
										
										
										
											2016-05-04 20:44:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |           xcode_version = xcodebuild_output[/Xcode (\d(\.\d)*)/, 1] | 
					
						
							|  |  |  |           return xcode_version if xcode_version | 
					
						
							| 
									
										
										
										
											2014-04-17 08:23:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |           # Xcode 2.x's xcodebuild has a different version string | 
					
						
							|  |  |  |           case xcodebuild_output[/DevToolsCore-(\d+\.\d)/, 1] | 
					
						
							|  |  |  |           when "515.0" then return "2.0" | 
					
						
							|  |  |  |           when "798.0" then return "2.5" | 
					
						
							| 
									
										
										
										
											2014-04-15 14:18:45 -05:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-15 14:25:27 -05:00
										 |  |  |         # The remaining logic provides a fake Xcode version for CLT-only | 
					
						
							|  |  |  |         # systems. This behavior only exists because Homebrew used to assume | 
					
						
							|  |  |  |         # Xcode.version would always be non-nil. This is deprecated, and will | 
					
						
							|  |  |  |         # be removed in a future version. To remain compatible, guard usage of | 
					
						
							|  |  |  |         # Xcode.version with an Xcode.installed? check. | 
					
						
							| 
									
										
										
										
											2016-05-22 09:40:08 +01:00
										 |  |  |         case (DevelopmentTools.clang_version.to_f * 10).to_i | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |         when 0       then "dunno" | 
					
						
							|  |  |  |         when 1..14   then "3.2.2" | 
					
						
							|  |  |  |         when 15      then "3.2.4" | 
					
						
							|  |  |  |         when 16      then "3.2.5" | 
					
						
							|  |  |  |         when 17..20  then "4.0" | 
					
						
							|  |  |  |         when 21      then "4.1" | 
					
						
							|  |  |  |         when 22..30  then "4.2" | 
					
						
							|  |  |  |         when 31      then "4.3" | 
					
						
							|  |  |  |         when 40      then "4.4" | 
					
						
							|  |  |  |         when 41      then "4.5" | 
					
						
							|  |  |  |         when 42      then "4.6" | 
					
						
							|  |  |  |         when 50      then "5.0" | 
					
						
							|  |  |  |         when 51      then "5.1" | 
					
						
							|  |  |  |         when 60      then "6.0" | 
					
						
							|  |  |  |         when 61      then "6.1" | 
					
						
							|  |  |  |         when 70      then "7.0" | 
					
						
							|  |  |  |         when 73      then "7.3" | 
					
						
							|  |  |  |         when 80      then "8.0" | 
					
						
							|  |  |  |         else "8.0" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-08-06 13:46:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def provides_gcc? | 
					
						
							| 
									
										
										
										
											2016-07-16 21:31:17 +02:00
										 |  |  |         installed? && version < "4.3" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-08-06 13:46:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-04 01:10:11 -06:00
										 |  |  |       def provides_cvs? | 
					
						
							| 
									
										
										
										
											2016-07-16 21:31:17 +02:00
										 |  |  |         installed? && version < "5.0" | 
					
						
							| 
									
										
										
										
											2013-12-04 01:10:11 -06:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def default_prefix? | 
					
						
							| 
									
										
										
										
											2013-10-22 20:48:22 -05:00
										 |  |  |         if version < "4.3" | 
					
						
							| 
									
										
										
										
											2016-01-14 13:57:18 +01:00
										 |  |  |           prefix.to_s.start_with? "/Developer" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2016-03-15 23:50:08 -07:00
										 |  |  |           prefix.to_s == "/Applications/Xcode.app/Contents/Developer" | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:18:02 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:19:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |     module CLT | 
					
						
							|  |  |  |       extend self | 
					
						
							| 
									
										
										
										
											2012-07-10 21:01:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-11 17:49:27 +01:00
										 |  |  |       STANDALONE_PKG_ID = "com.apple.pkg.DeveloperToolsCLILeo".freeze | 
					
						
							|  |  |  |       FROM_XCODE_PKG_ID = "com.apple.pkg.DeveloperToolsCLI".freeze | 
					
						
							|  |  |  |       MAVERICKS_PKG_ID = "com.apple.pkg.CLTools_Executables".freeze | 
					
						
							|  |  |  |       MAVERICKS_NEW_PKG_ID = "com.apple.pkg.CLTools_Base".freeze # obsolete | 
					
						
							|  |  |  |       MAVERICKS_PKG_PATH = "/Library/Developer/CommandLineTools".freeze | 
					
						
							| 
									
										
										
										
											2012-09-14 13:24:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-26 10:35:30 -07:00
										 |  |  |       # Returns true even if outdated tools are installed, e.g. | 
					
						
							|  |  |  |       # tools from Xcode 4.x on 10.9 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def installed? | 
					
						
							| 
									
										
										
										
											2016-09-20 22:24:31 +02:00
										 |  |  |         !detect_version.nil? | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-07-21 16:41:50 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-16 21:01:34 +01:00
										 |  |  |       def update_instructions | 
					
						
							|  |  |  |         if MacOS.version >= "10.9" | 
					
						
							|  |  |  |           <<-EOS.undent
 | 
					
						
							|  |  |  |             Update them from Software Update in the App Store. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         elsif MacOS.version == "10.8" || MacOS.version == "10.7" | 
					
						
							|  |  |  |           <<-EOS.undent
 | 
					
						
							|  |  |  |             The standalone package can be obtained from | 
					
						
							|  |  |  |               https://developer.apple.com/downloads | 
					
						
							|  |  |  |             or it can be installed via Xcode's preferences. | 
					
						
							|  |  |  |           EOS | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-01 16:16:01 -05:00
										 |  |  |       def latest_version | 
					
						
							| 
									
										
										
										
											2016-09-22 17:48:28 +01:00
										 |  |  |         # As of Xcode 8 CLT releases are no longer in sync with Xcode releases | 
					
						
							|  |  |  |         # on the older supported platform for that Xcode release, i.e there's no | 
					
						
							|  |  |  |         # CLT package for 10.11 that contains the Clang version from Xcode 8. | 
					
						
							| 
									
										
										
										
											2014-09-18 10:09:14 -05:00
										 |  |  |         case MacOS.version | 
					
						
							| 
									
										
										
										
											2016-08-19 01:59:33 +01:00
										 |  |  |         when "10.12" then "800.0.38" | 
					
						
							| 
									
										
										
										
											2016-09-22 17:48:28 +01:00
										 |  |  |         when "10.11" then "703.0.31" | 
					
						
							| 
									
										
										
										
											2015-12-09 05:06:00 +00:00
										 |  |  |         when "10.10" then "700.1.81" | 
					
						
							| 
									
										
										
										
											2015-03-10 00:08:43 +00:00
										 |  |  |         when "10.9"  then "600.0.57" | 
					
						
							| 
									
										
										
										
											2014-09-18 10:09:14 -05:00
										 |  |  |         when "10.8"  then "503.0.40" | 
					
						
							| 
									
										
										
										
											2013-11-01 16:16:01 -05:00
										 |  |  |         else | 
					
						
							|  |  |  |           "425.0.28" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2012-08-17 17:18:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def outdated? | 
					
						
							| 
									
										
										
										
											2014-05-12 14:33:41 -05:00
										 |  |  |         if MacOS.version >= :mavericks | 
					
						
							| 
									
										
										
										
											2016-09-22 17:51:04 +01:00
										 |  |  |           version = Utils.popen_read("#{MAVERICKS_PKG_PATH}/usr/bin/clang --version") | 
					
						
							| 
									
										
										
										
											2014-05-12 14:33:41 -05:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2016-09-22 17:51:04 +01:00
										 |  |  |           version = Utils.popen_read("/usr/bin/clang --version") | 
					
						
							| 
									
										
										
										
											2014-05-12 14:33:41 -05:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  |         version = version[/clang-(\d+\.\d+\.\d+(\.\d+)?)/, 1] || "0" | 
					
						
							| 
									
										
										
										
											2013-11-01 16:16:01 -05:00
										 |  |  |         version < latest_version | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-05-22 22:26:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-22 10:14:20 +00:00
										 |  |  |       # Version string (a pretty long one) of the CLT package. | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       # Note, that different ways to install the CLTs lead to different | 
					
						
							|  |  |  |       # version numbers. | 
					
						
							|  |  |  |       def version | 
					
						
							|  |  |  |         @version ||= detect_version | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-07-21 20:09:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |       def detect_version | 
					
						
							| 
									
										
										
										
											2014-04-20 16:14:37 -07:00
										 |  |  |         # CLT isn't a distinct entity pre-4.3, and pkgutil doesn't exist | 
					
						
							|  |  |  |         # at all on Tiger, so just count it as installed if Xcode is installed | 
					
						
							|  |  |  |         return MacOS::Xcode.version if MacOS::Xcode.installed? && MacOS::Xcode.version < "3.0" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-10 13:06:50 -05:00
										 |  |  |         [MAVERICKS_PKG_ID, MAVERICKS_NEW_PKG_ID, STANDALONE_PKG_ID, FROM_XCODE_PKG_ID].find do |id| | 
					
						
							| 
									
										
										
										
											2014-10-17 19:36:25 +01:00
										 |  |  |           if MacOS.version >= :mavericks | 
					
						
							|  |  |  |             next unless File.exist?("#{MAVERICKS_PKG_PATH}/usr/bin/clang") | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2013-10-18 12:56:51 -05:00
										 |  |  |           version = MacOS.pkgutil_info(id)[/version: (.+)$/, 1] | 
					
						
							|  |  |  |           return version if version | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2013-07-21 20:09:55 -05:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2012-07-09 15:18:02 -05:00
										 |  |  |   end | 
					
						
							|  |  |  | end |