diff --git a/docs/BUILD b/docs/BUILD
index 6fa677e1..f4f6b55d 100644
--- a/docs/BUILD
+++ b/docs/BUILD
@@ -105,3 +105,10 @@ stardoc(
input = "//rules:write_file.bzl",
deps = ["//rules:write_file"],
)
+
+stardoc(
+ name = "diff_test_docs",
+ out = "diff_test_doc_gen.md",
+ input = "//rules:diff_test.bzl",
+ deps = ["//rules:diff_test"],
+)
diff --git a/rules/BUILD b/rules/BUILD
index 0d7d150b..b008166b 100644
--- a/rules/BUILD
+++ b/rules/BUILD
@@ -22,6 +22,11 @@ bzl_library(
deps = ["//rules/private:write_file_private"],
)
+bzl_library(
+ name = "diff_test",
+ srcs = ["diff_test.bzl"],
+)
+
# Exported for build_test.bzl to make sure of, it is an implementation detail
# of the rule and should not be directly used by anything else.
exports_files(["empty_test.sh"])
diff --git a/rules/diff_test.bzl b/rules/diff_test.bzl
new file mode 100644
index 00000000..94acaba8
--- /dev/null
+++ b/rules/diff_test.bzl
@@ -0,0 +1,137 @@
+# Copyright 2019 The Bazel Authors. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+"""A test rule that compares two binary files.
+
+The rule uses a Bash command (diff) on Linux/macOS/non-Windows, and a cmd.exe
+command (fc.exe) on Windows (no Bash is required).
+"""
+
+def _runfiles_path(f):
+ if f.root.path:
+ return f.path[len(f.root.path) + 1:] # generated file
+ else:
+ return f.path # source file
+
+def _diff_test_impl(ctx):
+ if ctx.attr.is_windows:
+ test_bin = ctx.actions.declare_file(ctx.label.name + "-test.bat")
+ ctx.actions.write(
+ output = test_bin,
+ content = r"""@echo off
+setlocal enableextensions
+set MF=%RUNFILES_MANIFEST_FILE:/=\%
+set PATH=%SYSTEMROOT%\system32
+for /F "tokens=2* usebackq" %%i in (`findstr.exe /l /c:"%TEST_WORKSPACE%/{file1} " "%MF%"`) do (
+ set RF1=%%i
+ set RF1=%RF1:/=\%
+)
+if "%RF1%" equ "" (
+ echo>&2 ERROR: {file1} not found
+ exit /b 1
+)
+for /F "tokens=2* usebackq" %%i in (`findstr.exe /l /c:"%TEST_WORKSPACE%/{file2} " "%MF%"`) do (
+ set RF2=%%i
+ set RF2=%RF2:/=\%
+)
+if "%RF2%" equ "" (
+ echo>&2 ERROR: {file2} not found
+ exit /b 1
+)
+fc.exe 2>NUL 1>NUL /B "%RF1%" "%RF2%"
+if %ERRORLEVEL% equ 2 (
+ echo>&2 FAIL: "{file1}" and/or "{file2}" not found
+ exit /b 1
+) else (
+ if %ERRORLEVEL% equ 1 (
+ echo>&2 FAIL: files "{file1}" and "{file2}" differ
+ exit /b 1
+ )
+)
+""".format(
+ file1 = _runfiles_path(ctx.file.file1),
+ file2 = _runfiles_path(ctx.file.file2),
+ ),
+ is_executable = True,
+ )
+ else:
+ test_bin = ctx.actions.declare_file(ctx.label.name + "-test.sh")
+ ctx.actions.write(
+ output = test_bin,
+ content = r"""#!/bin/bash
+set -euo pipefail
+if [[ -d "${{RUNFILES_DIR:-/dev/null}}" ]]; then
+ RF1="$RUNFILES_DIR/$TEST_WORKSPACE/{file1}"
+ RF2="$RUNFILES_DIR/$TEST_WORKSPACE/{file2}"
+elif [[ -f "${{RUNFILES_MANIFEST_FILE:-/dev/null}}" ]]; then
+ RF1="$(grep -F -m1 '{file1} ' "$RUNFILES_MANIFEST_FILE" | sed 's/^[^ ]* //')"
+ RF2="$(grep -F -m1 '{file2} ' "$RUNFILES_MANIFEST_FILE" | sed 's/^[^ ]* //'))"
+else
+ echo >&2 "ERROR: could not find \"{file1}\" and \"{file2}\""
+fi
+if ! diff "$RF1" "$RF2"; then
+ echo >&2 "FAIL: files \"{file1}\" and \"{file2}\" differ"
+ exit 1
+fi
+""".format(
+ file1 = _runfiles_path(ctx.file.file1),
+ file2 = _runfiles_path(ctx.file.file2),
+ ),
+ is_executable = True,
+ )
+ return DefaultInfo(
+ executable = test_bin,
+ files = depset(direct = [test_bin]),
+ runfiles = ctx.runfiles(files = [test_bin, ctx.file.file1, ctx.file.file2]),
+ )
+
+_diff_test = rule(
+ attrs = {
+ "file1": attr.label(
+ allow_files = True,
+ mandatory = True,
+ single_file = True,
+ ),
+ "file2": attr.label(
+ allow_files = True,
+ mandatory = True,
+ single_file = True,
+ ),
+ "is_windows": attr.bool(mandatory = True),
+ },
+ test = True,
+ implementation = _diff_test_impl,
+)
+
+def diff_test(name, file1, file2, **kwargs):
+ """A test that compares two files.
+
+ The test succeeds if the files' contents match.
+
+ Args:
+ name: The name of the test rule.
+ file1: Label of the file to compare to file2
.
+ file2: Label of the file to compare to file1
.
+ **kwargs: The common attributes for tests.
+ """
+ _diff_test(
+ name = name,
+ file1 = file1,
+ file2 = file2,
+ is_windows = select({
+ "@bazel_tools//src/conditions:host_windows": True,
+ "//conditions:default": False,
+ }),
+ **kwargs
+ )
diff --git a/tests/diff_test/BUILD b/tests/diff_test/BUILD
new file mode 100644
index 00000000..3a33bf28
--- /dev/null
+++ b/tests/diff_test/BUILD
@@ -0,0 +1,42 @@
+# This package aids testing the 'diff_test' rule.
+
+load("//rules:diff_test.bzl", "diff_test")
+
+package(default_testonly = 1)
+
+sh_test(
+ name = "diff_test_tests",
+ srcs = ["diff_test_tests.sh"],
+ data = [
+ "//rules:diff_test",
+ "//tests:unittest.bash",
+ ],
+ deps = ["@bazel_tools//tools/bash/runfiles"],
+)
+
+diff_test(
+ name = "same_src_src",
+ file1 = "a.txt",
+ file2 = "aa.txt",
+)
+
+diff_test(
+ name = "same_src_gen",
+ file1 = "a.txt",
+ file2 = "a-gen.txt",
+)
+
+diff_test(
+ name = "same_gen_gen",
+ file1 = "a-gen.txt",
+ file2 = "aa-gen.txt",
+)
+
+genrule(
+ name = "gen",
+ outs = [
+ "a-gen.txt",
+ "aa-gen.txt",
+ ],
+ cmd = "echo -n 'potato' > $(location a-gen.txt) && echo -n 'potato' > $(location aa-gen.txt)",
+)
diff --git a/tests/diff_test/a.txt b/tests/diff_test/a.txt
new file mode 100644
index 00000000..86ba0095
--- /dev/null
+++ b/tests/diff_test/a.txt
@@ -0,0 +1 @@
+potato
\ No newline at end of file
diff --git a/tests/diff_test/aa.txt b/tests/diff_test/aa.txt
new file mode 100644
index 00000000..86ba0095
--- /dev/null
+++ b/tests/diff_test/aa.txt
@@ -0,0 +1 @@
+potato
\ No newline at end of file
diff --git a/tests/diff_test/b.txt b/tests/diff_test/b.txt
new file mode 100644
index 00000000..19102815
--- /dev/null
+++ b/tests/diff_test/b.txt
@@ -0,0 +1 @@
+foo
\ No newline at end of file
diff --git a/tests/diff_test/diff_test_tests.sh b/tests/diff_test/diff_test_tests.sh
new file mode 100755
index 00000000..48eef58b
--- /dev/null
+++ b/tests/diff_test/diff_test_tests.sh
@@ -0,0 +1,139 @@
+# Copyright 2019 The Bazel Authors. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+# --- begin runfiles.bash initialization ---
+# Copy-pasted from Bazel's Bash runfiles library (tools/bash/runfiles/runfiles.bash).
+set -euo pipefail
+if [[ ! -d "${RUNFILES_DIR:-/dev/null}" && ! -f "${RUNFILES_MANIFEST_FILE:-/dev/null}" ]]; then
+ if [[ -f "$0.runfiles_manifest" ]]; then
+ export RUNFILES_MANIFEST_FILE="$0.runfiles_manifest"
+ elif [[ -f "$0.runfiles/MANIFEST" ]]; then
+ export RUNFILES_MANIFEST_FILE="$0.runfiles/MANIFEST"
+ elif [[ -f "$0.runfiles/bazel_tools/tools/bash/runfiles/runfiles.bash" ]]; then
+ export RUNFILES_DIR="$0.runfiles"
+ fi
+fi
+if [[ -f "${RUNFILES_DIR:-/dev/null}/bazel_tools/tools/bash/runfiles/runfiles.bash" ]]; then
+ source "${RUNFILES_DIR}/bazel_tools/tools/bash/runfiles/runfiles.bash"
+elif [[ -f "${RUNFILES_MANIFEST_FILE:-/dev/null}" ]]; then
+ source "$(grep -m1 "^bazel_tools/tools/bash/runfiles/runfiles.bash " \
+ "$RUNFILES_MANIFEST_FILE" | cut -d ' ' -f 2-)"
+else
+ echo >&2 "ERROR: cannot find @bazel_tools//tools/bash/runfiles:runfiles.bash"
+ exit 1
+fi
+# --- end runfiles.bash initialization ---
+
+source "$(rlocation bazel_skylib/tests/unittest.bash)" \
+ || { echo "Could not source bazel_skylib/tests/unittest.bash" >&2; exit 1; }
+
+function test_diff_test() {
+ local -r ws="${TEST_TMPDIR}/${FUNCNAME[0]}"
+
+ mkdir -p "$ws/rules"
+ ln -sf "$(rlocation bazel_skylib/rules/diff_test.bzl)" "$ws/rules/diff_test.bzl"
+ echo "exports_files(['diff_test.bzl'])" > "$ws/rules/BUILD"
+ touch "$ws/WORKSPACE"
+ cat >"$ws/BUILD" <<'eof'
+load("//rules:diff_test.bzl", "diff_test")
+
+diff_test(
+ name = "same",
+ file1 = "a.txt",
+ file2 = "a.txt",
+)
+
+diff_test(
+ name = "different",
+ file1 = "a.txt",
+ file2 = "b.txt",
+)
+eof
+ echo foo > "$ws/a.txt"
+ echo bar > "$ws/b.txt"
+
+ (cd "$ws" && \
+ bazel test //:same --test_output=errors 1>"$TEST_log" 2>&1 \
+ || fail "expected success")
+
+ (cd "$ws" && \
+ bazel test //:different --test_output=errors 1>"$TEST_log" 2>&1 \
+ && fail "expected failure" || true)
+ expect_log 'FAIL: files "a.txt" and "b.txt" differ'
+}
+
+function test_from_ext_repo() {
+ local -r ws="${TEST_TMPDIR}/${FUNCNAME[0]}"
+
+ mkdir -p "$ws/rules" "$ws/ext1/foo" "$ws/ext2/foo"
+ ln -sf "$(rlocation bazel_skylib/rules/diff_test.bzl)" "$ws/rules/diff_test.bzl"
+ echo "exports_files(['diff_test.bzl'])" > "$ws/rules/BUILD"
+ cat >"$ws/WORKSPACE" <<'eof'
+local_repository(
+ name = "ext1",
+ path = "ext1",
+)
+
+local_repository(
+ name = "ext2",
+ path = "ext2",
+)
+eof
+
+ # ext1 has source files
+ touch "$ws/ext1/WORKSPACE"
+ echo 'exports_files(["foo.txt"])' >"$ws/ext1/foo/BUILD"
+ echo 'foo' > "$ws/ext1/foo/foo.txt"
+
+ # ext2 has generated files
+ touch "$ws/ext2/WORKSPACE"
+ cat >"$ws/ext2/foo/BUILD" <<'eof'
+genrule(
+ name = "gen",
+ outs = [
+ "foo.txt",
+ "bar.txt",
+ ],
+ cmd = "echo 'foo' > $(location foo.txt) && echo 'bar' > $(location bar.txt)",
+ visibility = ["//visibility:public"],
+)
+eof
+
+ cat >"$ws/BUILD" <<'eof'
+load("//rules:diff_test.bzl", "diff_test")
+
+diff_test(
+ name = "same",
+ file1 = "@ext1//foo:foo.txt",
+ file2 = "@ext2//foo:foo.txt",
+)
+
+diff_test(
+ name = "different",
+ file1 = "@ext1//foo:foo.txt",
+ file2 = "@ext2//foo:bar.txt",
+)
+eof
+
+ (cd "$ws" && \
+ bazel test //:same --test_output=errors 1>"$TEST_log" 2>&1 \
+ || fail "expected success")
+
+ (cd "$ws" && \
+ bazel test //:different --test_output=errors 1>"$TEST_log" 2>&1 \
+ && fail "expected failure" || true)
+ expect_log 'FAIL: files "external/ext1/foo/foo.txt" and "external/ext2/foo/bar.txt" differ'
+}
+
+run_suite "diff_test_tests test suite"