3 # Copyright 2013 The Chromium Authors. All rights reserved.
4 # Use of this source code is governed by a BSD-style license that can be
5 # found in the LICENSE file.
15 from util
import build_utils
16 from util
import md5_check
20 sys
.path
.append(build_utils
.COLORAMA_ROOT
)
24 def ColorJavacOutput(output
):
25 fileline_prefix
= r
'(?P<fileline>(?P<file>[-.\w/\\]+.java):(?P<line>[0-9]+):)'
26 warning_re
= re
.compile(
27 fileline_prefix
+ r
'(?P<full_message> warning: (?P<message>.*))$')
28 error_re
= re
.compile(
29 fileline_prefix
+ r
'(?P<full_message> (?P<message>.*))$')
30 marker_re
= re
.compile(r
'\s*(?P<marker>\^)\s*$')
32 warning_color
= ['full_message', colorama
.Fore
.YELLOW
+ colorama
.Style
.DIM
]
33 error_color
= ['full_message', colorama
.Fore
.MAGENTA
+ colorama
.Style
.BRIGHT
]
34 marker_color
= ['marker', colorama
.Fore
.BLUE
+ colorama
.Style
.BRIGHT
]
36 def Colorize(line
, regex
, color
):
37 match
= regex
.match(line
)
38 start
= match
.start(color
[0])
39 end
= match
.end(color
[0])
41 + color
[1] + line
[start
:end
]
42 + colorama
.Fore
.RESET
+ colorama
.Style
.RESET_ALL
46 if warning_re
.match(line
):
47 line
= Colorize(line
, warning_re
, warning_color
)
48 elif error_re
.match(line
):
49 line
= Colorize(line
, error_re
, error_color
)
50 elif marker_re
.match(line
):
51 line
= Colorize(line
, marker_re
, marker_color
)
54 return '\n'.join(map(ApplyColor
, output
.split('\n')))
58 classpath
, classes_dir
, chromium_code
, java_files
):
61 Builds |java_files| with the provided |classpath| and puts the generated
62 .class files into |classes_dir|. If |chromium_code| is true, extra lint
63 checking will be enabled.
67 for path
in classpath
:
68 if os
.path
.exists(path
+ '.TOC'):
69 jar_inputs
.append(path
+ '.TOC')
71 jar_inputs
.append(path
)
75 # Chromium only allows UTF8 source files. Being explicit avoids
76 # javac pulling a default encoding from the user's environment.
80 '-classpath', ':'.join(classpath
),
83 javac_args
.extend(['-Xlint:unchecked', '-Xlint:deprecation'])
85 # XDignore.symbol.file makes javac compile against rt.jar instead of
86 # ct.sym. This means that using a java internal package/class will not
87 # trigger a compile warning or error.
88 javac_args
.extend(['-XDignore.symbol.file'])
90 javac_cmd
= ['javac'] + javac_args
+ java_files
93 build_utils
.CheckOutput(
95 print_stdout
=chromium_code
,
96 stderr_filter
=ColorJavacOutput
)
98 record_path
= os
.path
.join(classes_dir
, 'javac.md5.stamp')
99 md5_check
.CallAndRecordIfStale(
101 record_path
=record_path
,
102 input_paths
=java_files
+ jar_inputs
,
103 input_strings
=javac_cmd
)
106 _MAX_MANIFEST_LINE_LEN
= 72
109 def CreateManifest(manifest_path
, classpath
, main_class
=None,
110 manifest_entries
=None):
111 """Creates a manifest file with the given parameters.
113 This generates a manifest file that compiles with the spec found at
114 http://docs.oracle.com/javase/7/docs/technotes/guides/jar/jar.html#JAR_Manifest
117 manifest_path: The path to the manifest file that should be created.
118 classpath: The JAR files that should be listed on the manifest file's
120 main_class: If present, the class containing the main() function.
121 manifest_entries: If present, a list of (key, value) pairs to add to
125 output
= ['Manifest-Version: 1.0']
127 output
.append('Main-Class: %s' % main_class
)
129 for k
, v
in manifest_entries
:
130 output
.append('%s: %s' % (k
, v
))
133 for path
in classpath
:
134 sanitized_paths
.append(os
.path
.basename(path
.strip('"')))
135 output
.append('Class-Path: %s' % ' '.join(sanitized_paths
))
136 output
.append('Created-By: ')
139 wrapper
= textwrap
.TextWrapper(break_long_words
=True,
140 drop_whitespace
=False,
141 subsequent_indent
=' ',
142 width
=_MAX_MANIFEST_LINE_LEN
- 2)
143 output
= '\r\n'.join(w
for l
in output
for w
in wrapper
.wrap(l
))
145 with
open(manifest_path
, 'w') as f
:
152 argv
= build_utils
.ExpandFileArgs(argv
)
154 parser
= optparse
.OptionParser()
155 build_utils
.AddDepfileOption(parser
)
159 help='Directories containing generated java files.')
164 help='List of srcjars to include in compilation.')
168 help='Classpath for javac. If this is specified multiple times, they '
169 'will all be appended to construct the classpath.')
172 help='A list of file patterns. If provided, only java files that match'
173 'one of the patterns will be compiled.')
175 '--jar-excluded-classes',
177 help='List of .class file patterns to exclude from the jar.')
182 help='Whether code being compiled should be built with stricter '
183 'warnings for chromium code.')
187 help='Directory for compiled .class files.')
188 parser
.add_option('--jar-path', help='Jar output path.')
191 help='The class containing the main method.')
195 help='Key:value pairs to add to the .jar manifest.')
197 parser
.add_option('--stamp', help='Path to touch on success.')
199 options
, args
= parser
.parse_args(argv
)
201 if options
.main_class
and not options
.jar_path
:
202 parser
.error('--main-class requires --jar-path')
205 for arg
in options
.classpath
:
206 classpath
+= build_utils
.ParseGypList(arg
)
209 for arg
in options
.java_srcjars
:
210 java_srcjars
+= build_utils
.ParseGypList(arg
)
213 if options
.src_gendirs
:
214 src_gendirs
= build_utils
.ParseGypList(options
.src_gendirs
)
215 java_files
+= build_utils
.FindInDirectories(src_gendirs
, '*.java')
217 input_files
= classpath
+ java_srcjars
+ java_files
218 with build_utils
.TempDir() as temp_dir
:
219 classes_dir
= os
.path
.join(temp_dir
, 'classes')
220 os
.makedirs(classes_dir
)
222 java_dir
= os
.path
.join(temp_dir
, 'java')
223 os
.makedirs(java_dir
)
224 for srcjar
in java_srcjars
:
225 build_utils
.ExtractAll(srcjar
, path
=java_dir
, pattern
='*.java')
226 java_files
+= build_utils
.FindInDirectory(java_dir
, '*.java')
228 if options
.javac_includes
:
229 javac_includes
= build_utils
.ParseGypList(options
.javac_includes
)
230 filtered_java_files
= []
232 for include
in javac_includes
:
233 if fnmatch
.fnmatch(f
, include
):
234 filtered_java_files
.append(f
)
236 java_files
= filtered_java_files
241 options
.chromium_code
,
245 if options
.main_class
or options
.manifest_entry
:
246 if options
.manifest_entry
:
247 entries
= map(lambda e
: e
.split(":"), options
.manifest_entry
)
250 manifest_file
= os
.path
.join(temp_dir
, 'manifest')
251 CreateManifest(manifest_file
, classpath
, options
.main_class
, entries
)
254 jar
.JarDirectory(classes_dir
,
255 build_utils
.ParseGypList(options
.jar_excluded_classes
),
257 manifest_file
=manifest_file
)
259 if options
.classes_dir
:
260 # Delete the old classes directory. This ensures that all .class files in
261 # the output are actually from the input .java files. For example, if a
262 # .java file is deleted or an inner class is removed, the classes
263 # directory should not contain the corresponding old .class file after
264 # running this action.
265 build_utils
.DeleteDirectory(options
.classes_dir
)
266 shutil
.copytree(classes_dir
, options
.classes_dir
)
269 build_utils
.WriteDepfile(
271 input_files
+ build_utils
.GetPythonDependencies())
274 build_utils
.Touch(options
.stamp
)
277 if __name__
== '__main__':
278 sys
.exit(main(sys
.argv
[1:]))