From 6e0cf9344778a82028c5bb35ab9528ec104af54e Mon Sep 17 00:00:00 2001 From: rbagd Date: Wed, 19 Apr 2023 00:20:47 +0200 Subject: [PATCH 1/2] Fixes format inconsistencies with docker for certain history fields Closes #17767 Closes #17768 System test for image list and history dates * Changed field separator in the test to `;` for easier parsing * Converted date output from image history and image list to be comparable Signed-off-by: rbagd --- cmd/podman/images/history.go | 8 ++------ test/e2e/history_test.go | 5 +++++ test/system/110-history.bats | 21 ++++++++++++++++----- 3 files changed, 23 insertions(+), 11 deletions(-) diff --git a/cmd/podman/images/history.go b/cmd/podman/images/history.go index 8f910f82da..b6b380658d 100644 --- a/cmd/podman/images/history.go +++ b/cmd/podman/images/history.go @@ -3,9 +3,7 @@ package images import ( "fmt" "os" - "strings" "time" - "unicode" "github.com/containers/common/pkg/report" "github.com/containers/podman/v4/cmd/podman/common" @@ -149,9 +147,7 @@ func (h historyReporter) Created() string { } func (h historyReporter) Size() string { - s := units.HumanSizeWithPrecision(float64(h.ImageHistoryLayer.Size), 3) - i := strings.LastIndexFunc(s, unicode.IsNumber) - return s[:i+1] + " " + s[i+1:] + return units.HumanSizeWithPrecision(float64(h.ImageHistoryLayer.Size), 3) } func (h historyReporter) CreatedBy() string { @@ -169,7 +165,7 @@ func (h historyReporter) ID() string { } func (h historyReporter) CreatedAt() string { - return time.Unix(h.ImageHistoryLayer.Created.Unix(), 0).UTC().String() + return time.Unix(h.ImageHistoryLayer.Created.Unix(), 0).Format(time.RFC3339) } func (h historyReporter) CreatedSince() string { diff --git a/test/e2e/history_test.go b/test/e2e/history_test.go index 8f0e08b125..a9be63b9ab 100644 --- a/test/e2e/history_test.go +++ b/test/e2e/history_test.go @@ -44,6 +44,11 @@ var _ = Describe("Podman history", func() { session.WaitWithDefaultTimeout() Expect(session).Should(Exit(0)) Expect(session.OutputToStringArray()).ToNot(BeEmpty()) + + session = podmanTest.Podman([]string{"history", "--format", "{{.CreatedAt}};{{.Size}}", ALPINE}) + session.WaitWithDefaultTimeout() + Expect(session).Should(Exit(0)) + Expect(session.OutputToString()).To(MatchRegexp("[0-9-]{10}T[0-9:]{8}[Z0-9+:-]+;[0-9.]+[MG]*B( .+)?")) }) It("podman history with human flag", func() { diff --git a/test/system/110-history.bats b/test/system/110-history.bats index da6f2177c6..bee533003e 100644 --- a/test/system/110-history.bats +++ b/test/system/110-history.bats @@ -57,15 +57,26 @@ size | -\\\?[0-9]\\\+ @test "podman image history Created" { # Values from image LIST - run_podman image list --format '{{.CreatedSince}}--{{.CreatedAt}}' $IMAGE + run_podman image list --format '{{.CreatedSince}};{{.CreatedAt}}' $IMAGE from_imagelist="$output" - assert "$from_imagelist" =~ "^[0-9].* ago--[0-9]+-[0-9]+-[0-9]+ [0-9:]+ " \ + assert "$from_imagelist" =~ "^[0-9].* ago;[0-9]+-[0-9]+-[0-9]+ [0-9:]+ " \ "CreatedSince and CreatedAt look reasonable" # Values from image HISTORY - run_podman image history --format '{{.CreatedSince}}--{{.CreatedAt}}' $IMAGE - assert "${lines[0]}" == "$from_imagelist" \ - "CreatedSince and CreatedAt from image history should == image list" + run_podman image history --format '{{.CreatedSince}};{{.CreatedAt}}' $IMAGE + from_imagehistory="${lines[0]}" + + imagelist_since=$(echo "$from_imagelist" | cut -d';' -f1) + imagehist_since=$(echo "$from_imagehistory" | cut -d';' -f1) + + assert "$imagehist_since" == "$imagelist_since" \ + "CreatedSince from image history should == image list" + + imagelist_at=$(date --rfc-3339=seconds -f <(echo "$from_imagelist" | cut -d';' -f2 | sed 's/ UTC//')) + imagehist_at=$(date --rfc-3339=seconds -f <(echo "$from_imagehistory" | cut -d';' -f2)) + + assert "$imagehist_at" == "$imagelist_at" \ + "CreatedAt from image history should == image list" } # vim: filetype=sh From bce38c1afb7b1687d19b85edddfde307872d15fe Mon Sep 17 00:00:00 2001 From: rbagd Date: Wed, 19 Apr 2023 21:10:40 +0200 Subject: [PATCH 2/2] Updated system test to be easier to read Following @edsantiago guidance, * Additional explanations for each step of the test * Timezone for tests normalized to UTC * Smarter choice of separator and use of shell substring extraction Signed-off-by: rbagd --- test/system/110-history.bats | 40 ++++++++++++++++++++---------------- 1 file changed, 22 insertions(+), 18 deletions(-) diff --git a/test/system/110-history.bats b/test/system/110-history.bats index bee533003e..47ad83ef8e 100644 --- a/test/system/110-history.bats +++ b/test/system/110-history.bats @@ -57,26 +57,30 @@ size | -\\\?[0-9]\\\+ @test "podman image history Created" { # Values from image LIST - run_podman image list --format '{{.CreatedSince}};{{.CreatedAt}}' $IMAGE - from_imagelist="$output" - assert "$from_imagelist" =~ "^[0-9].* ago;[0-9]+-[0-9]+-[0-9]+ [0-9:]+ " \ - "CreatedSince and CreatedAt look reasonable" - - # Values from image HISTORY - run_podman image history --format '{{.CreatedSince}};{{.CreatedAt}}' $IMAGE - from_imagehistory="${lines[0]}" - - imagelist_since=$(echo "$from_imagelist" | cut -d';' -f1) - imagehist_since=$(echo "$from_imagehistory" | cut -d';' -f1) - - assert "$imagehist_since" == "$imagelist_since" \ + run_podman image list --format '{{.CreatedSince}}\n{{.CreatedAt}}' $IMAGE + imagelist_since="${lines[0]}" + imagelist_at="${lines[1]}" + + assert "${imagelist_since}" =~ "^[0-9]+.* ago" \ + "image list: CreatedSince looks reasonable" + assert "${imagelist_at}" =~ "^[0-9]+-[0-9]+-[0-9]+ [0-9:]+ \+0000 UTC\$" \ + "image list: CreatedAt looks reasonable" + + # Values from image HISTORY. For docker compatibility, this command now + # honors $TZ (#18213) for CreatedAt. + TZ=UTC run_podman image history --format '{{.CreatedSince}}\n{{.CreatedAt}}' $IMAGE + imagehistory_since="${lines[0]}" + imagehistory_at="${lines[1]}" + + assert "$imagehistory_since" == "$imagelist_since" \ "CreatedSince from image history should == image list" - imagelist_at=$(date --rfc-3339=seconds -f <(echo "$from_imagelist" | cut -d';' -f2 | sed 's/ UTC//')) - imagehist_at=$(date --rfc-3339=seconds -f <(echo "$from_imagehistory" | cut -d';' -f2)) - - assert "$imagehist_at" == "$imagelist_at" \ - "CreatedAt from image history should == image list" + # More docker compatibility: both commands emit ISO8601-ish dates but + # with different separators so we need to compare date & time separately. + assert "${imagehistory_at:0:10}" == "${imagelist_at:0:10}" \ + "CreatedAt (date) from image history should == image list" + assert "${imagehistory_at:11:8}" == "${imagelist_at:11:8}" \ + "CreatedAt (time) from image history should == image list" } # vim: filetype=sh