blob: e21fc5b1209de302bb0c5c82f46af8e5bf0f3e27 [file] [log] [blame] [edit]
#!/usr/bin/env python3
#
# Copyright (c) 2012 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Prints the lowest locally available SDK version greater than or equal to a
given minimum sdk version to standard output.
Usage:
python find_sdk.py 10.6 # Ignores SDKs < 10.6
"""
import json
import os
import re
import subprocess
import sys
from optparse import OptionParser
sys.path.append(os.path.dirname(os.path.dirname(__file__)))
from pyutil.file_util import symlink
PREBUILTS = os.path.realpath(os.path.join(
os.path.dirname(__file__), os.pardir, os.pardir, 'flutter', 'prebuilts',
))
def parse_version(version_str):
"""'10.6' => [10, 6]"""
return [int(x) for x in re.findall(r'(\d+)', version_str)]
def main():
parser = OptionParser()
parser.add_option("--print_sdk_path",
action="store_true", dest="print_sdk_path", default=False,
help="Additionaly print the path the SDK (appears first).")
parser.add_option("--as-gclient-hook",
action="store_true", dest="as_gclient_hook", default=False,
help="Whether the script is running as a gclient hook.")
parser.add_option("--symlink",
action="store", type="string", dest="symlink",
help="Whether to create a symlink in the buildroot to the SDK.")
(options, args) = parser.parse_args()
min_sdk_version = args[0]
# On CI, Xcode is not yet installed when gclient hooks are being run.
# This is because the version of Xcode that CI installs might depend on the
# contents of the repo, so the repo must be set up first, which includes
# running the gclient hooks. Instead, on CI, this script will be run during
# GN.
running_on_luci = os.environ.get('LUCI_CONTEXT') is not None
if running_on_luci and options.as_gclient_hook:
return 0
symlink_path = options.symlink
if not running_on_luci and symlink_path is None:
symlink_path = PREBUILTS
job = subprocess.Popen(['xcode-select', '-print-path'],
universal_newlines=True,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT)
out, err = job.communicate()
if job.returncode != 0:
sys.stderr.writelines([out, err])
raise Exception(('Error %d running xcode-select, you might have to run '
'|sudo xcode-select --switch /Applications/Xcode.app/Contents/Developer| '
'if you are using Xcode 4.') % job.returncode)
sdk_command = ['xcodebuild',
'-showsdks',
'-json']
sdk_json_output = subprocess.check_output(sdk_command)
sdk_json = json.loads(sdk_json_output)
best_sdk = None
sdk_output = None
# Xcode can return the same version for different symlinked paths.
# Sort by path to keep the list stable between runs.
for properties in sorted(list(sdk_json), key=lambda d: d['sdkPath']):
# Filter out macOS DriverKit, watchOS, AppleTV, and other SDKs.
if properties.get('platform') != 'macosx' or 'driver' in properties.get('canonicalName'):
continue
sdk_version = properties['sdkVersion']
parsed_version = parse_version(sdk_version)
if (parsed_version >= parse_version(min_sdk_version) and
(not best_sdk or parsed_version < parse_version(best_sdk))):
best_sdk = sdk_version
sdk_output = properties['sdkPath']
if not best_sdk:
print(sdk_json_output)
raise Exception('No %s+ SDK found' % min_sdk_version)
if symlink_path:
sdks_path = os.path.join(symlink_path, 'SDKs')
symlink_target = os.path.join(sdks_path, os.path.basename(sdk_output))
symlink(sdk_output, symlink_target)
sdk_output = symlink_target
if not options.as_gclient_hook:
print(sdk_output)
print(best_sdk)
return 0
if __name__ == '__main__':
if sys.platform != 'darwin':
raise Exception("This script only runs on Mac")
sys.exit((main()))