forked from salesforce/rules_spring
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspringboot.bzl
438 lines (379 loc) · 16.1 KB
/
springboot.bzl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
#
# Copyright (c) 2017-2021, salesforce.com, inc.
# All rights reserved.
# Licensed under the BSD 3-Clause license.
# For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause
#
#
# Spring Boot Packager
#
# See the macro documentation below for details.
# Spring Boot Executable JAR Layout specification
# reverse engineered from the Spring Boot maven plugin
# /
# /META-INF/
# /META-INF/MANIFEST.MF <-- very specific manifest for Spring Boot (generated by this rule)
# /BOOT-INF
# /BOOT-INF/classes
# /BOOT-INF/classes/git.properties <-- properties file containing details of the current source tree via Git
# /BOOT-INF/classes/**/*.class <-- compiled application classes, must include @SpringBootApplication class
# /BOOT-INF/classes/META-INF/* <-- application level META-INF config files (e.g. spring.factories)
# /BOOT-INF/lib
# /BOOT-INF/lib/*.jar <-- all upstream transitive dependency jars must be here (except spring-boot-loader)
# /org/springframework/boot/loader
# /org/springframework/boot/loader/**/*.class <-- the Spring Boot Loader classes must be here
# ***************************************************************
# Dependency Aggregator Rule
# do not use directly, see the SpringBoot Macro below
def _depaggregator_rule_impl(ctx):
# magical incantation for getting upstream transitive closure of java deps
merged = java_common.merge([dep[java_common.provider] for dep in ctx.attr.deps])
jars = []
excludes = {}
for exclusion_info in ctx.attr.exclude:
for compile_jar in exclusion_info[JavaInfo].full_compile_jars.to_list():
excludes[compile_jar.path] = True
for dep in merged.transitive_runtime_jars.to_list():
if excludes.get(dep.path, None) == None:
# print("include ", dep.path)
jars.append(dep)
else:
# print("exclude " + dep.path)
pass
# print("AGGREGATED DEPS")
# print(jars)
return [DefaultInfo(files = depset(jars))]
_depaggregator_rule = rule(
implementation = _depaggregator_rule_impl,
attrs = {
"depaggregator_rule": attr.label(),
"deps": attr.label_list(providers = [java_common.provider]),
"exclude": attr.label_list(providers = [java_common.provider], allow_empty = True),
},
)
def _appjar_locator_impl(ctx):
if java_common.provider in ctx.attr.app_dep:
output_jars = ctx.attr.app_dep[java_common.provider].runtime_output_jars
if len(output_jars) != 1:
fail("springboot rule expected 1 app jar but found %s" % len(output_jars))
else:
fail("Unable to locate the app jar")
return [DefaultInfo(files = depset([output_jars[0]]))]
_appjar_locator_rule = rule(
implementation = _appjar_locator_impl,
attrs = {
"app_dep": attr.label(),
},
)
# ***************************************************************
# Check Dupe Classes Rule
def _dupeclasses_rule_impl(ctx):
# setup the output file (contains SUCCESS, NOT_RUN, or the list of errors)
output = ctx.actions.declare_file(ctx.attr.out)
outputs = [output]
if not ctx.attr.fail_on_duplicate_classes:
ctx.actions.write(output, "NOT_RUN", is_executable = False)
return [DefaultInfo(files = depset(outputs))]
inputs = []
input_args = ctx.actions.args()
# inputs (dupe checker python script, spring boot jar file, allowlist)
inputs.append(ctx.attr.script.files.to_list()[0])
input_args.add(ctx.attr.script.files.to_list()[0].path)
inputs.append(ctx.attr.springbootjar.files.to_list()[0])
input_args.add(ctx.attr.springbootjar.files.to_list()[0].path)
if ctx.attr.allowlist != None:
inputs.append(ctx.attr.allowlist.files.to_list()[0])
input_args.add(ctx.attr.allowlist.files.to_list()[0].path)
else:
input_args.add("no_allowlist")
# add the output file to the args, so python script knows where to write result
input_args.add(output.path)
# compute the location of python
python_interpreter = _compute_python_executable(ctx)
# run the dupe checker
ctx.actions.run(
executable = python_interpreter,
outputs = outputs,
inputs = inputs,
arguments = [input_args],
progress_message = "Checking for duplicate classes in the Spring Boot jar...",
mnemonic = "DupeCheck",
)
return [DefaultInfo(files = depset(outputs))]
_dupeclasses_rule = rule(
implementation = _dupeclasses_rule_impl,
attrs = {
"dupeclasses_rule": attr.label(),
"script": attr.label(),
"springbootjar": attr.label(),
"allowlist": attr.label(allow_files=True),
"fail_on_duplicate_classes": attr.bool(),
"out": attr.string(),
},
toolchains = ["@bazel_tools//tools/python:toolchain_type"],
)
def _compute_python_executable(ctx):
python_interpreter = None
# hard requirement on python3 being available
python_runtime = ctx.toolchains["@bazel_tools//tools/python:toolchain_type"].py3_runtime
if python_runtime != None:
if python_runtime.interpreter != None:
# registered python toolchain, or the Bazel python wrapper script (for system python)
python_interpreter = python_runtime.interpreter
elif python_runtime.interpreter_path != None:
# legacy python only?
python_interpreter = python_runtime.interpreter_path
# print(python_interpreter)
return python_interpreter
# ***************************************************************
# Entry point script for "bazel run"
_run_script_template = """
#!/bin/bash
# soon we will use one of the jdk locations already known to Bazel, see Issue #16
if [ -z ${JAVA_HOME} ]; then
java_cmd="$(which java)"
else
java_cmd="${JAVA_HOME}/bin/java"
fi
if [ -z "${java_cmd}" ]; then
echo "ERROR: no java found, either set JAVA_HOME or add the java executable to your PATH"
exit 1
fi
echo "Using Java at ${java_cmd}"
${java_cmd} -version
echo ""
# java args
echo "Using JAVA_OPTS from the environment: ${JAVA_OPTS}"
echo "Using jvm_flags from the BUILD file: %jvm_flags%"
# main args
main_args="$@"
# spring boot jar; these are replaced by the springboot starlark code:
path=%path%
jar=%jar%
# assemble the command
cmd="${java_cmd} %jvm_flags% ${JAVA_OPTS} -jar ${path}/${jar} ${main_args}"
echo "Running ${cmd}"
echo "In directory $(pwd)"
echo ""
echo "You can also run from the root of the repo:"
echo "java -jar bazel-bin/${path}/${jar}"
echo ""
${cmd}
"""
# ***************************************************************
# SpringBoot Rule
# do not use directly, see the SpringBoot Macro below
def _springboot_rule_impl(ctx):
outs = depset(transitive = [
ctx.attr.app_compile_rule.files,
ctx.attr.genmanifest_rule.files,
ctx.attr.gengitinfo_rule.files,
ctx.attr.genjar_rule.files,
])
# setup the script that runs "java -jar <springboot.jar>" when calling
# "bazel run" with the springboot target
jvm_flags = ""
if ctx.attr.jvm_flags != None:
jvm_flags = ctx.attr.jvm_flags
script = _run_script_template \
.replace("%path%", ctx.label.package) \
.replace("%jar%", _get_springboot_jar_file_name(str(ctx.label.name))) \
.replace("%jvm_flags%", jvm_flags)
script_out = ctx.actions.declare_file("%s" % ctx.label.name)
ctx.actions.write(script_out, script, is_executable = True)
# the jar we build needs to be part of runfiles so that it ends up in the
# working directory that "bazel run" uses
runfiles_list = ctx.attr.genjar_rule.files.to_list()
# and add any data files to runfiles
if ctx.attr.data != None:
for data_target in ctx.attr.data:
runfiles_list.append(data_target.files.to_list()[0])
return [DefaultInfo(
files = outs,
executable = script_out,
runfiles = ctx.runfiles(files = runfiles_list),
)]
_springboot_rule = rule(
implementation = _springboot_rule_impl,
executable = True,
attrs = {
"app_compile_rule": attr.label(),
"dep_aggregator_rule": attr.label(),
"genmanifest_rule": attr.label(),
"gengitinfo_rule": attr.label(),
"genjar_rule": attr.label(),
"dupecheck_rule": attr.label(),
"apprun_rule": attr.label(),
"jvm_flags": attr.string(),
"data": attr.label_list(allow_files=True),
},
)
# ***************************************************************
# SpringBoot Macro
# invoke this from your BUILD file, required params are marked *
#
# REQUIRED:
# name: name of your application
# java_library: the java_library rule that contains the compile source for the spring boot app
# boot_app_class: the classname (java package+type) of the @SpringBootApplication class in your app
# deps: the array of upstream dependencies
#
# OPTIONAL:
# visibility: standard rule visibility, defaults to "private" - https://docs.bazel.build/versions/master/visibility.html
# fail_on_duplicated_classes: if enabled, ensures that the final spring boot jar does not contain any duplicate classes (also checks nested jars)
# duplicate_class_allowlist: list of jar files that can have dupe classes without failing the rule
# tags: the array of tags to apply to this rule and subrules
# exclude: list of jar files to exclude from the final jar (i.e. unwanted transitives)
# jvm_flags: flags to pass to the java command when the spring boot application is invoked with 'bazel run
# classpath_index: file that contains the load order of jars (see Spring Boot Classpath Index docs)
#
def springboot(
name,
java_library,
boot_app_class,
deps = None,
visibility = None,
fail_on_duplicate_classes = False,
duplicate_class_allowlist = None,
tags = [],
exclude = [],
jvm_flags = None,
data = [],
classpath_index = None,
use_build_dependency_order = True):
# Create the subrule names
dep_aggregator_rule = native.package_name() + "_deps"
appjar_locator_rule = native.package_name() + "_appjar_locator"
genmanifest_rule = native.package_name() + "_genmanifest"
gengitinfo_rule = native.package_name() + "_gengitinfo"
genjar_rule = native.package_name() + "_genjar"
dupecheck_rule = native.package_name() + "_dupecheck"
apprun_rule = native.package_name() + "_apprun"
# assemble deps; generally all deps will come transtiviely through the java_library
# but a user may choose to add in more deps directly into the springboot jar (rare)
java_deps = [java_library]
if deps != None:
java_deps = [java_library] + deps
# SUBRULE 1: AGGREGATE UPSTREAM DEPS
# Aggregate transitive closure of upstream Java deps
_depaggregator_rule(
name = dep_aggregator_rule,
deps = java_deps,
exclude = exclude,
tags = tags,
)
# SUBRULE 2: GENERATE THE MANIFEST
# NICER: derive the Build JDK and Boot Version values by scanning transitive deps
genmanifest_out = "MANIFEST.MF"
native.genrule(
name = genmanifest_rule,
srcs = [":" + dep_aggregator_rule],
cmd = "$(location @bazel_springboot_rule//tools/springboot:write_manifest.sh) " + boot_app_class + " $@ $(SRCS)",
# message = "SpringBoot rule is writing the MANIFEST.MF...",
tools = ["@bazel_springboot_rule//tools/springboot:write_manifest.sh"],
outs = [genmanifest_out],
tags = tags,
)
# SUBRULE 2B: GENERATE THE GIT PROPERTIES
gengitinfo_out = "git.properties"
native.genrule(
name = gengitinfo_rule,
cmd = "$(location @bazel_springboot_rule//tools/springboot:write_gitinfo_properties.sh) $@",
tools = ["@bazel_springboot_rule//tools/springboot:write_gitinfo_properties.sh"],
outs = [gengitinfo_out],
tags = tags,
stamp = 1,
)
# SUBRULE 2C: CLASSPATH INDEX
if classpath_index == None:
classpath_index = "@bazel_springboot_rule//tools/springboot:empty.txt"
# see https://github.com/salesforce/bazel-springboot-rule/issues/81
_appjar_locator_rule(
name = appjar_locator_rule,
app_dep = java_library,
)
# SUBRULE 3: INVOKE THE BASH SCRIPT THAT DOES THE PACKAGING
# The resolved input_file_paths array is made available as the $(SRCS) token in the cmd string.
# Skylark will convert the logical input_file_paths into real file system paths when surfaced in $(SRCS)
# cmd format (see springboot_pkg.sh)
# param0: directory containing the springboot rule
# param1: location of the jar utility (singlejar)
# param2: boot application main classname (the @SpringBootApplication class)
# param3: jdk path for running java tools e.g. jar; $(JAVABASE)
# param4: compiled application jar name
# param5: use build file deps order [True|False]
# param6: executable jar output filename to write to
# param7: compiled application jar
# param8: manifest file
# param9: git.properties file
# param10: classpath_index file
# param11-N: upstream transitive dependency jar(s)
native.genrule(
name = genjar_rule,
srcs = [
":" + appjar_locator_rule,
":" + genmanifest_rule,
":" + gengitinfo_rule,
classpath_index,
":" + dep_aggregator_rule,
],
cmd = "$(location @bazel_springboot_rule//tools/springboot:springboot_pkg.sh) " +
"$(location @bazel_tools//tools/jdk:singlejar) " + boot_app_class +
" $(JAVABASE) " + name + " " + str(use_build_dependency_order) + " $@ $(SRCS)",
tools = [
"@bazel_springboot_rule//tools/springboot:springboot_pkg.sh",
"@bazel_tools//tools/jdk:singlejar",
],
tags = tags,
outs = [_get_springboot_jar_file_name(name)],
toolchains = ["@bazel_tools//tools/jdk:current_host_java_runtime"], # so that JAVABASE is computed
)
# SUBRULE 4: RUN THE DUPE CHECKER (if enabled)
# Skip the dupecheck_rule instantiation entirely if disabled because
# running this rule requires Python3 installed. If a workspace does not have
# Python3 available, they can just never enable fail_on_duplicate_classes and be ok
dupecheck_rule_label = None
if fail_on_duplicate_classes:
_dupeclasses_rule(
name = dupecheck_rule,
script = "@bazel_springboot_rule//tools/springboot:check_dupe_classes",
springbootjar = genjar_rule,
allowlist = duplicate_class_allowlist,
fail_on_duplicate_classes = fail_on_duplicate_classes,
out = "dupecheck_results.txt",
tags = tags,
)
dupecheck_rule_label = ":" + dupecheck_rule
# SUBRULE 5: PROVIDE A WELL KNOWN RUNNABLE RULE TYPE FOR IDE SUPPORT
# The presence of this rule makes a Spring Boot entry point class runnable
# in IntelliJ (it won't run as part of a packaged Spring Boot jar, ie this
# won't run java -jar springboot.jar, but close enough)
# Making the springboot rule itself executable is not recognized by IntelliJ
# (because IntelliJ doesn't know how to handle the springboot rule type or
# because of a misconfiguration on our end?)
native.java_binary(
name = apprun_rule,
main_class = boot_app_class,
runtime_deps = java_deps,
tags = tags,
)
# MASTER RULE: Create the composite rule that will aggregate the outputs of the subrules
_springboot_rule(
name = name,
app_compile_rule = java_library,
dep_aggregator_rule = ":" + dep_aggregator_rule,
genmanifest_rule = ":" + genmanifest_rule,
gengitinfo_rule = ":" + gengitinfo_rule,
genjar_rule = ":" + genjar_rule,
dupecheck_rule = dupecheck_rule_label,
apprun_rule = ":" + apprun_rule,
jvm_flags = jvm_flags,
data = data,
tags = tags,
visibility = visibility,
)
# end springboot macro
def _get_springboot_jar_file_name(name):
if name.endswith(".jar"):
fail("the name attribute of the springboot rule should not end with '.jar'")
return name + ".jar"