Index: tools/gyp/pylib/gyp/mac_tool.py |
diff --git a/tools/gyp/pylib/gyp/mac_tool.py b/tools/gyp/pylib/gyp/mac_tool.py |
new file mode 100755 |
index 0000000000000000000000000000000000000000..eeeaceb0c7aa2388635855260b3d63edc58cd73a |
--- /dev/null |
+++ b/tools/gyp/pylib/gyp/mac_tool.py |
@@ -0,0 +1,610 @@ |
+#!/usr/bin/env python |
+# Copyright (c) 2012 Google Inc. All rights reserved. |
+# Use of this source code is governed by a BSD-style license that can be |
+# found in the LICENSE file. |
+ |
+"""Utility functions to perform Xcode-style build steps. |
+ |
+These functions are executed via gyp-mac-tool when using the Makefile generator. |
+""" |
+ |
+import fcntl |
+import fnmatch |
+import glob |
+import json |
+import os |
+import plistlib |
+import re |
+import shutil |
+import string |
+import subprocess |
+import sys |
+import tempfile |
+ |
+ |
+def main(args): |
+ executor = MacTool() |
+ exit_code = executor.Dispatch(args) |
+ if exit_code is not None: |
+ sys.exit(exit_code) |
+ |
+ |
+class MacTool(object): |
+ """This class performs all the Mac tooling steps. The methods can either be |
+ executed directly, or dispatched from an argument list.""" |
+ |
+ def Dispatch(self, args): |
+ """Dispatches a string command to a method.""" |
+ if len(args) < 1: |
+ raise Exception("Not enough arguments") |
+ |
+ method = "Exec%s" % self._CommandifyName(args[0]) |
+ return getattr(self, method)(*args[1:]) |
+ |
+ def _CommandifyName(self, name_string): |
+ """Transforms a tool name like copy-info-plist to CopyInfoPlist""" |
+ return name_string.title().replace('-', '') |
+ |
+ def ExecCopyBundleResource(self, source, dest, convert_to_binary): |
+ """Copies a resource file to the bundle/Resources directory, performing any |
+ necessary compilation on each resource.""" |
+ extension = os.path.splitext(source)[1].lower() |
+ if os.path.isdir(source): |
+ # Copy tree. |
+ # TODO(thakis): This copies file attributes like mtime, while the |
+ # single-file branch below doesn't. This should probably be changed to |
+ # be consistent with the single-file branch. |
+ if os.path.exists(dest): |
+ shutil.rmtree(dest) |
+ shutil.copytree(source, dest) |
+ elif extension == '.xib': |
+ return self._CopyXIBFile(source, dest) |
+ elif extension == '.storyboard': |
+ return self._CopyXIBFile(source, dest) |
+ elif extension == '.strings': |
+ self._CopyStringsFile(source, dest, convert_to_binary) |
+ else: |
+ shutil.copy(source, dest) |
+ |
+ def _CopyXIBFile(self, source, dest): |
+ """Compiles a XIB file with ibtool into a binary plist in the bundle.""" |
+ |
+ # ibtool sometimes crashes with relative paths. See crbug.com/314728. |
+ base = os.path.dirname(os.path.realpath(__file__)) |
+ if os.path.relpath(source): |
+ source = os.path.join(base, source) |
+ if os.path.relpath(dest): |
+ dest = os.path.join(base, dest) |
+ |
+ args = ['xcrun', 'ibtool', '--errors', '--warnings', '--notices', |
+ '--output-format', 'human-readable-text', '--compile', dest, source] |
+ ibtool_section_re = re.compile(r'/\*.*\*/') |
+ ibtool_re = re.compile(r'.*note:.*is clipping its content') |
+ ibtoolout = subprocess.Popen(args, stdout=subprocess.PIPE) |
+ current_section_header = None |
+ for line in ibtoolout.stdout: |
+ if ibtool_section_re.match(line): |
+ current_section_header = line |
+ elif not ibtool_re.match(line): |
+ if current_section_header: |
+ sys.stdout.write(current_section_header) |
+ current_section_header = None |
+ sys.stdout.write(line) |
+ return ibtoolout.returncode |
+ |
+ def _ConvertToBinary(self, dest): |
+ subprocess.check_call([ |
+ 'xcrun', 'plutil', '-convert', 'binary1', '-o', dest, dest]) |
+ |
+ def _CopyStringsFile(self, source, dest, convert_to_binary): |
+ """Copies a .strings file using iconv to reconvert the input into UTF-16.""" |
+ input_code = self._DetectInputEncoding(source) or "UTF-8" |
+ |
+ # Xcode's CpyCopyStringsFile / builtin-copyStrings seems to call |
+ # CFPropertyListCreateFromXMLData() behind the scenes; at least it prints |
+ # CFPropertyListCreateFromXMLData(): Old-style plist parser: missing |
+ # semicolon in dictionary. |
+ # on invalid files. Do the same kind of validation. |
+ import CoreFoundation |
+ s = open(source, 'rb').read() |
+ d = CoreFoundation.CFDataCreate(None, s, len(s)) |
+ _, error = CoreFoundation.CFPropertyListCreateFromXMLData(None, d, 0, None) |
+ if error: |
+ return |
+ |
+ fp = open(dest, 'wb') |
+ fp.write(s.decode(input_code).encode('UTF-16')) |
+ fp.close() |
+ |
+ if convert_to_binary == 'True': |
+ self._ConvertToBinary(dest) |
+ |
+ def _DetectInputEncoding(self, file_name): |
+ """Reads the first few bytes from file_name and tries to guess the text |
+ encoding. Returns None as a guess if it can't detect it.""" |
+ fp = open(file_name, 'rb') |
+ try: |
+ header = fp.read(3) |
+ except e: |
+ fp.close() |
+ return None |
+ fp.close() |
+ if header.startswith("\xFE\xFF"): |
+ return "UTF-16" |
+ elif header.startswith("\xFF\xFE"): |
+ return "UTF-16" |
+ elif header.startswith("\xEF\xBB\xBF"): |
+ return "UTF-8" |
+ else: |
+ return None |
+ |
+ def ExecCopyInfoPlist(self, source, dest, convert_to_binary, *keys): |
+ """Copies the |source| Info.plist to the destination directory |dest|.""" |
+ # Read the source Info.plist into memory. |
+ fd = open(source, 'r') |
+ lines = fd.read() |
+ fd.close() |
+ |
+ # Insert synthesized key/value pairs (e.g. BuildMachineOSBuild). |
+ plist = plistlib.readPlistFromString(lines) |
+ if keys: |
+ plist = dict(plist.items() + json.loads(keys[0]).items()) |
+ lines = plistlib.writePlistToString(plist) |
+ |
+ # Go through all the environment variables and replace them as variables in |
+ # the file. |
+ IDENT_RE = re.compile(r'[/\s]') |
+ for key in os.environ: |
+ if key.startswith('_'): |
+ continue |
+ evar = '${%s}' % key |
+ evalue = os.environ[key] |
+ lines = string.replace(lines, evar, evalue) |
+ |
+ # Xcode supports various suffices on environment variables, which are |
+ # all undocumented. :rfc1034identifier is used in the standard project |
+ # template these days, and :identifier was used earlier. They are used to |
+ # convert non-url characters into things that look like valid urls -- |
+ # except that the replacement character for :identifier, '_' isn't valid |
+ # in a URL either -- oops, hence :rfc1034identifier was born. |
+ evar = '${%s:identifier}' % key |
+ evalue = IDENT_RE.sub('_', os.environ[key]) |
+ lines = string.replace(lines, evar, evalue) |
+ |
+ evar = '${%s:rfc1034identifier}' % key |
+ evalue = IDENT_RE.sub('-', os.environ[key]) |
+ lines = string.replace(lines, evar, evalue) |
+ |
+ # Remove any keys with values that haven't been replaced. |
+ lines = lines.split('\n') |
+ for i in range(len(lines)): |
+ if lines[i].strip().startswith("<string>${"): |
+ lines[i] = None |
+ lines[i - 1] = None |
+ lines = '\n'.join(filter(lambda x: x is not None, lines)) |
+ |
+ # Write out the file with variables replaced. |
+ fd = open(dest, 'w') |
+ fd.write(lines) |
+ fd.close() |
+ |
+ # Now write out PkgInfo file now that the Info.plist file has been |
+ # "compiled". |
+ self._WritePkgInfo(dest) |
+ |
+ if convert_to_binary == 'True': |
+ self._ConvertToBinary(dest) |
+ |
+ def _WritePkgInfo(self, info_plist): |
+ """This writes the PkgInfo file from the data stored in Info.plist.""" |
+ plist = plistlib.readPlist(info_plist) |
+ if not plist: |
+ return |
+ |
+ # Only create PkgInfo for executable types. |
+ package_type = plist['CFBundlePackageType'] |
+ if package_type != 'APPL': |
+ return |
+ |
+ # The format of PkgInfo is eight characters, representing the bundle type |
+ # and bundle signature, each four characters. If that is missing, four |
+ # '?' characters are used instead. |
+ signature_code = plist.get('CFBundleSignature', '????') |
+ if len(signature_code) != 4: # Wrong length resets everything, too. |
+ signature_code = '?' * 4 |
+ |
+ dest = os.path.join(os.path.dirname(info_plist), 'PkgInfo') |
+ fp = open(dest, 'w') |
+ fp.write('%s%s' % (package_type, signature_code)) |
+ fp.close() |
+ |
+ def ExecFlock(self, lockfile, *cmd_list): |
+ """Emulates the most basic behavior of Linux's flock(1).""" |
+ # Rely on exception handling to report errors. |
+ fd = os.open(lockfile, os.O_RDONLY|os.O_NOCTTY|os.O_CREAT, 0o666) |
+ fcntl.flock(fd, fcntl.LOCK_EX) |
+ return subprocess.call(cmd_list) |
+ |
+ def ExecFilterLibtool(self, *cmd_list): |
+ """Calls libtool and filters out '/path/to/libtool: file: foo.o has no |
+ symbols'.""" |
+ libtool_re = re.compile(r'^.*libtool: file: .* has no symbols$') |
+ libtool_re5 = re.compile( |
+ r'^.*libtool: warning for library: ' + |
+ r'.* the table of contents is empty ' + |
+ r'\(no object file members in the library define global symbols\)$') |
+ env = os.environ.copy() |
+ # Ref: |
+ # http://www.opensource.apple.com/source/cctools/cctools-809/misc/libtool.c |
+ # The problem with this flag is that it resets the file mtime on the file to |
+ # epoch=0, e.g. 1970-1-1 or 1969-12-31 depending on timezone. |
+ env['ZERO_AR_DATE'] = '1' |
+ libtoolout = subprocess.Popen(cmd_list, stderr=subprocess.PIPE, env=env) |
+ _, err = libtoolout.communicate() |
+ for line in err.splitlines(): |
+ if not libtool_re.match(line) and not libtool_re5.match(line): |
+ print >>sys.stderr, line |
+ # Unconditionally touch the output .a file on the command line if present |
+ # and the command succeeded. A bit hacky. |
+ if not libtoolout.returncode: |
+ for i in range(len(cmd_list) - 1): |
+ if cmd_list[i] == "-o" and cmd_list[i+1].endswith('.a'): |
+ os.utime(cmd_list[i+1], None) |
+ break |
+ return libtoolout.returncode |
+ |
+ def ExecPackageFramework(self, framework, version): |
+ """Takes a path to Something.framework and the Current version of that and |
+ sets up all the symlinks.""" |
+ # Find the name of the binary based on the part before the ".framework". |
+ binary = os.path.basename(framework).split('.')[0] |
+ |
+ CURRENT = 'Current' |
+ RESOURCES = 'Resources' |
+ VERSIONS = 'Versions' |
+ |
+ if not os.path.exists(os.path.join(framework, VERSIONS, version, binary)): |
+ # Binary-less frameworks don't seem to contain symlinks (see e.g. |
+ # chromium's out/Debug/org.chromium.Chromium.manifest/ bundle). |
+ return |
+ |
+ # Move into the framework directory to set the symlinks correctly. |
+ pwd = os.getcwd() |
+ os.chdir(framework) |
+ |
+ # Set up the Current version. |
+ self._Relink(version, os.path.join(VERSIONS, CURRENT)) |
+ |
+ # Set up the root symlinks. |
+ self._Relink(os.path.join(VERSIONS, CURRENT, binary), binary) |
+ self._Relink(os.path.join(VERSIONS, CURRENT, RESOURCES), RESOURCES) |
+ |
+ # Back to where we were before! |
+ os.chdir(pwd) |
+ |
+ def _Relink(self, dest, link): |
+ """Creates a symlink to |dest| named |link|. If |link| already exists, |
+ it is overwritten.""" |
+ if os.path.lexists(link): |
+ os.remove(link) |
+ os.symlink(dest, link) |
+ |
+ def ExecCompileXcassets(self, keys, *inputs): |
+ """Compiles multiple .xcassets files into a single .car file. |
+ |
+ This invokes 'actool' to compile all the inputs .xcassets files. The |
+ |keys| arguments is a json-encoded dictionary of extra arguments to |
+ pass to 'actool' when the asset catalogs contains an application icon |
+ or a launch image. |
+ |
+ Note that 'actool' does not create the Assets.car file if the asset |
+ catalogs does not contains imageset. |
+ """ |
+ command_line = [ |
+ 'xcrun', 'actool', '--output-format', 'human-readable-text', |
+ '--compress-pngs', '--notices', '--warnings', '--errors', |
+ ] |
+ is_iphone_target = 'IPHONEOS_DEPLOYMENT_TARGET' in os.environ |
+ if is_iphone_target: |
+ platform = os.environ['CONFIGURATION'].split('-')[-1] |
+ if platform not in ('iphoneos', 'iphonesimulator'): |
+ platform = 'iphonesimulator' |
+ command_line.extend([ |
+ '--platform', platform, '--target-device', 'iphone', |
+ '--target-device', 'ipad', '--minimum-deployment-target', |
+ os.environ['IPHONEOS_DEPLOYMENT_TARGET'], '--compile', |
+ os.path.abspath(os.environ['CONTENTS_FOLDER_PATH']), |
+ ]) |
+ else: |
+ command_line.extend([ |
+ '--platform', 'macosx', '--target-device', 'mac', |
+ '--minimum-deployment-target', os.environ['MACOSX_DEPLOYMENT_TARGET'], |
+ '--compile', |
+ os.path.abspath(os.environ['UNLOCALIZED_RESOURCES_FOLDER_PATH']), |
+ ]) |
+ if keys: |
+ keys = json.loads(keys) |
+ for key, value in keys.iteritems(): |
+ arg_name = '--' + key |
+ if isinstance(value, bool): |
+ if value: |
+ command_line.append(arg_name) |
+ elif isinstance(value, list): |
+ for v in value: |
+ command_line.append(arg_name) |
+ command_line.append(str(v)) |
+ else: |
+ command_line.append(arg_name) |
+ command_line.append(str(value)) |
+ # Note: actool crashes if inputs path are relative, so use os.path.abspath |
+ # to get absolute path name for inputs. |
+ command_line.extend(map(os.path.abspath, inputs)) |
+ subprocess.check_call(command_line) |
+ |
+ def ExecMergeInfoPlist(self, output, *inputs): |
+ """Merge multiple .plist files into a single .plist file.""" |
+ merged_plist = {} |
+ for path in inputs: |
+ plist = self._LoadPlistMaybeBinary(path) |
+ self._MergePlist(merged_plist, plist) |
+ plistlib.writePlist(merged_plist, output) |
+ |
+ def ExecCodeSignBundle(self, key, resource_rules, entitlements, provisioning): |
+ """Code sign a bundle. |
+ |
+ This function tries to code sign an iOS bundle, following the same |
+ algorithm as Xcode: |
+ 1. copy ResourceRules.plist from the user or the SDK into the bundle, |
+ 2. pick the provisioning profile that best match the bundle identifier, |
+ and copy it into the bundle as embedded.mobileprovision, |
+ 3. copy Entitlements.plist from user or SDK next to the bundle, |
+ 4. code sign the bundle. |
+ """ |
+ resource_rules_path = self._InstallResourceRules(resource_rules) |
+ substitutions, overrides = self._InstallProvisioningProfile( |
+ provisioning, self._GetCFBundleIdentifier()) |
+ entitlements_path = self._InstallEntitlements( |
+ entitlements, substitutions, overrides) |
+ subprocess.check_call([ |
+ 'codesign', '--force', '--sign', key, '--resource-rules', |
+ resource_rules_path, '--entitlements', entitlements_path, |
+ os.path.join( |
+ os.environ['TARGET_BUILD_DIR'], |
+ os.environ['FULL_PRODUCT_NAME'])]) |
+ |
+ def _InstallResourceRules(self, resource_rules): |
+ """Installs ResourceRules.plist from user or SDK into the bundle. |
+ |
+ Args: |
+ resource_rules: string, optional, path to the ResourceRules.plist file |
+ to use, default to "${SDKROOT}/ResourceRules.plist" |
+ |
+ Returns: |
+ Path to the copy of ResourceRules.plist into the bundle. |
+ """ |
+ source_path = resource_rules |
+ target_path = os.path.join( |
+ os.environ['BUILT_PRODUCTS_DIR'], |
+ os.environ['CONTENTS_FOLDER_PATH'], |
+ 'ResourceRules.plist') |
+ if not source_path: |
+ source_path = os.path.join( |
+ os.environ['SDKROOT'], 'ResourceRules.plist') |
+ shutil.copy2(source_path, target_path) |
+ return target_path |
+ |
+ def _InstallProvisioningProfile(self, profile, bundle_identifier): |
+ """Installs embedded.mobileprovision into the bundle. |
+ |
+ Args: |
+ profile: string, optional, short name of the .mobileprovision file |
+ to use, if empty or the file is missing, the best file installed |
+ will be used |
+ bundle_identifier: string, value of CFBundleIdentifier from Info.plist |
+ |
+ Returns: |
+ A tuple containing two dictionary: variables substitutions and values |
+ to overrides when generating the entitlements file. |
+ """ |
+ source_path, provisioning_data, team_id = self._FindProvisioningProfile( |
+ profile, bundle_identifier) |
+ target_path = os.path.join( |
+ os.environ['BUILT_PRODUCTS_DIR'], |
+ os.environ['CONTENTS_FOLDER_PATH'], |
+ 'embedded.mobileprovision') |
+ shutil.copy2(source_path, target_path) |
+ substitutions = self._GetSubstitutions(bundle_identifier, team_id + '.') |
+ return substitutions, provisioning_data['Entitlements'] |
+ |
+ def _FindProvisioningProfile(self, profile, bundle_identifier): |
+ """Finds the .mobileprovision file to use for signing the bundle. |
+ |
+ Checks all the installed provisioning profiles (or if the user specified |
+ the PROVISIONING_PROFILE variable, only consult it) and select the most |
+ specific that correspond to the bundle identifier. |
+ |
+ Args: |
+ profile: string, optional, short name of the .mobileprovision file |
+ to use, if empty or the file is missing, the best file installed |
+ will be used |
+ bundle_identifier: string, value of CFBundleIdentifier from Info.plist |
+ |
+ Returns: |
+ A tuple of the path to the selected provisioning profile, the data of |
+ the embedded plist in the provisioning profile and the team identifier |
+ to use for code signing. |
+ |
+ Raises: |
+ SystemExit: if no .mobileprovision can be used to sign the bundle. |
+ """ |
+ profiles_dir = os.path.join( |
+ os.environ['HOME'], 'Library', 'MobileDevice', 'Provisioning Profiles') |
+ if not os.path.isdir(profiles_dir): |
+ print >>sys.stderr, ( |
+ 'cannot find mobile provisioning for %s' % bundle_identifier) |
+ sys.exit(1) |
+ provisioning_profiles = None |
+ if profile: |
+ profile_path = os.path.join(profiles_dir, profile + '.mobileprovision') |
+ if os.path.exists(profile_path): |
+ provisioning_profiles = [profile_path] |
+ if not provisioning_profiles: |
+ provisioning_profiles = glob.glob( |
+ os.path.join(profiles_dir, '*.mobileprovision')) |
+ valid_provisioning_profiles = {} |
+ for profile_path in provisioning_profiles: |
+ profile_data = self._LoadProvisioningProfile(profile_path) |
+ app_id_pattern = profile_data.get( |
+ 'Entitlements', {}).get('application-identifier', '') |
+ for team_identifier in profile_data.get('TeamIdentifier', []): |
+ app_id = '%s.%s' % (team_identifier, bundle_identifier) |
+ if fnmatch.fnmatch(app_id, app_id_pattern): |
+ valid_provisioning_profiles[app_id_pattern] = ( |
+ profile_path, profile_data, team_identifier) |
+ if not valid_provisioning_profiles: |
+ print >>sys.stderr, ( |
+ 'cannot find mobile provisioning for %s' % bundle_identifier) |
+ sys.exit(1) |
+ # If the user has multiple provisioning profiles installed that can be |
+ # used for ${bundle_identifier}, pick the most specific one (ie. the |
+ # provisioning profile whose pattern is the longest). |
+ selected_key = max(valid_provisioning_profiles, key=lambda v: len(v)) |
+ return valid_provisioning_profiles[selected_key] |
+ |
+ def _LoadProvisioningProfile(self, profile_path): |
+ """Extracts the plist embedded in a provisioning profile. |
+ |
+ Args: |
+ profile_path: string, path to the .mobileprovision file |
+ |
+ Returns: |
+ Content of the plist embedded in the provisioning profile as a dictionary. |
+ """ |
+ with tempfile.NamedTemporaryFile() as temp: |
+ subprocess.check_call([ |
+ 'security', 'cms', '-D', '-i', profile_path, '-o', temp.name]) |
+ return self._LoadPlistMaybeBinary(temp.name) |
+ |
+ def _MergePlist(self, merged_plist, plist): |
+ """Merge |plist| into |merged_plist|.""" |
+ for key, value in plist.iteritems(): |
+ if isinstance(value, dict): |
+ merged_value = merged_plist.get(key, {}) |
+ if isinstance(merged_value, dict): |
+ self._MergePlist(merged_value, value) |
+ merged_plist[key] = merged_value |
+ else: |
+ merged_plist[key] = value |
+ else: |
+ merged_plist[key] = value |
+ |
+ def _LoadPlistMaybeBinary(self, plist_path): |
+ """Loads into a memory a plist possibly encoded in binary format. |
+ |
+ This is a wrapper around plistlib.readPlist that tries to convert the |
+ plist to the XML format if it can't be parsed (assuming that it is in |
+ the binary format). |
+ |
+ Args: |
+ plist_path: string, path to a plist file, in XML or binary format |
+ |
+ Returns: |
+ Content of the plist as a dictionary. |
+ """ |
+ try: |
+ # First, try to read the file using plistlib that only supports XML, |
+ # and if an exception is raised, convert a temporary copy to XML and |
+ # load that copy. |
+ return plistlib.readPlist(plist_path) |
+ except: |
+ pass |
+ with tempfile.NamedTemporaryFile() as temp: |
+ shutil.copy2(plist_path, temp.name) |
+ subprocess.check_call(['plutil', '-convert', 'xml1', temp.name]) |
+ return plistlib.readPlist(temp.name) |
+ |
+ def _GetSubstitutions(self, bundle_identifier, app_identifier_prefix): |
+ """Constructs a dictionary of variable substitutions for Entitlements.plist. |
+ |
+ Args: |
+ bundle_identifier: string, value of CFBundleIdentifier from Info.plist |
+ app_identifier_prefix: string, value for AppIdentifierPrefix |
+ |
+ Returns: |
+ Dictionary of substitutions to apply when generating Entitlements.plist. |
+ """ |
+ return { |
+ 'CFBundleIdentifier': bundle_identifier, |
+ 'AppIdentifierPrefix': app_identifier_prefix, |
+ } |
+ |
+ def _GetCFBundleIdentifier(self): |
+ """Extracts CFBundleIdentifier value from Info.plist in the bundle. |
+ |
+ Returns: |
+ Value of CFBundleIdentifier in the Info.plist located in the bundle. |
+ """ |
+ info_plist_path = os.path.join( |
+ os.environ['TARGET_BUILD_DIR'], |
+ os.environ['INFOPLIST_PATH']) |
+ info_plist_data = self._LoadPlistMaybeBinary(info_plist_path) |
+ return info_plist_data['CFBundleIdentifier'] |
+ |
+ def _InstallEntitlements(self, entitlements, substitutions, overrides): |
+ """Generates and install the ${BundleName}.xcent entitlements file. |
+ |
+ Expands variables "$(variable)" pattern in the source entitlements file, |
+ add extra entitlements defined in the .mobileprovision file and the copy |
+ the generated plist to "${BundlePath}.xcent". |
+ |
+ Args: |
+ entitlements: string, optional, path to the Entitlements.plist template |
+ to use, defaults to "${SDKROOT}/Entitlements.plist" |
+ substitutions: dictionary, variable substitutions |
+ overrides: dictionary, values to add to the entitlements |
+ |
+ Returns: |
+ Path to the generated entitlements file. |
+ """ |
+ source_path = entitlements |
+ target_path = os.path.join( |
+ os.environ['BUILT_PRODUCTS_DIR'], |
+ os.environ['PRODUCT_NAME'] + '.xcent') |
+ if not source_path: |
+ source_path = os.path.join( |
+ os.environ['SDKROOT'], |
+ 'Entitlements.plist') |
+ shutil.copy2(source_path, target_path) |
+ data = self._LoadPlistMaybeBinary(target_path) |
+ data = self._ExpandVariables(data, substitutions) |
+ if overrides: |
+ for key in overrides: |
+ if key not in data: |
+ data[key] = overrides[key] |
+ plistlib.writePlist(data, target_path) |
+ return target_path |
+ |
+ def _ExpandVariables(self, data, substitutions): |
+ """Expands variables "$(variable)" in data. |
+ |
+ Args: |
+ data: object, can be either string, list or dictionary |
+ substitutions: dictionary, variable substitutions to perform |
+ |
+ Returns: |
+ Copy of data where each references to "$(variable)" has been replaced |
+ by the corresponding value found in substitutions, or left intact if |
+ the key was not found. |
+ """ |
+ if isinstance(data, str): |
+ for key, value in substitutions.iteritems(): |
+ data = data.replace('$(%s)' % key, value) |
+ return data |
+ if isinstance(data, list): |
+ return [self._ExpandVariables(v, substitutions) for v in data] |
+ if isinstance(data, dict): |
+ return {k: self._ExpandVariables(data[k], substitutions) for k in data} |
+ return data |
+ |
+if __name__ == '__main__': |
+ sys.exit(main(sys.argv[1:])) |