2 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
6 '''Tool to determine inputs and outputs of a grit file.
14 from grit import grd_reader
17 class WrongNumberOfArguments(Exception):
21 def Outputs(filename, defines, ids_file, target_platform=None):
22 grd = grd_reader.Parse(
23 filename, defines=defines, tags_to_ignore=set(['messages']),
24 first_ids_file=ids_file, target_platform=target_platform)
28 # Add all explicitly-specified output files
29 for output in grd.GetOutputFiles():
30 path = output.GetFilename()
33 if path.endswith('.h'):
34 path, filename = os.path.split(path)
35 if output.attrs['lang']:
36 lang_folders[output.attrs['lang']] = os.path.dirname(path)
38 # Add all generated files, once for each output language.
40 if node.name == 'structure':
42 # TODO(joi) Should remove the "if sconsdep is true" thing as it is a
43 # hack - see grit/node/structure.py
44 if node.HasFileForLanguage() and node.attrs['sconsdep'] == 'true':
45 for lang in lang_folders:
46 path = node.FileForLanguage(lang, lang_folders[lang],
48 return_if_not_generated=False)
52 return [t.replace('\\', '/') for t in target]
55 def GritSourceFiles():
57 grit_root_dir = os.path.relpath(os.path.dirname(__file__), os.getcwd())
58 for root, dirs, filenames in os.walk(grit_root_dir):
59 grit_src = [os.path.join(root, f) for f in filenames
60 if f.endswith('.py') and not f.endswith('_unittest.py')]
61 files.extend(grit_src)
65 def Inputs(filename, defines, ids_file, target_platform=None):
66 grd = grd_reader.Parse(
67 filename, debug=False, defines=defines, tags_to_ignore=set(['message']),
68 first_ids_file=ids_file, target_platform=target_platform)
70 for lang, ctx in grd.GetConfigurations():
71 grd.SetOutputLanguage(lang or grd.GetSourceLanguage())
72 grd.SetOutputContext(ctx)
73 for node in grd.ActiveDescendants():
75 if (node.name == 'structure' or node.name == 'skeleton' or
76 (node.name == 'file' and node.parent and
77 node.parent.name == 'translations')):
78 files.add(grd.ToRealPath(node.GetInputPath()))
79 # If it's a flattened node, grab inlined resources too.
80 if node.name == 'structure' and node.attrs['flattenhtml'] == 'true':
81 node.RunPreSubstitutionGatherer()
82 files.update(node.GetHtmlResourceFilenames())
83 elif node.name == 'grit':
84 first_ids_file = node.GetFirstIdsFile()
86 files.add(first_ids_file)
87 elif node.name == 'include':
88 files.add(grd.ToRealPath(node.GetInputPath()))
89 # If it's a flattened node, grab inlined resources too.
90 if node.attrs['flattenhtml'] == 'true':
91 files.update(node.GetHtmlResourceFilenames())
92 elif node.name == 'part':
93 files.add(util.normpath(os.path.join(os.path.dirname(filename),
94 node.GetInputPath())))
97 return [os.path.relpath(f, cwd) for f in sorted(files)]
101 print 'USAGE: ./grit_info.py --inputs [-D foo] [-f resource_ids] <grd-file>'
102 print (' ./grit_info.py --outputs [-D foo] [-f resource_ids] ' +
103 '<out-prefix> <grd-file>')
107 parser = optparse.OptionParser()
108 parser.add_option("--inputs", action="store_true", dest="inputs")
109 parser.add_option("--outputs", action="store_true", dest="outputs")
110 parser.add_option("-D", action="append", dest="defines", default=[])
111 # grit build also supports '-E KEY=VALUE', support that to share command
113 parser.add_option("-E", action="append", dest="build_env", default=[])
114 parser.add_option("-w", action="append", dest="whitelist_files", default=[])
115 parser.add_option("--output-all-resource-defines", action="store_true",
116 dest="output_all_resource_defines", default=True,
118 parser.add_option("--no-output-all-resource-defines", action="store_false",
119 dest="output_all_resource_defines", default=True,
121 parser.add_option("-f", dest="ids_file",
122 default="GRIT_DIR/../gritsettings/resource_ids")
123 parser.add_option("-t", dest="target_platform", default=None)
125 options, args = parser.parse_args(argv)
128 for define in options.defines:
129 name, val = util.ParseDefine(define)
132 for env_pair in options.build_env:
133 (env_name, env_value) = env_pair.split('=', 1)
134 os.environ[env_name] = env_value
138 raise WrongNumberOfArguments("Expected 0 or 1 arguments for --inputs.")
143 inputs = Inputs(filename, defines, options.ids_file,
144 options.target_platform)
146 # Add in the grit source files. If one of these change, we want to re-run
148 inputs.extend(GritSourceFiles())
149 inputs = [f.replace('\\', '/') for f in inputs]
152 # Include grd file as second input (works around gyp expecting it).
153 inputs.insert(1, args[0])
154 if options.whitelist_files:
155 inputs.extend(options.whitelist_files)
156 return '\n'.join(inputs)
157 elif options.outputs:
159 raise WrongNumberOfArguments(
160 "Expected exactly 2 arguments for --outputs.")
162 prefix, filename = args
163 outputs = [posixpath.join(prefix, f)
164 for f in Outputs(filename, defines,
165 options.ids_file, options.target_platform)]
166 return '\n'.join(outputs)
168 raise WrongNumberOfArguments("Expected --inputs or --outputs.")
172 if sys.version_info < (2, 6):
173 print "GRIT requires Python 2.6 or later."
177 result = DoMain(argv[1:])
178 except WrongNumberOfArguments, e:
186 if __name__ == '__main__':
187 sys.exit(main(sys.argv))