blob: 197efff24dc9687b4bf22320fdde5334213b52d6 [file] [log] [blame]
# Copyright 2017 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.
"""Used by a js_binary action to compile javascript files.
This script takes in a list of sources and dependencies and compiles them all
together into a single compiled .js file. The dependencies are ordered in a
post-order, left-to-right traversal order. If multiple instances of the same
source file are read, only the first is kept. The script can also take in
optional --flags argument which will add custom flags to the compiler. Any
extern files can also be passed in using the --extern flag.
"""
import argparse
import os
import sys
import compiler
def ParseDepList(dep):
"""Parses a dependency list, returns |sources, deps, externs|."""
assert os.path.isfile(dep), (dep +
' is not a js_library target')
with open(dep, 'r') as dep_list:
lines = dep_list.read().splitlines()
assert 'deps:' in lines, dep + ' is not formated correctly, missing "deps:"'
deps_start = lines.index('deps:')
assert 'externs:' in lines, dep + ' is not formated correctly, missing "externs:"'
externs_start = lines.index('externs:')
return (lines[1:deps_start],
lines[deps_start+1:externs_start],
lines[externs_start+1:])
# Cache, to avoid reading the same file twice in the dependency tree and
# processing its dependencies again.
depcache = {}
def AppendUnique(items, new_items):
"""Append items in |new_items| to |items|, avoiding duplicates."""
# Note this is O(n*n), and assumes |new_items| is already unique, but this is
# not a bottleneck overall.
items += [i for i in new_items if i not in items]
def CrawlDepsTree(deps):
"""Parses the dependency tree creating a post-order listing of sources."""
global depcache
if len(deps) == 0:
return ([], [])
new_sources = []
new_externs = []
for dep in deps:
if dep in depcache:
cur_sources, cur_externs = depcache[dep]
else:
dep_sources, dep_deps, dep_externs = ParseDepList(dep)
cur_sources, cur_externs = CrawlDepsTree(dep_deps)
# Add child dependencies of this node before the current node, then cache.
AppendUnique(cur_sources, dep_sources)
AppendUnique(cur_externs, dep_externs)
depcache[dep] = (cur_sources, cur_externs)
# Add the current node's sources and dedupe.
AppendUnique(new_sources, cur_sources)
AppendUnique(new_externs, cur_externs)
return new_sources, new_externs
def CrawlRootDepsTree(deps, target_sources, target_externs):
"""Parses the dependency tree and adds target sources."""
sources, externs = CrawlDepsTree(deps)
AppendUnique(sources, target_sources)
AppendUnique(externs, target_externs)
return sources, externs
def main():
parser = argparse.ArgumentParser()
parser.add_argument('-c', '--compiler', required=True,
help='Path to compiler')
parser.add_argument('-s', '--sources', nargs='*', default=[],
help='List of js source files')
parser.add_argument('-o', '--output', required=True,
help='Compile to output')
parser.add_argument('-d', '--deps', nargs='*', default=[],
help='List of js_libarary dependencies')
parser.add_argument('-b', '--bootstrap',
help='A file to include before all others')
parser.add_argument('-cf', '--config', nargs='*', default=[],
help='A list of files to include after bootstrap and '
'before all others')
parser.add_argument('-f', '--flags', nargs='*', default=[],
help='A list of custom flags to pass to the compiler. '
'Do not include leading dashes')
parser.add_argument('-e', '--externs', nargs='*', default=[],
help='A list of extern files to pass to the compiler')
parser.add_argument('-co', '--checks-only', action='store_true',
help='Only performs checks and writes an empty output')
args = parser.parse_args()
sources, externs = CrawlRootDepsTree(args.deps, args.sources, args.externs)
compiler_args = ['--%s' % flag for flag in args.flags]
compiler_args += ['--externs=%s' % e for e in externs]
compiler_args += [
'--js_output_file',
args.output,
'--js',
]
if args.bootstrap:
compiler_args += [args.bootstrap]
compiler_args += args.config
compiler_args += sources
if args.checks_only:
compiler_args += ['--checks-only']
open(args.output, 'w').close()
returncode, errors = compiler.Compiler().run_jar(args.compiler, compiler_args)
if returncode != 0:
print errors
return returncode
if __name__ == '__main__':
sys.exit(main())