blob: 073d2cb7f72cf30e1e878163e87474cfe35313bc [file] [log] [blame]
#!/usr/bin/env python
# Copyright 2016 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.
"""Compress and upload Mac toolchain files."""
import argparse
import glob
import os
import plistlib
import re
import subprocess
import sys
import tarfile
import tempfile
TOOLCHAIN_URL = "gs://chrome-mac-sdk"
# It's important to at least remove unused Platform folders to cut down on the
# size of the toolchain folder. There are other various unused folders that
# have been removed through trial and error. If future versions of Xcode become
# problematic it's possible this list is incorrect, and can be reduced to just
# the unused platforms. On the flip side, it's likely more directories can be
# excluded.
EXCLUDE_FOLDERS = [
'Contents/Applications',
'Contents/Developer/Documentation',
'Contents/Developer/Platforms/AppleTVOS.platform',
'Contents/Developer/Platforms/AppleTVSimulator.platform',
'Contents/Developer/Platforms/WatchOS.platform',
'Contents/Developer/Platforms/WatchSimulator.platform',
'Contents/Developer/Platforms/iPhoneOS.platform',
'Contents/Developer/Platforms/iPhoneSimulator.platform',
'Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/lib/swift-migrator',
'Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/lib/swift',
'Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.11.sdk/'
'usr/share/man',
'Contents/Developer/Library/Xcode/Templates'
]
def main():
"""Compress |target_dir| and upload to |TOOLCHAIN_URL|"""
parser = argparse.ArgumentParser()
parser.add_argument('target_dir',
help="Xcode installation directory.")
args = parser.parse_args()
# Verify this looks like an Xcode directory.
contents_dir = os.path.join(args.target_dir, 'Contents')
plist_file = os.path.join(contents_dir, 'version.plist')
try:
info = plistlib.readPlist(plist_file)
except:
print "Invalid Xcode dir."
return 0
build_version = info['ProductBuildVersion']
# Look for previous toolchain tgz files with the same |build_version|.
wildcard_filename = '%s/toolchain-%s-*.tgz' % (TOOLCHAIN_URL, build_version)
p = subprocess.Popen(['gsutil.py', 'ls', wildcard_filename],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE)
output = p.communicate()[0]
next_count = 1
if p.returncode == 0:
next_count = len(output.split('\n'))
sys.stdout.write("%s already exists (%s). "
"Do you want to create another? [y/n] "
% (build_version, next_count - 1))
if raw_input().lower() not in set(['yes','y', 'ye']):
print "Skipping duplicate upload."
return 0
os.chdir(args.target_dir)
toolchain_file_name = "toolchain-%s-%s" % (build_version, next_count)
toolchain_name = tempfile.mktemp(suffix='toolchain.tgz')
print "Creating %s (%s)." % (toolchain_file_name, toolchain_name)
os.environ["COPYFILE_DISABLE"] = "1"
args = ['tar', '-cvzf', toolchain_name]
args.extend(map('--exclude={0}'.format, EXCLUDE_FOLDERS))
args.extend(['.'])
subprocess.check_call(args)
print "Uploading %s toolchain." % toolchain_file_name
destination_path = '%s/%s.tgz' % (TOOLCHAIN_URL, toolchain_file_name)
subprocess.check_call(['gsutil.py', 'cp', '-n', '-a', 'public-read',
toolchain_name, destination_path])
print "Done with %s upload." % toolchain_file_name
return 0
if __name__ == '__main__':
sys.exit(main())