Skip to content

Commit

Permalink
docs(tar): point to the tests as useful examples (bazel-contrib#892)
Browse files Browse the repository at this point in the history
* docs(tar): point to the tests as useful examples

Improve the content to make it easier to reference as examples of usage.

* fix broken link
  • Loading branch information
alexeagle authored Aug 5, 2024
1 parent cdbfe41 commit 109f32e
Show file tree
Hide file tree
Showing 3 changed files with 79 additions and 56 deletions.
22 changes: 15 additions & 7 deletions docs/tar.md

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

22 changes: 15 additions & 7 deletions lib/tar.bzl
Original file line number Diff line number Diff line change
@@ -1,18 +1,24 @@
"""General-purpose rule to create tar archives.
Unlike [pkg_tar from rules_pkg](https://github.com/bazelbuild/rules_pkg/blob/main/docs/latest.md#pkg_tar)
this:
Unlike [pkg_tar from rules_pkg](https://github.com/bazelbuild/rules_pkg/blob/main/docs/latest.md#pkg_tar):
- Does not depend on any Python interpreter setup
- It does not depend on any Python interpreter setup
- The "manifest" specification is a mature public API and uses a compact tabular format, fixing
https://github.com/bazelbuild/rules_pkg/pull/238
- Does not have any custom program to produce the output, instead
we rely on a well-known C++ program called "tar".
- It doesn't rely custom program to produce the output, instead
we rely on the well-known C++ program called "tar".
Specifically, we use the BSD variant of tar since it provides a means
of controlling mtimes, uid, symlinks, etc.
We also provide full control for tar'ring binaries including their runfiles.
The `tar` binary is hermetic and fully statically-linked.
It is fetched as a toolchain from https://github.com/aspect-build/bsdtar-prebuilt.
## Examples
See the [`tar` tests](/lib/tests/tar/BUILD.bazel) for examples of usage.
## Mutating the tar contents
The `mtree_spec` rule can be used to create an mtree manifest for the tar file.
Expand All @@ -22,9 +28,11 @@ as the `mtree` attribute of the `tar` rule.
For example, to set the owner uid of files in the tar, you could:
```starlark
_TAR_SRCS = ["//some:files"]
mtree_spec(
name = "mtree",
srcs = ["//some:files"],
srcs = _TAR_SRCS,
)
mtree_mutate(
Expand All @@ -35,7 +43,7 @@ mtree_mutate(
tar(
name = "tar",
srcs = ["//some:files"],
srcs = _TAR_SRCS,
mtree = "change_owner",
)
```
Expand Down
91 changes: 49 additions & 42 deletions lib/tests/tar/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,20 @@ load("@aspect_bazel_lib//lib:copy_directory.bzl", "copy_directory")
load("@aspect_bazel_lib//lib:diff_test.bzl", "diff_test")
load("@aspect_bazel_lib//lib:tar.bzl", "mtree_mutate", "mtree_spec", "tar")
load("@aspect_bazel_lib//lib:testing.bzl", "assert_archive_contains")
load("@bazel_skylib//:bzl_library.bzl", "bzl_library")
load("@bazel_skylib//rules:write_file.bzl", "write_file")
load(":asserts.bzl", "assert_tar_listing")

# The examples below work with both source files and generated files.
# Here we generate a file to use in the examples.
write_file(
name = "fixture1",
out = "a",
out = "generated.txt",
content = ["hello a"],
)

# Case 1: Show that you can run any `tar` command you like, using a genrule.
#############
# Example 1: Show that you can run any `tar` command you like, using a genrule.
# This is advanced, atypical usage where you need such a level of control and don't want to use the `tar` rule.
genrule(
name = "tar_genrule",
srcs = [
Expand All @@ -33,15 +36,16 @@ assert_archive_contains(
name = "test_genrule",
archive = "1.tar",
expected = [
"lib/tests/tar/a",
"lib/tests/tar/generated.txt",
"lib/tests/tar/src_file",
],
)

# Case 2: demonstrate using a custom mtree formatted specification.
#############
# Example 2: exact control of the resulting tar file, using a custom specification in the "mtree" format.
# Copied from the output of `man tar`:
# An input file in mtree(5) format can be used to create an output
# archive with arbitrary ownership, permissions, or names that differ
# An input file in mtree(5) format can be used to create an output
# archive with arbitrary ownership, permissions, or names that differ
# from existing data on disk:
# $ cat input.mtree
# #mtree
Expand All @@ -66,22 +70,25 @@ assert_tar_listing(
],
)

# Case 3: compression
#############
# Example 3: compression.
# This uses gzip, see the `compress` attribute documentation for other legal values.
tar(
name = "tar_compress",
srcs = ["a"],
srcs = ["generated.txt"],
out = "3.tgz",
compress = "gzip",
)

assert_archive_contains(
name = "test_compress",
archive = "3.tgz",
expected = ["lib/tests/tar/a"],
expected = ["lib/tests/tar/generated.txt"],
type = "tar",
)

# Case 4: permit arbitrary flags
#############
# Example 4: you can pass arbitrary command-line flags to the bsdtar executable.
write_file(
name = "fixture4",
out = ".git",
Expand All @@ -92,8 +99,8 @@ tar(
name = "tar_flags",
srcs = [
".git",
"a",
"src_file",
":fixture1",
],
out = "4.tar",
# Due to this argument, .git should not appear in the resulting tar
Expand All @@ -107,12 +114,14 @@ assert_tar_listing(
"drwxr-xr-x 0 0 0 0 Jan 1 2023 lib/",
"drwxr-xr-x 0 0 0 0 Jan 1 2023 lib/tests/",
"drwxr-xr-x 0 0 0 0 Jan 1 2023 lib/tests/tar/",
"-rwxr-xr-x 0 0 0 7 Jan 1 2023 lib/tests/tar/a",
"-rwxr-xr-x 0 0 0 21 Jan 1 2023 lib/tests/tar/src_file",
"-rwxr-xr-x 0 0 0 7 Jan 1 2023 lib/tests/tar/generated.txt",
],
)

# Case 5: strip_prefix
#############
# Example 5: features like `strip_prefix` are supported by `mtree_mutate`.
# This lets you port code that used the `pkg_tar` rule from bazelbuild/rules_pkg.
_SRCS5 = [
":fixture1",
"src_file",
Expand Down Expand Up @@ -140,22 +149,13 @@ assert_tar_listing(
name = "test_strip_prefix",
actual = "tar_strip_prefix",
expected = [
"-rwxr-xr-x 0 0 0 7 Jan 1 2023 a",
"-rwxr-xr-x 0 0 0 7 Jan 1 2023 generated.txt",
"-rwxr-xr-x 0 0 0 21 Jan 1 2023 src_file",
],
)

bzl_library(
name = "asserts",
srcs = ["asserts.bzl"],
visibility = ["//visibility:public"],
deps = [
"//lib:diff_test",
"@bazel_skylib//rules:write_file",
],
)

# Case 6: Runfiles
#############
# Example 6: When archiving a binary, the "runfiles" are included.
sh_binary(
name = "cat_src_file",
srcs = ["cat_src_file.sh"],
Expand Down Expand Up @@ -199,7 +199,9 @@ diff_test(
file2 = "cat_src_file_output",
)

# Case 7: treeartifacts and source directories
#############
# Example 7: You can archive directories,
# both those in the source tree and those produced by rules that understand "tree artifacts".
copy_directory(
name = "treeartifact",
src = "srcdir",
Expand All @@ -209,16 +211,14 @@ copy_directory(
tar(
name = "dirs",
# Note, testonly should be propagated, proven by
# % bazel query --output=label_kind 'attr("testonly", 1,lib/tests/tar:all)'
# % bazel query --output=label_kind 'attr("testonly", 1, lib/tests/tar:all)'
# mtree_spec rule //lib/tests/tar:_dirs.mtree
# tar rule //lib/tests/tar:dirs
testonly = True,
srcs = glob(["srcdir/**"]) + [
"treeartifact",
],
out = "7.tar",
# When running remote, BB lays out files with inodes that mtree optimizes into a hardlink
tags = ["no-remote-exec"],
)

assert_tar_listing(
Expand All @@ -239,7 +239,10 @@ assert_tar_listing(
],
)

# Case 8: setting owner of files
#############
# Example 8: arbitrary mutations of the mtree spec can be performed.
# Typically use the `mtree_mutate` rule which supports specific mutations using a more ergonomic API,
# see Example 12 below.
_SRCS8 = [
":fixture1",
"src_file",
Expand Down Expand Up @@ -277,15 +280,15 @@ assert_tar_listing(
"drwxr-xr-x 0 1000 500 0 Jan 1 2023 lib/",
"drwxr-xr-x 0 1000 500 0 Jan 1 2023 lib/tests/",
"drwxr-xr-x 0 1000 500 0 Jan 1 2023 lib/tests/tar/",
"-rwxr-xr-x 0 1000 500 7 Jan 1 2023 lib/tests/tar/a",
"-rwxr-xr-x 0 1000 500 7 Jan 1 2023 lib/tests/tar/generated.txt",
"-rwxr-xr-x 0 1000 500 21 Jan 1 2023 lib/tests/tar/src_file",
],
)

# Case 9: Files from a different repository (#697)
#############
# Example 9: Files from a different repository (see #697)
# Note: This test uses an exported file from skylib, so we do not need to create
# an additional workspace just for this test.

tar(
name = "tar_different_repo",
srcs = ["@bazel_skylib//:LICENSE"],
Expand All @@ -300,25 +303,27 @@ assert_archive_contains(
],
)

# Case 10: Can reference generated files
#############
# Example 10: Similar to Example 9, you can reference generated files in the `mtree` attribute as well.
tar(
name = "tar_location_expansion",
srcs = ["@bazel_skylib//:LICENSE"],
out = "10.tar",
mtree = [
"a uid=0 gid=0 time=1672560000 mode=0755 type=file content=$(location @bazel_skylib//:LICENSE)",
"license uid=0 gid=0 time=1672560000 mode=0755 type=file content=$(location @bazel_skylib//:LICENSE)",
],
)

assert_tar_listing(
name = "test_tar_location_expansion",
actual = "tar_location_expansion",
expected = [
"-rwxr-xr-x 0 0 0 11358 Jan 1 2023 a",
"-rwxr-xr-x 0 0 0 11358 Jan 1 2023 license",
],
)

# Case 11: Can create tar without srcs
#############
# Example 11: You can create a tar without srcs, only empty directories
tar(
name = "create_tmp",
mtree = ["./tmp time=1501783453.0 mode=1777 gid=0 uid=0 type=dir"],
Expand All @@ -332,7 +337,8 @@ assert_tar_listing(
],
)

# Case 12: arbitrary mtree modifications
#############
# Example 12: arbitrary mtree modifications
mtree_mutate(
name = "modified1",
mtree = "source-casync.mtree",
Expand Down Expand Up @@ -360,14 +366,15 @@ diff_test(
file2 = "expected2.mtree",
)

# Case 13: Ensure that multiple entries at the root directory are handled correctly (bug #851)
#############
# Example 13: Ensure that multiple entries at the root directory are handled correctly (bug #851)
# NOTE: The mtree_spec part of this test is placed at the root BUILD.bazel because
# that's the only way to ensure that the mtree_spec generates single-component
# entries (which would trigger the bug).
exports_files(["expected13.mtree"])

# Case 14: Ensure mtree_mutate correctly handles prefix stripping for top-level directories (bug #851)

#############
# Example 14: Ensure mtree_mutate correctly handles prefix stripping for top-level directories (bug #851)
write_file(
name = "test14_main",
out = "14project/__main__.py",
Expand Down

0 comments on commit 109f32e

Please sign in to comment.