3 # Copyright (C) 2010 Google Inc. All rights reserved.
5 # Redistribution and use in source and binary forms, with or without
6 # modification, are permitted provided that the following conditions are
9 # * Redistributions of source code must retain the above copyright
10 # notice, this list of conditions and the following disclaimer.
11 # * Redistributions in binary form must reproduce the above
12 # copyright notice, this list of conditions and the following disclaimer
13 # in the documentation and/or other materials provided with the
15 # * Neither the name of Google Inc. nor the names of its
16 # contributors may be used to endorse or promote products derived from
17 # this software without specific prior written permission.
19 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 # This script concatenates in place JS files in the order specified
33 # using <script> tags in a given 'order.html' file.
35 from __future__ import with_statement
37 from HTMLParser import HTMLParser
38 from cStringIO import StringIO
43 rjsmin_path = os.path.abspath(os.path.join(
44 os.path.dirname(__file__),
49 sys.path.append(rjsmin_path)
53 class OrderedJSFilesExtractor(HTMLParser):
55 def __init__(self, order_html):
56 HTMLParser.__init__(self)
57 self.ordered_js_files = []
60 def handle_starttag(self, tag, attrs):
62 attrs_dict = dict(attrs)
63 if ('type' in attrs_dict and attrs_dict['type'] == 'text/javascript' and 'src' in attrs_dict):
64 self.ordered_js_files.append(attrs_dict['src'])
68 def __init__(self, paths):
71 def expand(self, filename):
72 for path in self.paths:
73 fname = os.path.join(path, filename)
74 if (os.access(fname, os.F_OK)):
82 print('usage: %s order.html input_source_dir_1 input_source_dir_2 ... '
83 'output_file' % argv[0])
86 output_file_name = argv.pop()
87 input_order_file_name = argv[1]
88 with open(input_order_file_name, 'r') as order_html:
89 extractor = OrderedJSFilesExtractor(order_html.read())
91 expander = PathExpander(argv[2:])
94 for input_file_name in extractor.ordered_js_files:
95 full_path = expander.expand(input_file_name)
96 if (full_path is None):
97 raise Exception('File %s referenced in %s not found on any source paths, '
98 'check source tree for consistency' %
99 (input_file_name, input_order_file_name))
100 output.write('/* %s */\n\n' % input_file_name)
101 input_file = open(full_path, 'r')
102 output.write(input_file.read())
106 if os.path.exists(output_file_name):
107 os.remove(output_file_name)
108 output_file = open(output_file_name, 'w')
109 output_file.write(rjsmin.jsmin(output.getvalue()))
113 # Touch output file directory to make sure that Xcode will copy
114 # modified resource files.
115 if sys.platform == 'darwin':
116 output_dir_name = os.path.dirname(output_file_name)
117 os.utime(output_dir_name, None)
119 if __name__ == '__main__':
120 sys.exit(main(sys.argv))