chef/cookbooks/build-essential/libraries/xcode_command_line_tools.rb
2017-03-20 13:19:10 +00:00

213 lines
6.3 KiB
Ruby

#
# Cookbook Name:: build-essential
# Library:: xcode_command_line_tools
#
# Copyright 2014-2016, Chef Software, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
class Chef
class Resource::XcodeCommandLineTools < Resource::LWRPBase
def self.resource_name
:xcode_command_line_tools
end
actions :install
default_action :install
def initialize(name, run_context = nil)
super
# => Break down SemVer
major, minor, _patch = node['platform_version'].split('.').map { |v| String(v) }
@provider = case [major, minor].join('.')
when '10.7', '10.8'
Provider::XcodeCommandLineToolsFromDmg
when '10.9', '10.10', '10.11'
Provider::XcodeCommandLineToolsFromSoftwareUpdate
else
Chef::Log.warn <<-EOH
OSX #{node['platform_version']} is not an officially supported platform for the
build-essential cookbook. I am going to try and install the command line tools
from Software Update, but there is a high probability that it will fail...
If you have tested and verified OSX #{node['platform_version']} and you are sick
of seeing this warning in your Chef Client runs, please submit a Pull Request to
https://github.com/chef-cookbooks/build-essential and add this version of OSX
to provider list.
EOH
Provider::XcodeCommandLineToolsFromSoftwareUpdate
end
end
end
end
#
# This is a legacy provider for installing OSX from DMGs. It only supports OSX
# versions 10.7 and 10.8 and will (hopefully) be deprecated in the future. It
# downloads a remote .dmg file, mounts it, installs it, and unmounts it
# automatically. In later versions of OSX, the operating system handles this for
# the end user.
#
class Chef
class Provider::XcodeCommandLineToolsFromDmg < Provider::LWRPBase
action(:install) do
if installed?
Chef::Log.debug("#{new_resource} already installed - skipping")
else
converge_by("Install #{new_resource}") do
download
attach
install
detach
end
end
end
private
#
# Determine if the XCode Command Line Tools are installed
#
# @return [true, false]
#
def installed?
cmd = Mixlib::ShellOut.new('pkgutil --pkgs=com.apple.pkg.DeveloperToolsCLI')
cmd.run_command
cmd.error!
true
rescue Mixlib::ShellOut::ShellCommandFailed
false
end
#
# The path where the dmg should be cached on disk.
#
# @return [String]
#
def dmg_cache_path
::File.join(Chef::Config[:file_cache_path], 'osx-command-line-tools.dmg')
end
#
# The path where the dmg should be downloaded from. This is intentionally
# not a configurable object by the end user. If you do not like where we
# are downloading XCode from - too bad.
#
# @return [String]
#
def dmg_remote_source
case node['platform_version'].to_f
when 10.7
'http://devimages.apple.com/downloads/xcode/command_line_tools_for_xcode_os_x_lion_april_2013.dmg'
when 10.8
'http://devimages.apple.com/downloads/xcode/command_line_tools_for_xcode_os_x_mountain_lion_march_2014.dmg'
else
raise "Unknown DMG download URL for OSX #{node['platform_version']}"
end
end
#
# The path where the volume should be mounted.
#
# @return [String]
#
def mount_path
::File.join(Chef::Config[:file_cache_path], 'osx-command-line-tools')
end
#
# Action: download the remote dmg.
#
# @return [void]
#
def download
remote_file = Resource::RemoteFile.new(dmg_cache_path, run_context)
remote_file.source(dmg_remote_source)
remote_file.backup(false)
remote_file.run_action(:create)
end
#
# Action: attach the dmg (basically, double-click on it)
#
# @return [void]
#
def attach
execute %(hdiutil attach "#{dmg_cache_path}" -mountpoint "#{mount_path}")
end
#
# Action: install the package inside the dmg
#
# @return [void]
#
def install
execute %|installer -package "$(find '#{mount_path}' -name *.mpkg)" -target "/"|
end
#
# Action: detach the dmg (basically, drag it to eject on the dock)
#
# @return [void]
#
def detach
execute %(hdiutil detach "#{mount_path}")
end
end
end
class Chef
class Provider::XcodeCommandLineToolsFromSoftwareUpdate < Provider::LWRPBase
action(:install) do
if installed?
Chef::Log.debug("#{new_resource} already installed - skipping")
else
converge_by("Install #{new_resource}") do
# This script was graciously borrowed and modified from Tim Sutton's
# osx-vm-templates at https://github.com/timsutton/osx-vm-templates/blob/b001475df54a9808d3d56d06e71b8fa3001fff42/scripts/xcode-cli-tools.sh
execute 'install XCode Command Line tools' do
command <<-EOH.gsub(/^ {14}/, '')
# create the placeholder file that's checked by CLI updates' .dist code
# in Apple's SUS catalog
touch /tmp/.com.apple.dt.CommandLineTools.installondemand.in-progress
# find the CLI Tools update
PROD=$(softwareupdate -l | grep "\*.*Command Line" | head -n 1 | awk -F"*" '{print $2}' | sed -e 's/^ *//' | tr -d '\n')
# install it
softwareupdate -i "$PROD" -v
EOH
# rubocop:enable Metrics/LineLength
end
end
end
end
private
#
# Determine if the XCode Command Line Tools are installed
#
# @return [true, false]
#
def installed?
cmd = Mixlib::ShellOut.new('pkgutil --pkgs=com.apple.pkg.CLTools_Executables')
cmd.run_command
cmd.error!
true
rescue Mixlib::ShellOut::ShellCommandFailed
false
end
end
end