Skip to content

Commit

Permalink
Merge pull request #1959 from apache/tristan/fix-list-content-symlinks
Browse files Browse the repository at this point in the history
Fix `bst artifact list-content --long` with symlinks
  • Loading branch information
gtristan authored Oct 4, 2024
2 parents 152e5fe + fa3d663 commit 0edf2dc
Show file tree
Hide file tree
Showing 9 changed files with 88 additions and 34 deletions.
2 changes: 1 addition & 1 deletion src/buildstream/_frontend/widget.py
Original file line number Diff line number Diff line change
Expand Up @@ -1040,7 +1040,7 @@ def _get_filestats(self, directory, filename, filestat, list_long=False):
+ "{}".format(filename)
)
elif filestat.file_type == FileType.SYMLINK:
target = directory.readlink(*filename.split(os.path.sep))
target = directory.readlink(filename)
return (
"lrwxrwxrwx link {}".format(size)
+ "{} ".format(" " * (max_v_len - len(size)))
Expand Down
90 changes: 57 additions & 33 deletions tests/frontend/artifact_list_contents.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,47 +25,61 @@
# Project directory
DATA_DIR = os.path.join(
os.path.dirname(os.path.realpath(__file__)),
"project",
"artifact_list_contents",
)


def prepare_symlink(project):
# Create the link before running the tests.
# This is needed for users working on Windows, git checks out symlinks as files which content is the name
# of the symlink and the test therefore doesn't have the correct content
os.symlink(
os.path.join("..", "basicfile"),
os.path.join(project, "files", "files-and-links", "basicfolder", "basicsymlink"),
)


@pytest.mark.datafiles(DATA_DIR)
@pytest.mark.parametrize("target", ["element-name", "artifact-name"])
@pytest.mark.parametrize("with_project", [True, False], ids=["with-project", "without-project"])
def test_artifact_list_exact_contents(cli, datafiles, target, with_project):
project = str(datafiles)

# Get the cache key of our test element
key = cli.get_element_key(project, "import-bin.bst")
prepare_symlink(project)

# Ensure we have an artifact to read
result = cli.run(project=project, args=["build", "import-bin.bst"])
result = cli.run(project=project, args=["build", "target.bst"])
result.assert_success()

if target == "element-name":
arg = "import-bin.bst"
arg_bin = "import-bin.bst"
arg_links = "import-links.bst"
elif target == "artifact-name":
key = cli.get_element_key(project, "import-bin.bst")
arg = "test/import-bin/" + key
key_bin = cli.get_element_key(project, "import-bin.bst")
key_links = cli.get_element_key(project, "import-links.bst")
arg_bin = "test/import-bin/" + key_bin
arg_links = "test/import-links/" + key_links
else:
assert False, "unreachable"

# Delete the project.conf if we're going to try this without a project
if not with_project:
os.remove(os.path.join(project, "project.conf"))

# List the contents via the key
result = cli.run(project=project, args=["artifact", "list-contents", arg])
expected_output_bin = ("{target}:\n" "\tusr\n" "\tusr/bin\n" "\tusr/bin/hello\n\n").format(target=arg_bin)
expected_output_links = (
"{target}:\n" "\tbasicfile\n" "\tbasicfolder\n" "\tbasicfolder/basicsymlink\n" "\tbasicfolder/subdir-file\n\n"
).format(target=arg_links)

# Expect to fail if we try to list by element name and there is no project
if target == "element-name" and not with_project:
result.assert_main_error(ErrorDomain.STREAM, "project-not-loaded")
else:
result.assert_success()
for arg, expected_output in [(arg_bin, expected_output_bin), (arg_links, expected_output_links)]:
# List the contents via the key
result = cli.run(project=project, args=["artifact", "list-contents", arg])

expected_output_template = "{target}:\n\tusr\n\tusr/bin\n\tusr/bin/hello\n\n"
expected_output = expected_output_template.format(target=arg)
assert expected_output in result.output
# Expect to fail if we try to list by element name and there is no project
if target == "element-name" and not with_project:
result.assert_main_error(ErrorDomain.STREAM, "project-not-loaded")
else:
result.assert_success()
assert expected_output in result.output


# NOTE: The pytest-datafiles package has an issue where it fails to transfer any
Expand All @@ -81,36 +95,48 @@ def test_artifact_list_exact_contents(cli, datafiles, target, with_project):
@pytest.mark.parametrize("target", ["element-name", "artifact-name"])
def test_artifact_list_exact_contents_long(cli, datafiles, target):
project = str(datafiles)
prepare_symlink(project)

# Ensure we have an artifact to read
result = cli.run(project=project, args=["build", "import-bin.bst"])
result = cli.run(project=project, args=["build", "target.bst"])
assert result.exit_code == 0

if target == "element-name":
arg = "import-bin.bst"
arg_bin = "import-bin.bst"
arg_links = "import-links.bst"
elif target == "artifact-name":
key = cli.get_element_key(project, "import-bin.bst")
arg = "test/import-bin/" + key
key_bin = cli.get_element_key(project, "import-bin.bst")
key_links = cli.get_element_key(project, "import-links.bst")
arg_bin = "test/import-bin/" + key_bin
arg_links = "test/import-links/" + key_links
else:
assert False, "unreachable"

# List the contents via the element name
result = cli.run(project=project, args=["artifact", "list-contents", "--long", arg])
assert result.exit_code == 0
expected_output_template = (
expected_output_bin = (
"{target}:\n"
"\tdrwxr-xr-x dir 0 usr\n"
"\tdrwxr-xr-x dir 0 usr/bin\n"
"\t-rwxr-xr-x exe 28 usr/bin/hello\n\n"
)
expected_output = expected_output_template.format(target=arg)
).format(target=arg_bin)
expected_output_links = (
"{target}:\n"
"\t-rw-r--r-- reg 14 basicfile\n"
"\tdrwxr-xr-x dir 0 basicfolder\n"
"\tlrwxrwxrwx link 12 basicfolder/basicsymlink -> ../basicfile\n"
"\t-rw-r--r-- reg 0 basicfolder/subdir-file\n\n"
).format(target=arg_links)

assert expected_output in result.output
# List the contents via the element name
for arg, expected_output in [(arg_bin, expected_output_bin), (arg_links, expected_output_links)]:
result = cli.run(project=project, args=["artifact", "list-contents", "--long", arg])
assert result.exit_code == 0
assert expected_output in result.output


@pytest.mark.datafiles(DATA_DIR)
def test_artifact_list_exact_contents_glob(cli, datafiles):
project = str(datafiles)
prepare_symlink(project)

# Ensure we have an artifact to read
result = cli.run(project=project, args=["build", "target.bst"])
Expand All @@ -122,14 +148,12 @@ def test_artifact_list_exact_contents_glob(cli, datafiles):

# get the cahe keys for each element in the glob
import_bin_key = cli.get_element_key(project, "import-bin.bst")
import_dev_key = cli.get_element_key(project, "import-dev.bst")
compose_all_key = cli.get_element_key(project, "compose-all.bst")
import_links_key = cli.get_element_key(project, "import-links.bst")
target_key = cli.get_element_key(project, "target.bst")

expected_artifacts = [
"test/import-bin/" + import_bin_key,
"test/import-dev/" + import_dev_key,
"test/compose-all/" + compose_all_key,
"test/import-links/" + import_links_key,
"test/target/" + target_key,
]

Expand Down
4 changes: 4 additions & 0 deletions tests/frontend/artifact_list_contents/elements/import-bin.bst
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
kind: import
sources:
- kind: local
path: files/bin-files
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
kind: import
sources:
- kind: local
path: files/files-and-links
8 changes: 8 additions & 0 deletions tests/frontend/artifact_list_contents/elements/target.bst
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
kind: stack
description: |

Main stack target for the bst build test

depends:
- import-bin.bst
- import-links.bst
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
#!/bin/bash

echo "Hello !"
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
file contents
Empty file.
10 changes: 10 additions & 0 deletions tests/frontend/artifact_list_contents/project.conf
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
# Project config for frontend build test
name: test
min-version: 2.0
element-path: elements

plugins:
- origin: pip
package-name: sample-plugins
sources:
- git

0 comments on commit 0edf2dc

Please sign in to comment.