Skip to content

Commit 30b80b0

Browse files
authored
When building JMH benchmarks, fail on error, instead of proceeding. (bazelbuild#977)
* When building JMH benchmarks, fail on error, instead of proceeding. Currently, if there are errors in some (but not all) benchmarks, running `bazel run //path/to/benchmark` will compile them, fail on some, and then run the rest. This changes that behavior so that any JMH build failure will fail the build. * Add a test for expected failures in JMH benchmarks. * Document the fix to JMH builds as a breaking change. * Split "test_benchmark_jmh_failure" from "test_benchmark_jmh". * We don't need ValidBenchmark.scala when testing JMH failures.
1 parent dd9ce77 commit 30b80b0

File tree

5 files changed

+63
-13
lines changed

5 files changed

+63
-13
lines changed

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -123,6 +123,12 @@ for an example workspace using another scala version.
123123
| 0.14.x | 3b9ab9be31ac217d3337c709cb6bfeb89c8dcbb1 |
124124
| 0.13.x | 3c987b6ae8a453886759b132f1572c0efca2eca2 |
125125

126+
## Breaking changes
127+
128+
If you're upgrading to a version containing one of these commits, you may encounter a breaking change where there was previously undefined behavior.
129+
130+
- [929b318](https://github.com/bazelbuild/rules_scala/commit/929b3180cc099ba76859f5e88710d2ac087fbfa3) on 2020-01-30: Fixed a bug in the JMH benchmark build that was allowing build failures to creep through. Previously you were able to build a benchmark suite with JMH build errors. Running the benchmark suite would only run the successfully-built benchmarks.
131+
126132
## Usage with [bazel-deps](https://github.com/johnynek/bazel-deps)
127133

128134
Bazel-deps allows you to generate bazel dependencies transitively for maven artifacts. Generally we don't want bazel-deps to fetch

src/scala/io/bazel/rules_scala/jmh_support/BenchmarkGenerator.scala

+18-11
Original file line numberDiff line numberDiff line change
@@ -41,15 +41,24 @@ object BenchmarkGenerator {
4141
classPath: List[Path]
4242
)
4343

44+
private case class GenerationException(messageLines: Seq[String])
45+
extends RuntimeException(messageLines.mkString("\n"))
46+
4447
def main(argv: Array[String]): Unit = {
4548
val args = parseArgs(argv)
46-
generateJmhBenchmark(
47-
args.generatorType,
48-
args.resultSourceJar,
49-
args.resultResourceJar,
50-
args.inputJar,
51-
args.classPath
52-
)
49+
try {
50+
generateJmhBenchmark(
51+
args.generatorType,
52+
args.resultSourceJar,
53+
args.resultResourceJar,
54+
args.inputJar,
55+
args.classPath
56+
)
57+
} catch {
58+
case GenerationException(messageLines) =>
59+
messageLines.foreach(log)
60+
sys.exit(1)
61+
}
5362
}
5463

5564
private def parseArgs(argv: Array[String]): BenchmarkGeneratorArgs = {
@@ -168,10 +177,8 @@ object BenchmarkGenerator {
168177
generator.generate(source, destination)
169178
generator.complete(source, destination)
170179
if (destination.hasErrors) {
171-
log("JMH Benchmark generator failed")
172-
for (e <- destination.getErrors.asScala) {
173-
log(e.toString)
174-
}
180+
throw new GenerationException(
181+
"JHM Benchmark generator failed" +: destination.getErrors.asScala.map(_.toString).toSeq)
175182
}
176183
}
177184
constructJar(sourceJarOut, tmpSourceDir)

test/shell/test_misc.sh

+18-2
Original file line numberDiff line numberDiff line change
@@ -55,12 +55,27 @@ test_repl() {
5555

5656
test_benchmark_jmh() {
5757
RES=$(bazel run -- test/jmh:test_benchmark -i1 -f1 -wi 1)
58-
RESPONSE_CODE=$?
58+
if [ $? -ne 0 ]; then
59+
exit 1
60+
fi
5961
if [[ $RES != *Result*Benchmark* ]]; then
6062
echo "Benchmark did not produce expected output:\n$RES"
6163
exit 1
6264
fi
63-
exit $RESPONSE_CODE
65+
66+
exit 0
67+
}
68+
69+
test_benchmark_jmh_failure() {
70+
set +e
71+
72+
bazel build test_expect_failure/jmh:jmh_reports_failure
73+
if [ $? -eq 0 ]; then
74+
echo "'bazel build test_expect_failure/jmh:jmh_reports_failure' should have failed."
75+
exit 1
76+
fi
77+
78+
exit 0
6479
}
6580

6681
scala_test_test_filters() {
@@ -117,6 +132,7 @@ $runner test_disappearing_class
117132
$runner test_transitive_deps
118133
$runner test_repl
119134
$runner test_benchmark_jmh
135+
$runner test_benchmark_jmh_failure
120136
$runner scala_test_test_filters
121137
$runner test_multi_service_manifest
122138
$runner test_override_javabin

test_expect_failure/jmh/BUILD

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
load(
2+
"//jmh:jmh.bzl",
3+
"scala_benchmark_jmh",
4+
)
5+
6+
scala_benchmark_jmh(
7+
name = "jmh_reports_failure",
8+
srcs = [
9+
"InvalidBenchmark.scala",
10+
],
11+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
package foo
2+
3+
import org.openjdk.jmh.annotations.Benchmark
4+
5+
// Benchmark classes cannot be final.
6+
final class InvalidBenchmark {
7+
@Benchmark
8+
def sumIntegersBenchmark: Int =
9+
(1 to 100).sum
10+
}

0 commit comments

Comments
 (0)