blob: 3ab288f66efbccc96971de7586cd1bdb76b7fd87 [file] [log] [blame]
Francois-Rene Rideau608b4cd2015-07-29 15:40:46 +00001"""Utilities for testing bazel."""
2#
Damien Martin-Guillerezf88f4d82015-09-25 13:56:55 +00003# Copyright 2015 The Bazel Authors. All rights reserved.
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +00004#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000016
Googlerb89ec432024-10-07 10:05:35 -070017load("@rules_shell//shell:sh_test.bzl", "sh_test")
Laszlo Csomor77b98752019-05-17 03:15:22 -070018load(":test_rules_private.bzl", "BASH_RUNFILES_DEP", "INIT_BASH_RUNFILES")
19
20_SH_STUB = "\n".join(["#!/bin/bash"] + INIT_BASH_RUNFILES + [
21 "function add_ws_name() {",
22 ' [[ "$1" =~ external/* ]] && echo "${1#external/}" || echo "$TEST_WORKSPACE/$1"',
23 "}",
24 "",
25])
26
27def _bash_rlocation(f):
28 return '"$(rlocation "$(add_ws_name "%s")")"' % f.short_path
29
30def _make_sh_test(name, **kwargs):
Googlerb89ec432024-10-07 10:05:35 -070031 sh_test(
Laszlo Csomor77b98752019-05-17 03:15:22 -070032 name = name,
33 srcs = [name + "_impl"],
34 data = [name + "_impl"],
35 deps = [BASH_RUNFILES_DEP],
36 **kwargs
37 )
38
Laszlo Csomor6d0b14b2019-07-04 01:20:48 -070039_TEST_ATTRS = {
40 "args": None,
41 "size": None,
42 "timeout": None,
43 "flaky": None,
44 "local": None,
45 "shard_count": None,
46}
47
48def _helper_rule_attrs(test_attrs, own_attrs):
49 r = {}
50 r.update({k: v for k, v in test_attrs.items() if k not in _TEST_ATTRS})
51 r.update(own_attrs)
52 r.update(
53 dict(
54 testonly = 1,
55 visibility = ["//visibility:private"],
56 ),
57 )
58 return r
59
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000060### First, trivial tests that either always pass, always fail,
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +000061### or sometimes pass depending on a trivial computation.
62
Laszlo Csomor77b98752019-05-17 03:15:22 -070063def success_target(ctx, msg, exe = None):
vladmos20a042f2018-06-01 04:51:21 -070064 """Return a success for an analysis test.
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000065
vladmos20a042f2018-06-01 04:51:21 -070066 The test rule must have an executable output.
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +000067
vladmos20a042f2018-06-01 04:51:21 -070068 Args:
69 ctx: the Bazel rule context
70 msg: an informative message to display
Laszlo Csomor77b98752019-05-17 03:15:22 -070071 exe: the output artifact (must have been created with
72 ctx.actions.declare_file or declared in ctx.output), or None meaning
73 ctx.outputs.executable
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +000074
vladmos20a042f2018-06-01 04:51:21 -070075 Returns:
Laszlo Csomor77b98752019-05-17 03:15:22 -070076 DefaultInfo that can be added to a sh_test's srcs AND data. The test will
77 always pass.
vladmos20a042f2018-06-01 04:51:21 -070078 """
Laszlo Csomor77b98752019-05-17 03:15:22 -070079 exe = exe or ctx.outputs.executable
vladmos20a042f2018-06-01 04:51:21 -070080 ctx.actions.write(
81 output = exe,
Laszlo Csomor77b98752019-05-17 03:15:22 -070082 content = "#!/bin/bash\ncat <<'__eof__'\n" + msg + "\n__eof__\necho",
vladmos20a042f2018-06-01 04:51:21 -070083 is_executable = True,
84 )
Laszlo Csomor77b98752019-05-17 03:15:22 -070085 return [DefaultInfo(files = depset([exe]))]
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000086
87def _successful_test_impl(ctx):
Laszlo Csomor77b98752019-05-17 03:15:22 -070088 return success_target(ctx, ctx.attr.msg, exe = ctx.outputs.out)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000089
Laszlo Csomor77b98752019-05-17 03:15:22 -070090_successful_rule = rule(
91 attrs = {
92 "msg": attr.string(mandatory = True),
93 "out": attr.output(),
94 },
Googler9c2989d2015-08-25 17:57:13 +000095 implementation = _successful_test_impl,
96)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +000097
Laszlo Csomor77b98752019-05-17 03:15:22 -070098def successful_test(name, msg, **kwargs):
99 _successful_rule(
Laszlo Csomor6d0b14b2019-07-04 01:20:48 -0700100 **_helper_rule_attrs(
101 kwargs,
102 dict(
103 name = name + "_impl",
104 msg = msg,
105 out = name + "_impl.sh",
106 ),
107 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700108 )
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000109
Laszlo Csomor77b98752019-05-17 03:15:22 -0700110 _make_sh_test(name, **kwargs)
111
112def failure_target(ctx, msg, exe = None):
113 """Return a failure for an analysis test.
laszlocsomor28f8af7c2019-05-17 01:48:56 -0700114
vladmos20a042f2018-06-01 04:51:21 -0700115 Args:
116 ctx: the Bazel rule context
117 msg: an informative message to display
Laszlo Csomor77b98752019-05-17 03:15:22 -0700118 exe: the output artifact (must have been created with
119 ctx.actions.declare_file or declared in ctx.output), or None meaning
120 ctx.outputs.executable
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000121
vladmos20a042f2018-06-01 04:51:21 -0700122 Returns:
Laszlo Csomor77b98752019-05-17 03:15:22 -0700123 DefaultInfo that can be added to a sh_test's srcs AND data. The test will
124 always fail.
vladmos20a042f2018-06-01 04:51:21 -0700125 """
126
127 ### fail(msg) ### <--- This would fail at analysis time.
Laszlo Csomor77b98752019-05-17 03:15:22 -0700128 exe = exe or ctx.outputs.executable
laurentlbf09429c2018-11-14 06:07:43 -0800129 ctx.actions.write(
vladmos20a042f2018-06-01 04:51:21 -0700130 output = exe,
Laszlo Csomor77b98752019-05-17 03:15:22 -0700131 content = "#!/bin/bash\ncat >&2 <<'__eof__'\n" + msg + "\n__eof__\nexit 1",
laurentlbfe2791b2018-11-14 08:54:01 -0800132 is_executable = True,
vladmos20a042f2018-06-01 04:51:21 -0700133 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700134 return [DefaultInfo(files = depset([exe]))]
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000135
136def _failed_test_impl(ctx):
Laszlo Csomor77b98752019-05-17 03:15:22 -0700137 return failure_target(ctx, ctx.attr.msg, exe = ctx.outputs.out)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000138
Laszlo Csomor77b98752019-05-17 03:15:22 -0700139_failed_rule = rule(
140 attrs = {
141 "msg": attr.string(mandatory = True),
142 "out": attr.output(),
143 },
Googler9c2989d2015-08-25 17:57:13 +0000144 implementation = _failed_test_impl,
145)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000146
Laszlo Csomor77b98752019-05-17 03:15:22 -0700147def failed_test(name, msg, **kwargs):
148 _failed_rule(
Laszlo Csomor6d0b14b2019-07-04 01:20:48 -0700149 **_helper_rule_attrs(
150 kwargs,
151 dict(
152 name = name + "_impl",
153 msg = msg,
154 out = name + "_impl.sh",
155 ),
156 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700157 )
158
159 _make_sh_test(name, **kwargs)
160
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000161### Second, general purpose utilities
162
vladmos20a042f2018-06-01 04:51:21 -0700163def assert_(condition, string = "assertion failed", *args):
164 """Trivial assertion mechanism.
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000165
vladmos20a042f2018-06-01 04:51:21 -0700166 Args:
167 condition: a generalized boolean expected to be true
168 string: a format string for the error message should the assertion fail
169 *args: format arguments for the error message should the assertion fail
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000170
vladmos20a042f2018-06-01 04:51:21 -0700171 Returns:
172 None.
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000173
vladmos20a042f2018-06-01 04:51:21 -0700174 Raises:
175 an error if the condition isn't true.
176 """
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000177
vladmos20a042f2018-06-01 04:51:21 -0700178 if not condition:
179 fail(string % args)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000180
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000181def strip_prefix(prefix, string):
vladmos20a042f2018-06-01 04:51:21 -0700182 assert_(
183 string.startswith(prefix),
184 "%s does not start with %s",
185 string,
186 prefix,
187 )
188 return string[len(prefix):len(string)]
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000189
vladmos20a042f2018-06-01 04:51:21 -0700190def expectation_description(expect = None, expect_failure = None):
191 """Turn expectation of result or error into a string."""
192 if expect_failure:
193 return "failure " + str(expect_failure)
194 else:
195 return "result " + repr(expect)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000196
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000197def check_results(result, failure, expect, expect_failure):
vladmos20a042f2018-06-01 04:51:21 -0700198 """See if actual computation results match expectations.
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000199
vladmos20a042f2018-06-01 04:51:21 -0700200 Args:
201 result: the result returned by the test if it ran to completion
202 failure: the failure message caught while testing, if any
203 expect: the expected result for a successful test, if no failure expected
204 expect_failure: the expected failure message for the test, if any
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000205
vladmos20a042f2018-06-01 04:51:21 -0700206 Returns:
207 a pair (tuple) of a boolean (true if success) and a message (string).
208 """
209 wanted = expectation_description(expect, expect_failure)
210 found = expectation_description(result, failure)
211 if wanted == found:
212 return (True, "successfully computed " + wanted)
213 else:
214 return (False, "expect " + wanted + " but found " + found)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000215
vladmos20a042f2018-06-01 04:51:21 -0700216def load_results(
217 name,
218 result = None,
219 failure = None,
220 expect = None,
221 expect_failure = None):
222 """issue load-time results of a test.
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000223
vladmos20a042f2018-06-01 04:51:21 -0700224 Args:
225 name: the name of the Bazel rule at load time.
226 result: the result returned by the test if it ran to completion
227 failure: the failure message caught while testing, if any
228 expect: the expected result for a successful test, if no failure expected
229 expect_failure: the expected failure message for the test, if any
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000230
vladmos20a042f2018-06-01 04:51:21 -0700231 Returns:
232 None, after issuing a rule that will succeed at execution time if
233 expectations were met.
234 """
235 (is_success, msg) = check_results(result, failure, expect, expect_failure)
236 this_test = successful_test if is_success else failed_test
237 return this_test(name = name, msg = msg)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000238
vladmos20a042f2018-06-01 04:51:21 -0700239def analysis_results(
240 ctx,
241 result = None,
242 failure = None,
243 expect = None,
244 expect_failure = None):
245 """issue analysis-time results of a test.
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000246
vladmos20a042f2018-06-01 04:51:21 -0700247 Args:
248 ctx: the Bazel rule context
249 result: the result returned by the test if it ran to completion
250 failure: the failure message caught while testing, if any
251 expect: the expected result for a successful test, if no failure expected
252 expect_failure: the expected failure message for the test, if any
Francois-Rene Rideau73893cc2015-07-20 23:40:48 +0000253
vladmos20a042f2018-06-01 04:51:21 -0700254 Returns:
Laszlo Csomor77b98752019-05-17 03:15:22 -0700255 DefaultInfo that can be added to a sh_test's srcs AND data. The test will
256 always succeed at execution time if expectation were met,
vladmos20a042f2018-06-01 04:51:21 -0700257 or fail at execution time if they didn't.
258 """
259 (is_success, msg) = check_results(result, failure, expect, expect_failure)
260 this_test = success_target if is_success else failure_target
261 return this_test(ctx, msg)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000262
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000263### Simple tests
264
Laszlo Csomor77b98752019-05-17 03:15:22 -0700265def _rule_test_rule_impl(ctx):
vladmos20a042f2018-06-01 04:51:21 -0700266 """check that a rule generates the desired outputs and providers."""
267 rule_ = ctx.attr.rule
268 rule_name = str(rule_.label)
Laszlo Csomor77b98752019-05-17 03:15:22 -0700269 exe = ctx.outputs.out
vladmos20a042f2018-06-01 04:51:21 -0700270 if ctx.attr.generates:
271 # Generate the proper prefix to remove from generated files.
272 prefix_parts = []
John Caterb89f30e2016-11-30 19:01:23 +0000273
vladmos20a042f2018-06-01 04:51:21 -0700274 if rule_.label.workspace_root:
275 # Create a prefix that is correctly relative to the output of this rule.
276 prefix_parts = ["..", strip_prefix("external/", rule_.label.workspace_root)]
John Caterb89f30e2016-11-30 19:01:23 +0000277
vladmos20a042f2018-06-01 04:51:21 -0700278 if rule_.label.package:
279 prefix_parts.append(rule_.label.package)
John Caterb89f30e2016-11-30 19:01:23 +0000280
vladmos20a042f2018-06-01 04:51:21 -0700281 prefix = "/".join(prefix_parts)
John Caterb89f30e2016-11-30 19:01:23 +0000282
vladmos20a042f2018-06-01 04:51:21 -0700283 if prefix:
284 # If the prefix isn't empty, it needs a trailing slash.
285 prefix = prefix + "/"
John Caterb89f30e2016-11-30 19:01:23 +0000286
vladmos20a042f2018-06-01 04:51:21 -0700287 # TODO(bazel-team): Use set() instead of sorted() once
288 # set comparison is implemented.
289 # TODO(bazel-team): Use a better way to determine if two paths refer to
290 # the same file.
291 generates = sorted(ctx.attr.generates)
292 generated = sorted([
293 strip_prefix(prefix, f.short_path)
294 for f in rule_.files.to_list()
295 ])
296 if generates != generated:
297 fail("rule %s generates %s not %s" %
298 (rule_name, repr(generated), repr(generates)))
299 provides = ctx.attr.provides
300 if provides:
301 files = []
302 commands = []
303 for k in provides.keys():
304 if hasattr(rule_, k):
305 v = repr(getattr(rule_, k))
306 else:
307 fail(("rule %s doesn't provide attribute %s. " +
308 "Its list of attributes is: %s") %
309 (rule_name, k, dir(rule_)))
laurentlbf09429c2018-11-14 06:07:43 -0800310 file_ = ctx.actions.declare_file(exe.basename + "." + k)
vladmos20a042f2018-06-01 04:51:21 -0700311 files += [file_]
312 regexp = provides[k]
313 commands += [
Laszlo Csomor77b98752019-05-17 03:15:22 -0700314 "file_=%s" % _bash_rlocation(file_),
315 "if ! grep %s \"$file_\" ; then echo 'bad %s:' ; cat \"$file_\" ; echo ; exit 1 ; fi" %
316 (repr(regexp), k),
vladmos20a042f2018-06-01 04:51:21 -0700317 ]
laurentlbf09429c2018-11-14 06:07:43 -0800318 ctx.actions.write(output = file_, content = v)
Laszlo Csomor77b98752019-05-17 03:15:22 -0700319 script = _SH_STUB + "\n".join(commands)
laurentlbfe2791b2018-11-14 08:54:01 -0800320 ctx.actions.write(output = exe, content = script, is_executable = True)
Laszlo Csomor77b98752019-05-17 03:15:22 -0700321 return [DefaultInfo(files = depset([exe]), runfiles = ctx.runfiles([exe] + files))]
vladmos20a042f2018-06-01 04:51:21 -0700322 else:
Laszlo Csomor77b98752019-05-17 03:15:22 -0700323 return success_target(ctx, "success", exe = exe)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000324
Laszlo Csomor77b98752019-05-17 03:15:22 -0700325_rule_test_rule = rule(
Googler9c2989d2015-08-25 17:57:13 +0000326 attrs = {
327 "rule": attr.label(mandatory = True),
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000328 "generates": attr.string_list(),
Googler9c2989d2015-08-25 17:57:13 +0000329 "provides": attr.string_dict(),
Laszlo Csomor77b98752019-05-17 03:15:22 -0700330 "out": attr.output(),
Googler9c2989d2015-08-25 17:57:13 +0000331 },
Laszlo Csomor77b98752019-05-17 03:15:22 -0700332 implementation = _rule_test_rule_impl,
Googler9c2989d2015-08-25 17:57:13 +0000333)
Francois-Rene Rideaub29ebdc2015-06-19 15:10:47 +0000334
Laszlo Csomor77b98752019-05-17 03:15:22 -0700335def rule_test(name, rule, generates = None, provides = None, **kwargs):
336 _rule_test_rule(
Laszlo Csomor6d0b14b2019-07-04 01:20:48 -0700337 **_helper_rule_attrs(
338 kwargs,
339 dict(
340 name = name + "_impl",
341 rule = rule,
342 generates = generates,
343 provides = provides,
344 out = name + ".sh",
345 ),
346 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700347 )
348
349 _make_sh_test(name, **kwargs)
350
351def _file_test_rule_impl(ctx):
vladmos20a042f2018-06-01 04:51:21 -0700352 """check that a file has a given content."""
Laszlo Csomor77b98752019-05-17 03:15:22 -0700353 exe = ctx.outputs.out
Googlera780be32018-10-16 13:29:06 -0700354 file_ = ctx.file.file
355 content = ctx.attr.content
356 regexp = ctx.attr.regexp
357 matches = ctx.attr.matches
358 if bool(content) == bool(regexp):
359 fail("Must specify one and only one of content or regexp")
360 if content and matches != -1:
361 fail("matches only makes sense with regexp")
362 if content:
laurentlbf09429c2018-11-14 06:07:43 -0800363 dat = ctx.actions.declare_file(exe.basename + ".dat")
364 ctx.actions.write(
Googlera780be32018-10-16 13:29:06 -0700365 output = dat,
366 content = content,
367 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700368 script = "diff -u %s %s" % (_bash_rlocation(dat), _bash_rlocation(file_))
laurentlbf09429c2018-11-14 06:07:43 -0800369 ctx.actions.write(
Googlera780be32018-10-16 13:29:06 -0700370 output = exe,
Laszlo Csomor77b98752019-05-17 03:15:22 -0700371 content = _SH_STUB + script,
laurentlbf09429c2018-11-14 06:07:43 -0800372 is_executable = True,
Googlera780be32018-10-16 13:29:06 -0700373 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700374 return [DefaultInfo(files = depset([exe]), runfiles = ctx.runfiles([exe, dat, file_]))]
Googlera780be32018-10-16 13:29:06 -0700375 if matches != -1:
376 script = "[ %s == $(grep -c %s %s) ]" % (
377 matches,
378 repr(regexp),
Laszlo Csomor77b98752019-05-17 03:15:22 -0700379 _bash_rlocation(file_),
Googlera780be32018-10-16 13:29:06 -0700380 )
381 else:
Laszlo Csomor77b98752019-05-17 03:15:22 -0700382 script = "grep %s %s" % (repr(regexp), _bash_rlocation(file_))
laurentlbf09429c2018-11-14 06:07:43 -0800383 ctx.actions.write(
Googlera780be32018-10-16 13:29:06 -0700384 output = exe,
Laszlo Csomor77b98752019-05-17 03:15:22 -0700385 content = _SH_STUB + script,
laurentlbf09429c2018-11-14 06:07:43 -0800386 is_executable = True,
Laszlo Csomord5501452018-10-16 08:28:21 -0700387 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700388 return [DefaultInfo(files = depset([exe]), runfiles = ctx.runfiles([exe, file_]))]
Laszlo Csomord5501452018-10-16 08:28:21 -0700389
Laszlo Csomor77b98752019-05-17 03:15:22 -0700390_file_test_rule = rule(
Googler9c2989d2015-08-25 17:57:13 +0000391 attrs = {
Googlera780be32018-10-16 13:29:06 -0700392 "file": attr.label(
Googler9c2989d2015-08-25 17:57:13 +0000393 mandatory = True,
laurentlbf09429c2018-11-14 06:07:43 -0800394 allow_single_file = True,
Googler9c2989d2015-08-25 17:57:13 +0000395 ),
396 "content": attr.string(default = ""),
397 "regexp": attr.string(default = ""),
398 "matches": attr.int(default = -1),
Laszlo Csomor77b98752019-05-17 03:15:22 -0700399 "out": attr.output(),
Googler9c2989d2015-08-25 17:57:13 +0000400 },
Laszlo Csomor77b98752019-05-17 03:15:22 -0700401 implementation = _file_test_rule_impl,
Googler9c2989d2015-08-25 17:57:13 +0000402)
Laszlo Csomor77b98752019-05-17 03:15:22 -0700403
404def file_test(name, file, content = None, regexp = None, matches = None, **kwargs):
405 _file_test_rule(
Laszlo Csomor6d0b14b2019-07-04 01:20:48 -0700406 **_helper_rule_attrs(
407 kwargs,
408 dict(
409 name = name + "_impl",
410 file = file,
411 content = content or "",
412 regexp = regexp or "",
413 matches = matches if (matches != None) else -1,
414 out = name + "_impl.sh",
415 ),
416 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700417 )
Laszlo Csomor77b98752019-05-17 03:15:22 -0700418 _make_sh_test(name, **kwargs)