Index: gn/gypi_to_gn.py |
diff --git a/gn/gypi_to_gn.py b/gn/gypi_to_gn.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..08007088a89698cacb669267e0639817d01d1bd7 |
--- /dev/null |
+++ b/gn/gypi_to_gn.py |
@@ -0,0 +1,191 @@ |
+# Copyright 2014 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. |
+ |
+"""Converts a given gypi file to a python scope and writes the result to stdout. |
+ |
+USING THIS SCRIPT IN CHROMIUM |
+ |
+Forking Python to run this script in the middle of GN is slow, especially on |
+Windows, and it makes both the GYP and GN files harder to follow. You can't |
+use "git grep" to find files in the GN build any more, and tracking everything |
+in GYP down requires a level of indirection. Any calls will have to be removed |
+and cleaned up once the GYP-to-GN transition is complete. |
+ |
+As a result, we only use this script when the list of files is large and |
+frequently-changing. In these cases, having one canonical list outweights the |
+downsides. |
+ |
+As of this writing, the GN build is basically complete. It's likely that all |
+large and frequently changing targets where this is appropriate use this |
+mechanism already. And since we hope to turn down the GYP build soon, the time |
+horizon is also relatively short. As a result, it is likely that no additional |
+uses of this script should every be added to the build. During this later part |
+of the transition period, we should be focusing more and more on the absolute |
+readability of the GN build. |
+ |
+ |
+HOW TO USE |
+ |
+It is assumed that the file contains a toplevel dictionary, and this script |
+will return that dictionary as a GN "scope" (see example below). This script |
+does not know anything about GYP and it will not expand variables or execute |
+conditions. |
+ |
+It will strip conditions blocks. |
+ |
+A variables block at the top level will be flattened so that the variables |
+appear in the root dictionary. This way they can be returned to the GN code. |
+ |
+Say your_file.gypi looked like this: |
+ { |
+ 'sources': [ 'a.cc', 'b.cc' ], |
+ 'defines': [ 'ENABLE_DOOM_MELON' ], |
+ } |
+ |
+You would call it like this: |
+ gypi_values = exec_script("//build/gypi_to_gn.py", |
+ [ rebase_path("your_file.gypi") ], |
+ "scope", |
+ [ "your_file.gypi" ]) |
+ |
+Notes: |
+ - The rebase_path call converts the gypi file from being relative to the |
+ current build file to being system absolute for calling the script, which |
+ will have a different current directory than this file. |
+ |
+ - The "scope" parameter tells GN to interpret the result as a series of GN |
+ variable assignments. |
+ |
+ - The last file argument to exec_script tells GN that the given file is a |
+ dependency of the build so Ninja can automatically re-run GN if the file |
+ changes. |
+ |
+Read the values into a target like this: |
+ component("mycomponent") { |
+ sources = gypi_values.sources |
+ defines = gypi_values.defines |
+ } |
+ |
+Sometimes your .gypi file will include paths relative to a different |
+directory than the current .gn file. In this case, you can rebase them to |
+be relative to the current directory. |
+ sources = rebase_path(gypi_values.sources, ".", |
+ "//path/gypi/input/values/are/relative/to") |
+ |
+This script will tolerate a 'variables' in the toplevel dictionary or not. If |
+the toplevel dictionary just contains one item called 'variables', it will be |
+collapsed away and the result will be the contents of that dictinoary. Some |
+.gypi files are written with or without this, depending on how they expect to |
+be embedded into a .gyp file. |
+ |
+This script also has the ability to replace certain substrings in the input. |
+Generally this is used to emulate GYP variable expansion. If you passed the |
+argument "--replace=<(foo)=bar" then all instances of "<(foo)" in strings in |
+the input will be replaced with "bar": |
+ |
+ gypi_values = exec_script("//build/gypi_to_gn.py", |
+ [ rebase_path("your_file.gypi"), |
+ "--replace=<(foo)=bar"], |
+ "scope", |
+ [ "your_file.gypi" ]) |
+ |
+""" |
+ |
+import gn_helpers |
+from optparse import OptionParser |
+import sys |
+ |
+def LoadPythonDictionary(path): |
+ file_string = open(path).read() |
+ try: |
+ file_data = eval(file_string, {'__builtins__': None}, None) |
+ except SyntaxError, e: |
+ e.filename = path |
+ raise |
+ except Exception, e: |
+ raise Exception("Unexpected error while reading %s: %s" % (path, str(e))) |
+ |
+ assert isinstance(file_data, dict), "%s does not eval to a dictionary" % path |
+ |
+ # Flatten any variables to the top level. |
+ if 'variables' in file_data: |
+ file_data.update(file_data['variables']) |
+ del file_data['variables'] |
+ |
+ # Strip all elements that this script can't process. |
+ elements_to_strip = [ |
+ 'conditions', |
+ 'target_conditions', |
+ 'targets', |
+ 'includes', |
+ 'actions', |
+ ] |
+ for element in elements_to_strip: |
+ if element in file_data: |
+ del file_data[element] |
+ |
+ return file_data |
+ |
+ |
+def ReplaceSubstrings(values, search_for, replace_with): |
+ """Recursively replaces substrings in a value. |
+ |
+ Replaces all substrings of the "search_for" with "repace_with" for all |
+ strings occurring in "values". This is done by recursively iterating into |
+ lists as well as the keys and values of dictionaries.""" |
+ if isinstance(values, str): |
+ return values.replace(search_for, replace_with) |
+ |
+ if isinstance(values, list): |
+ return [ReplaceSubstrings(v, search_for, replace_with) for v in values] |
+ |
+ if isinstance(values, dict): |
+ # For dictionaries, do the search for both the key and values. |
+ result = {} |
+ for key, value in values.items(): |
+ new_key = ReplaceSubstrings(key, search_for, replace_with) |
+ new_value = ReplaceSubstrings(value, search_for, replace_with) |
+ result[new_key] = new_value |
+ return result |
+ |
+ # Assume everything else is unchanged. |
+ return values |
+ |
+def main(): |
+ parser = OptionParser() |
+ parser.add_option("-r", "--replace", action="append", |
+ help="Replaces substrings. If passed a=b, replaces all substrs a with b.") |
+ (options, args) = parser.parse_args() |
+ |
+ if len(args) != 1: |
+ raise Exception("Need one argument which is the .gypi file to read.") |
+ |
+ data = LoadPythonDictionary(args[0]) |
+ if options.replace: |
+ # Do replacements for all specified patterns. |
+ for replace in options.replace: |
+ split = replace.split('=') |
+ # Allow "foo=" to replace with nothing. |
+ if len(split) == 1: |
+ split.append('') |
+ assert len(split) == 2, "Replacement must be of the form 'key=value'." |
+ data = ReplaceSubstrings(data, split[0], split[1]) |
+ |
+ # Sometimes .gypi files use the GYP syntax with percents at the end of the |
+ # variable name (to indicate not to overwrite a previously-defined value): |
+ # 'foo%': 'bar', |
+ # Convert these to regular variables. |
+ for key in data: |
+ if len(key) > 1 and key[len(key) - 1] == '%': |
+ data[key[:-1]] = data[key] |
+ del data[key] |
+ |
+ print gn_helpers.ToGNString(data) |
+ |
+if __name__ == '__main__': |
+ try: |
+ main() |
+ except Exception, e: |
+ print str(e) |
+ sys.exit(1) |