560 lines
19 KiB
Python
560 lines
19 KiB
Python
#!/usr/bin/env python3
|
|
|
|
# Copyright 2020 The Crashpad Authors
|
|
#
|
|
# 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.
|
|
"""Convert GN Xcode projects to platform and configuration independent targets.
|
|
|
|
GN generates Xcode projects that build one configuration only. However, typical
|
|
iOS development involves using the Xcode IDE to toggle the platform and
|
|
configuration. This script replaces the 'gn' configuration with 'Debug',
|
|
'Release' and 'Profile', and changes the ninja invocation to honor these
|
|
configurations.
|
|
"""
|
|
|
|
import argparse
|
|
import collections
|
|
import copy
|
|
import filecmp
|
|
import functools
|
|
import hashlib
|
|
import io
|
|
import json
|
|
import os
|
|
import re
|
|
import shutil
|
|
import string
|
|
import subprocess
|
|
import sys
|
|
import tempfile
|
|
import xml.etree.ElementTree
|
|
|
|
|
|
LLDBINIT_PATH = '$(PROJECT_DIR)/.lldbinit'
|
|
|
|
PYTHON_RE = re.compile('[ /]python[23]?$')
|
|
|
|
XCTEST_PRODUCT_TYPES = frozenset((
|
|
'com.apple.product-type.bundle.unit-test',
|
|
'com.apple.product-type.bundle.ui-testing',
|
|
))
|
|
|
|
SCHEME_PRODUCT_TYPES = frozenset((
|
|
'com.apple.product-type.app-extension',
|
|
'com.apple.product-type.application',
|
|
'com.apple.product-type.framework'
|
|
))
|
|
|
|
|
|
class Template(string.Template):
|
|
|
|
"""A subclass of string.Template that changes delimiter."""
|
|
|
|
delimiter = '@'
|
|
|
|
|
|
@functools.lru_cache
|
|
def LoadSchemeTemplate(root, name):
|
|
"""Return a string.Template object for scheme file loaded relative to root."""
|
|
path = os.path.join(root, 'build', 'ios', name + '.template')
|
|
with open(path) as file:
|
|
return Template(file.read())
|
|
|
|
|
|
def CreateIdentifier(str_id):
|
|
"""Return a 24 characters string that can be used as an identifier."""
|
|
return hashlib.sha1(str_id.encode("utf-8")).hexdigest()[:24].upper()
|
|
|
|
|
|
def GenerateSchemeForTarget(root, project, old_project, name, path, is_test):
|
|
"""Generates the .xcsheme file for target named |name|.
|
|
|
|
The file is generated in the new project schemes directory from a template.
|
|
If there is an existing previous project, then the old scheme file is copied
|
|
and the lldbinit setting is set. If lldbinit setting is already correct, the
|
|
file is not modified, just copied.
|
|
"""
|
|
project_name = os.path.basename(project)
|
|
relative_path = os.path.join('xcshareddata', 'xcschemes', name + '.xcscheme')
|
|
identifier = CreateIdentifier('%s %s' % (name, path))
|
|
|
|
scheme_path = os.path.join(project, relative_path)
|
|
if not os.path.isdir(os.path.dirname(scheme_path)):
|
|
os.makedirs(os.path.dirname(scheme_path))
|
|
|
|
substitutions = {
|
|
'LLDBINIT_PATH': LLDBINIT_PATH,
|
|
'BLUEPRINT_IDENTIFIER': identifier,
|
|
'BUILDABLE_NAME': path,
|
|
'BLUEPRINT_NAME': name,
|
|
'PROJECT_NAME': project_name
|
|
}
|
|
|
|
if is_test:
|
|
template = LoadSchemeTemplate(root, 'xcodescheme-testable')
|
|
substitutions['PATH'] = os.environ['PATH']
|
|
|
|
else:
|
|
template = LoadSchemeTemplate(root, 'xcodescheme')
|
|
|
|
old_scheme_path = os.path.join(old_project, relative_path)
|
|
if os.path.exists(old_scheme_path):
|
|
|
|
tree = xml.etree.ElementTree.parse(old_scheme_path)
|
|
tree_root = tree.getroot()
|
|
|
|
for reference in tree_root.findall('.//BuildableReference'):
|
|
for (attr, value) in (
|
|
('BuildableName', path),
|
|
('BlueprintName', name),
|
|
('BlueprintIdentifier', identifier)):
|
|
if reference.get(attr) != value:
|
|
reference.set(attr, value)
|
|
|
|
for child in tree_root:
|
|
if child.tag not in ('TestAction', 'LaunchAction'):
|
|
continue
|
|
|
|
if child.get('customLLDBInitFile') != LLDBINIT_PATH:
|
|
child.set('customLLDBInitFile', LLDBINIT_PATH)
|
|
|
|
if is_test:
|
|
|
|
template_tree = xml.etree.ElementTree.parse(
|
|
io.StringIO(template.substitute(**substitutions)))
|
|
|
|
template_tree_root = template_tree.getroot()
|
|
for child in tree_root:
|
|
if child.tag != 'BuildAction':
|
|
continue
|
|
|
|
for subchild in list(child):
|
|
child.remove(subchild)
|
|
|
|
for post_action in template_tree_root.findall('.//PostActions'):
|
|
child.append(post_action)
|
|
|
|
tree.write(scheme_path, xml_declaration=True, encoding='UTF-8')
|
|
|
|
else:
|
|
|
|
with open(scheme_path, 'w') as scheme_file:
|
|
scheme_file.write(template.substitute(**substitutions))
|
|
|
|
|
|
class XcodeProject(object):
|
|
|
|
def __init__(self, objects, counter = 0):
|
|
self.objects = objects
|
|
self.counter = 0
|
|
|
|
def AddObject(self, parent_name, obj):
|
|
while True:
|
|
self.counter += 1
|
|
str_id = "%s %s %d" % (parent_name, obj['isa'], self.counter)
|
|
new_id = CreateIdentifier(str_id)
|
|
|
|
# Make sure ID is unique. It's possible there could be an id conflict
|
|
# since this is run after GN runs.
|
|
if new_id not in self.objects:
|
|
self.objects[new_id] = obj
|
|
return new_id
|
|
|
|
def IterObjectsByIsa(self, isa):
|
|
"""Iterates overs objects of the |isa| type."""
|
|
for key, obj in self.objects.items():
|
|
if obj['isa'] == isa:
|
|
yield (key, obj)
|
|
|
|
def IterNativeTargetByProductType(self, product_types):
|
|
"""Iterates over PBXNativeTarget objects of any |product_types| types."""
|
|
for key, obj in self.IterObjectsByIsa('PBXNativeTarget'):
|
|
if obj['productType'] in product_types:
|
|
yield (key, obj)
|
|
|
|
def UpdateBuildScripts(self):
|
|
"""Update build scripts to respect configuration and platforms."""
|
|
for key, obj in self.IterObjectsByIsa('PBXShellScriptBuildPhase'):
|
|
|
|
shell_path = obj['shellPath']
|
|
shell_code = obj['shellScript']
|
|
if shell_path.endswith('/sh'):
|
|
shell_code = shell_code.replace(
|
|
'ninja -C .',
|
|
'ninja -C "../${CONFIGURATION}${EFFECTIVE_PLATFORM_NAME}"')
|
|
elif PYTHON_RE.search(shell_path):
|
|
shell_code = shell_code.replace(
|
|
'''ninja_params = [ '-C', '.' ]''',
|
|
'''ninja_params = [ '-C', '../' + os.environ['CONFIGURATION']'''
|
|
''' + os.environ['EFFECTIVE_PLATFORM_NAME'] ]''')
|
|
|
|
# Replace the build script in the object.
|
|
obj['shellScript'] = shell_code
|
|
|
|
|
|
def UpdateBuildConfigurations(self, configurations):
|
|
"""Add new configurations, using the first one as default."""
|
|
|
|
# Create a list with all the objects of interest. This is needed
|
|
# because objects will be added to/removed from the project upon
|
|
# iterating this list and python dictionaries cannot be mutated
|
|
# during iteration.
|
|
|
|
for key, obj in list(self.IterObjectsByIsa('XCConfigurationList')):
|
|
# Use the first build configuration as template for creating all the
|
|
# new build configurations.
|
|
build_config_template = self.objects[obj['buildConfigurations'][0]]
|
|
build_config_template['buildSettings']['CONFIGURATION_BUILD_DIR'] = \
|
|
'$(PROJECT_DIR)/../$(CONFIGURATION)$(EFFECTIVE_PLATFORM_NAME)'
|
|
|
|
# Remove the existing build configurations from the project before
|
|
# creating the new ones.
|
|
for build_config_id in obj['buildConfigurations']:
|
|
del self.objects[build_config_id]
|
|
obj['buildConfigurations'] = []
|
|
|
|
for configuration in configurations:
|
|
build_config = copy.copy(build_config_template)
|
|
build_config['name'] = configuration
|
|
build_config_id = self.AddObject('products', build_config)
|
|
obj['buildConfigurations'].append(build_config_id)
|
|
|
|
def GetHostMappingForXCTests(self):
|
|
"""Returns a dict from targets to the list of their xctests modules."""
|
|
mapping = collections.defaultdict(list)
|
|
for key, obj in self.IterNativeTargetByProductType(XCTEST_PRODUCT_TYPES):
|
|
build_config_lists_id = obj['buildConfigurationList']
|
|
build_configs = self.objects[build_config_lists_id]['buildConfigurations']
|
|
|
|
# Use the first build configuration to get the name of the host target.
|
|
# This is arbitrary, but since the build configuration are all identical
|
|
# after UpdateBuildConfiguration, except for their 'name', it is fine.
|
|
build_config = self.objects[build_configs[0]]
|
|
if obj['productType'] == 'com.apple.product-type.bundle.unit-test':
|
|
# The test_host value will look like this:
|
|
# `$(BUILD_PRODUCTS_DIR)/host_app_name.app/host_app_name`
|
|
#
|
|
# Extract the `host_app_name.app` part as key for the output.
|
|
test_host_path = build_config['buildSettings']['TEST_HOST']
|
|
test_host_name = os.path.basename(os.path.dirname(test_host_path))
|
|
else:
|
|
test_host_name = build_config['buildSettings']['TEST_TARGET_NAME']
|
|
|
|
test_name = obj['name']
|
|
test_path = self.objects[obj['productReference']]['path']
|
|
|
|
mapping[test_host_name].append((key, test_name, test_path))
|
|
|
|
return dict(mapping)
|
|
|
|
|
|
def check_output(command):
|
|
"""Wrapper around subprocess.check_output that decode output as utf-8."""
|
|
return subprocess.check_output(command).decode('utf-8')
|
|
|
|
|
|
def CopyFileIfChanged(source_path, target_path):
|
|
"""Copy |source_path| to |target_path| if different."""
|
|
target_dir = os.path.dirname(target_path)
|
|
if not os.path.isdir(target_dir):
|
|
os.makedirs(target_dir)
|
|
if not os.path.exists(target_path) or \
|
|
not filecmp.cmp(source_path, target_path):
|
|
shutil.copyfile(source_path, target_path)
|
|
|
|
|
|
def CopyTreeIfChanged(source, target):
|
|
"""Copy |source| to |target| recursively; files are copied iff changed."""
|
|
if os.path.isfile(source):
|
|
return CopyFileIfChanged(source, target)
|
|
if not os.path.isdir(target):
|
|
os.makedirs(target)
|
|
for name in os.listdir(source):
|
|
CopyTreeIfChanged(
|
|
os.path.join(source, name),
|
|
os.path.join(target, name))
|
|
|
|
|
|
def LoadXcodeProjectAsJSON(project_dir):
|
|
"""Return Xcode project at |path| as a JSON string."""
|
|
return check_output([
|
|
'plutil', '-convert', 'json', '-o', '-',
|
|
os.path.join(project_dir, 'project.pbxproj')])
|
|
|
|
|
|
def WriteXcodeProject(output_path, json_string):
|
|
"""Save Xcode project to |output_path| as XML."""
|
|
with tempfile.NamedTemporaryFile() as temp_file:
|
|
temp_file.write(json_string.encode("utf-8"))
|
|
temp_file.flush()
|
|
subprocess.check_call(['plutil', '-convert', 'xml1', temp_file.name])
|
|
CopyFileIfChanged(
|
|
temp_file.name,
|
|
os.path.join(output_path, 'project.pbxproj'))
|
|
|
|
|
|
def UpdateXcodeProject(project_dir, old_project_dir, configurations, root_dir):
|
|
"""Update inplace Xcode project to support multiple configurations.
|
|
|
|
Args:
|
|
project_dir: path to the input Xcode project
|
|
configurations: list of string corresponding to the configurations that
|
|
need to be supported by the tweaked Xcode projects, must contains at
|
|
least one value.
|
|
root_dir: path to the root directory used to find markdown files
|
|
"""
|
|
json_data = json.loads(LoadXcodeProjectAsJSON(project_dir))
|
|
project = XcodeProject(json_data['objects'])
|
|
|
|
project.UpdateBuildScripts()
|
|
project.UpdateBuildConfigurations(configurations)
|
|
|
|
mapping = project.GetHostMappingForXCTests()
|
|
|
|
# Generate schemes for application, extensions and framework targets
|
|
for key, obj in project.IterNativeTargetByProductType(SCHEME_PRODUCT_TYPES):
|
|
product = project.objects[obj['productReference']]
|
|
product_path = product['path']
|
|
|
|
# Do not generate scheme for the XCTests and XXCUITests target app.
|
|
# Instead, a scheme will be generated for each test modules.
|
|
tests = mapping.get(product_path, []) + mapping.get(obj['name'], [])
|
|
if not tests:
|
|
GenerateSchemeForTarget(
|
|
root_dir, project_dir, old_project_dir,
|
|
obj['name'], product_path, False)
|
|
|
|
else:
|
|
for (_, test_name, test_path) in tests:
|
|
GenerateSchemeForTarget(
|
|
root_dir, project_dir, old_project_dir,
|
|
test_name, test_path, True)
|
|
|
|
root_object = project.objects[json_data['rootObject']]
|
|
main_group = project.objects[root_object['mainGroup']]
|
|
|
|
sources = None
|
|
for child_key in main_group['children']:
|
|
child = project.objects[child_key]
|
|
if child.get('name') == 'Source' or child.get('name') == 'Sources':
|
|
sources = child
|
|
break
|
|
|
|
if sources is None:
|
|
sources = main_group
|
|
|
|
AddMarkdownToProject(project, root_dir, sources, sources is main_group)
|
|
SortFileReferencesByName(project, sources, root_object.get('productRefGroup'))
|
|
|
|
objects = collections.OrderedDict(sorted(project.objects.items()))
|
|
# WriteXcodeProject(project_dir, json.dumps(json_data))
|
|
|
|
|
|
def CreateGroup(project, parent_group, group_name, use_relative_paths):
|
|
group_object = {
|
|
'children': [],
|
|
'isa': 'PBXGroup',
|
|
'sourceTree': '<group>',
|
|
}
|
|
if use_relative_paths:
|
|
group_object['path'] = group_name
|
|
else:
|
|
group_object['name'] = group_name
|
|
parent_group_name = parent_group.get('name', '')
|
|
group_object_key = project.AddObject(parent_group_name, group_object)
|
|
parent_group['children'].append(group_object_key)
|
|
return group_object
|
|
|
|
|
|
class ObjectKey(object):
|
|
|
|
"""Wrapper around PBXFileReference and PBXGroup for sorting.
|
|
|
|
A PBXGroup represents a "directory" containing a list of files in an
|
|
Xcode project; it can contain references to a list of directories or
|
|
files.
|
|
|
|
A PBXFileReference represents a "file".
|
|
|
|
The type is stored in the object "isa" property as a string. Since we
|
|
want to sort all directories before all files, the < and > operators
|
|
are defined so that if "isa" is different, they are sorted in the
|
|
reverse of alphabetic ordering, otherwise the name (or path) property
|
|
is checked and compared in alphabetic order.
|
|
"""
|
|
|
|
def __init__(self, obj, last):
|
|
self.isa = obj['isa']
|
|
if 'name' in obj:
|
|
self.name = obj['name']
|
|
else:
|
|
self.name = obj['path']
|
|
self.last = last
|
|
|
|
def __lt__(self, other):
|
|
if self.last != other.last:
|
|
return other.last
|
|
if self.isa != other.isa:
|
|
return self.isa > other.isa
|
|
return self.name < other.name
|
|
|
|
def __gt__(self, other):
|
|
if self.last != other.last:
|
|
return self.last
|
|
if self.isa != other.isa:
|
|
return self.isa < other.isa
|
|
return self.name > other.name
|
|
|
|
def __eq__(self, other):
|
|
return self.isa == other.isa and self.name == other.name
|
|
|
|
|
|
def SortFileReferencesByName(project, group_object, products_group_ref):
|
|
SortFileReferencesByNameWithSortKey(
|
|
project, group_object,
|
|
lambda ref: ObjectKey(project.objects[ref], ref == products_group_ref))
|
|
|
|
|
|
def SortFileReferencesByNameWithSortKey(project, group_object, sort_key):
|
|
group_object['children'].sort(key=sort_key)
|
|
for key in group_object['children']:
|
|
child = project.objects[key]
|
|
if child['isa'] == 'PBXGroup':
|
|
SortFileReferencesByNameWithSortKey(project, child, sort_key)
|
|
|
|
|
|
def AddMarkdownToProject(project, root_dir, group_object, use_relative_paths):
|
|
list_files_cmd = ['git', '-C', root_dir, 'ls-files', '*.md']
|
|
paths = check_output(list_files_cmd).splitlines()
|
|
ios_internal_dir = os.path.join(root_dir, 'ios_internal')
|
|
if os.path.exists(ios_internal_dir):
|
|
list_files_cmd = ['git', '-C', ios_internal_dir, 'ls-files', '*.md']
|
|
ios_paths = check_output(list_files_cmd).splitlines()
|
|
paths.extend([os.path.join("ios_internal", path) for path in ios_paths])
|
|
for path in paths:
|
|
new_markdown_entry = {
|
|
"fileEncoding": "4",
|
|
"isa": "PBXFileReference",
|
|
"lastKnownFileType": "net.daringfireball.markdown",
|
|
"sourceTree": "<group>"
|
|
}
|
|
if use_relative_paths:
|
|
new_markdown_entry['path'] = os.path.basename(path)
|
|
else:
|
|
new_markdown_entry['name'] = os.path.basename(path)
|
|
new_markdown_entry['path'] = path
|
|
folder = GetFolderForPath(
|
|
project, group_object, os.path.dirname(path),
|
|
use_relative_paths)
|
|
folder_name = folder.get('name', None)
|
|
if folder_name is None:
|
|
folder_name = folder.get('path', 'sources')
|
|
new_markdown_entry_id = project.AddObject(folder_name, new_markdown_entry)
|
|
folder['children'].append(new_markdown_entry_id)
|
|
|
|
|
|
def GetFolderForPath(project, group_object, path, use_relative_paths):
|
|
objects = project.objects
|
|
if not path:
|
|
return group_object
|
|
for folder in path.split('/'):
|
|
children = group_object['children']
|
|
new_root = None
|
|
for child_key in children:
|
|
child = objects[child_key]
|
|
if child['isa'] == 'PBXGroup':
|
|
child_name = child.get('name', None)
|
|
if child_name is None:
|
|
child_name = child.get('path')
|
|
if child_name == folder:
|
|
new_root = child
|
|
break
|
|
if not new_root:
|
|
# If the folder isn't found we could just cram it into the leaf existing
|
|
# folder, but that leads to folders with tons of README.md inside.
|
|
new_root = CreateGroup(project, group_object, folder, use_relative_paths)
|
|
group_object = new_root
|
|
return group_object
|
|
|
|
|
|
def ConvertGnXcodeProject(root_dir, proj_name, input_dir, output_dir, configs):
|
|
'''Tweak the Xcode project generated by gn to support multiple configurations.
|
|
|
|
The Xcode projects generated by "gn gen --ide" only supports a single
|
|
platform and configuration (as the platform and configuration are set
|
|
per output directory). This method takes as input such projects and
|
|
add support for multiple configurations and platforms (to allow devs
|
|
to select them in Xcode).
|
|
|
|
Args:
|
|
root_dir: directory that is the root of the project
|
|
proj_name: name of the Xcode project "file" (usually `all.xcodeproj`)
|
|
input_dir: directory containing the XCode projects created by "gn gen --ide"
|
|
output_dir: directory where the tweaked Xcode projects will be saved
|
|
configs: list of string corresponding to the configurations that need to be
|
|
supported by the tweaked Xcode projects, must contains at least one
|
|
value.
|
|
'''
|
|
|
|
UpdateXcodeProject(
|
|
os.path.join(input_dir, proj_name),
|
|
os.path.join(output_dir, proj_name),
|
|
configs, root_dir)
|
|
|
|
CopyTreeIfChanged(os.path.join(input_dir, proj_name),
|
|
os.path.join(output_dir, proj_name))
|
|
|
|
|
|
def Main(args):
|
|
parser = argparse.ArgumentParser(
|
|
description='Convert GN Xcode projects for iOS.')
|
|
parser.add_argument(
|
|
'input',
|
|
help='directory containing [product|all] Xcode projects.')
|
|
parser.add_argument(
|
|
'output',
|
|
help='directory where to generate the iOS configuration.')
|
|
parser.add_argument(
|
|
'--add-config', dest='configurations', default=[], action='append',
|
|
help='configuration to add to the Xcode project')
|
|
parser.add_argument(
|
|
'--root', type=os.path.abspath, required=True,
|
|
help='root directory of the project')
|
|
parser.add_argument(
|
|
'--project-name', default='all.xcodeproj', dest='proj_name',
|
|
help='name of the Xcode project (default: %(default)s)')
|
|
args = parser.parse_args(args)
|
|
|
|
if not os.path.isdir(args.input):
|
|
sys.stderr.write('Input directory does not exists.\n')
|
|
return 1
|
|
|
|
if args.proj_name not in os.listdir(args.input):
|
|
sys.stderr.write(
|
|
'Input directory does not contain the Xcode project.\n')
|
|
return 1
|
|
|
|
if not args.configurations:
|
|
sys.stderr.write('At least one configuration required, see --add-config.\n')
|
|
return 1
|
|
|
|
ConvertGnXcodeProject(
|
|
args.root,
|
|
args.proj_name,
|
|
args.input,
|
|
args.output,
|
|
args.configurations)
|
|
|
|
if __name__ == '__main__':
|
|
sys.exit(Main(sys.argv[1:]))
|