Skip to content

Commit

Permalink
refactor: Successor getting with separation of concerns
Browse files Browse the repository at this point in the history
Signed-off-by: Terry Howe <terrylhowe@gmail.com>
  • Loading branch information
TerryHowe committed Jul 26, 2024
1 parent c420cb1 commit 5d5d6ee
Show file tree
Hide file tree
Showing 22 changed files with 476 additions and 269 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ debug

# Custom
coverage.txt
coverage-all.txt
test/e2e/coverage.txt
**/covcounters.*
**/covmeta.*
Expand Down
3 changes: 2 additions & 1 deletion Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,8 @@ LDFLAGS += -X $(PROJECT_PKG)/internal/version.GitTreeState=${GIT_DIRTY}

.PHONY: test
test: tidy vendor check-encoding ## tidy and run tests
$(GO_EXE) test -race -v -coverprofile=coverage.txt -covermode=atomic -coverpkg=./... ./...
$(GO_EXE) test -race -v -coverprofile=coverage-all.txt -covermode=atomic -coverpkg=./... ./...
@grep -v testutil coverage-all.txt >coverage.txt; rm -f coverage-all.txt

.PHONY: teste2e
teste2e: ## run end to end tests
Expand Down
2 changes: 1 addition & 1 deletion cmd/oras/internal/display/status/progress/manager_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import (
"testing"

"oras.land/oras/cmd/oras/internal/display/status/console"
"oras.land/oras/cmd/oras/internal/display/status/console/testutils"
"oras.land/oras/internal/testutils"
)

func Test_manager_render(t *testing.T) {
Expand Down
2 changes: 1 addition & 1 deletion cmd/oras/internal/display/status/progress/status_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,8 @@ import (

ocispec "github.com/opencontainers/image-spec/specs-go/v1"
"oras.land/oras/cmd/oras/internal/display/status/console"
"oras.land/oras/cmd/oras/internal/display/status/console/testutils"
"oras.land/oras/cmd/oras/internal/display/status/progress/humanize"
"oras.land/oras/internal/testutils"
)

func Test_status_String(t *testing.T) {
Expand Down
20 changes: 16 additions & 4 deletions cmd/oras/internal/display/status/text.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,13 +17,13 @@ package status

import (
"context"
"oras.land/oras/internal/graph"
"sync"

"oras.land/oras/cmd/oras/internal/output"

ocispec "github.com/opencontainers/image-spec/specs-go/v1"
"oras.land/oras-go/v2"
"oras.land/oras-go/v2/content"
"oras.land/oras/cmd/oras/internal/output"
)

// TextPushHandler handles text status output for push events.
Expand Down Expand Up @@ -65,9 +65,15 @@ func (ph *TextPushHandler) UpdateCopyOptions(opts *oras.CopyGraphOptions, fetche
}
opts.PostCopy = func(ctx context.Context, desc ocispec.Descriptor) error {
committed.Store(desc.Digest.String(), desc.Annotations[ocispec.AnnotationTitle])
if err := output.PrintSuccessorStatus(ctx, desc, fetcher, committed, ph.printer.StatusPrinter(PushPromptSkipped)); err != nil {
successors, err := graph.FilteredSuccessors(ctx, desc, fetcher, DeduplicatedFilter(committed))
if err != nil {
return err
}
for _, successor := range successors {
if err = ph.printer.PrintStatus(successor, PushPromptSkipped); err != nil {
return err

Check warning on line 74 in cmd/oras/internal/display/status/text.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/text.go#L74

Added line #L74 was not covered by tests
}
}
return ph.printer.PrintStatus(desc, PushPromptUploaded)
}
}
Expand Down Expand Up @@ -149,9 +155,15 @@ func (ch *TextCopyHandler) PreCopy(_ context.Context, desc ocispec.Descriptor) e
// PostCopy implements PostCopy of CopyHandler.
func (ch *TextCopyHandler) PostCopy(ctx context.Context, desc ocispec.Descriptor) error {
ch.committed.Store(desc.Digest.String(), desc.Annotations[ocispec.AnnotationTitle])
if err := output.PrintSuccessorStatus(ctx, desc, ch.fetcher, ch.committed, ch.printer.StatusPrinter(copyPromptSkipped)); err != nil {
successors, err := graph.FilteredSuccessors(ctx, desc, ch.fetcher, DeduplicatedFilter(ch.committed))
if err != nil {
return err
}
for _, successor := range successors {
if err = ch.printer.PrintStatus(successor, copyPromptSkipped); err != nil {
return err

Check warning on line 164 in cmd/oras/internal/display/status/text.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/text.go#L164

Added line #L164 was not covered by tests
}
}
return ch.printer.PrintStatus(desc, copyPromptCopied)
}

Expand Down
131 changes: 131 additions & 0 deletions cmd/oras/internal/display/status/text_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
/*
Copyright The ORAS Authors.
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.
*/

package status

import (
"bytes"
"context"
"encoding/json"
"fmt"
"github.com/opencontainers/go-digest"
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
"oras.land/oras-go/v2/content/memory"
"oras.land/oras/cmd/oras/internal/output"
"oras.land/oras/internal/testutils"
"os"
"strings"
"testing"
)

var (
ctx context.Context
builder *strings.Builder
printer *output.Printer
bogus ocispec.Descriptor
memStore *memory.Store
memDesc ocispec.Descriptor
manifestDesc ocispec.Descriptor
)

func TestMain(m *testing.M) {
// memory store for testing
memStore = memory.New()
content := []byte("test")
r := bytes.NewReader(content)
memDesc = ocispec.Descriptor{
MediaType: "application/octet-stream",
Digest: digest.FromBytes(content),
Size: int64(len(content)),
}
if err := memStore.Push(context.Background(), memDesc, r); err != nil {
fmt.Println("Setup failed:", err)
os.Exit(1)
}
if err := memStore.Tag(context.Background(), memDesc, memDesc.Digest.String()); err != nil {
fmt.Println("Setup failed:", err)
os.Exit(1)
}

layer1Desc := memDesc
layer1Desc.Annotations = map[string]string{ocispec.AnnotationTitle: "layer1"}
layer2Desc := memDesc
layer2Desc.Annotations = map[string]string{ocispec.AnnotationTitle: "layer2"}
manifest := ocispec.Manifest{
MediaType: ocispec.MediaTypeImageManifest,
Layers: []ocispec.Descriptor{layer1Desc, layer2Desc},
Config: memDesc,
}
manifestContent, err := json.Marshal(&manifest)
if err != nil {
fmt.Println("Setup failed:", err)
os.Exit(1)
}
manifestDesc = ocispec.Descriptor{
MediaType: manifest.MediaType,
Size: int64(len(manifestContent)),
Digest: digest.FromBytes(manifestContent),
}
if err := memStore.Push(context.Background(), manifestDesc, strings.NewReader(string(manifestContent))); err != nil {
fmt.Println("Setup failed:", err)
os.Exit(1)
}
if err := memStore.Tag(context.Background(), memDesc, memDesc.Digest.String()); err != nil {
fmt.Println("Setup failed:", err)
os.Exit(1)
}

ctx = context.Background()
builder = &strings.Builder{}
printer = output.NewPrinter(builder, os.Stderr, false)
bogus = ocispec.Descriptor{MediaType: ocispec.MediaTypeImageManifest}
m.Run()
}

func TestTextCopyHandler_OnMounted(t *testing.T) {
fetcher := testutils.NewMockFetcher(t)
ch := NewTextCopyHandler(printer, fetcher.Fetcher)
if ch.OnMounted(ctx, fetcher.OciImage) != nil {
t.Error("OnMounted() should not return an error")
}

}

func TestTextCopyHandler_OnCopySkipped(t *testing.T) {
fetcher := testutils.NewMockFetcher(t)
ch := NewTextCopyHandler(printer, fetcher.Fetcher)
if ch.OnCopySkipped(ctx, fetcher.OciImage) != nil {
t.Error("OnCopySkipped() should not return an error")
}
}

func TestTextCopyHandler_PostCopy(t *testing.T) {
fetcher := testutils.NewMockFetcher(t)
ch := NewTextCopyHandler(printer, fetcher.Fetcher)
if ch.PostCopy(ctx, fetcher.OciImage) != nil {
t.Error("PostCopy() should not return an error")
}
if ch.PostCopy(ctx, bogus) == nil {
t.Error("PostCopy() should return an error")
}
}

func TestTextCopyHandler_PreCopy(t *testing.T) {
fetcher := testutils.NewMockFetcher(t)
ch := NewTextCopyHandler(printer, fetcher.Fetcher)
if ch.PreCopy(ctx, fetcher.OciImage) != nil {
t.Error("PreCopy() should not return an error")
}
}
2 changes: 1 addition & 1 deletion cmd/oras/internal/display/status/track/target_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ import (
"oras.land/oras-go/v2"
"oras.land/oras-go/v2/content/memory"
"oras.land/oras-go/v2/registry/remote"
"oras.land/oras/cmd/oras/internal/display/status/console/testutils"
"oras.land/oras/internal/testutils"
)

type testReferenceGraphTarget struct {
Expand Down
17 changes: 12 additions & 5 deletions cmd/oras/internal/display/status/tty.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,11 +17,10 @@ package status

import (
"context"
"oras.land/oras/internal/graph"
"os"
"sync"

"oras.land/oras/cmd/oras/internal/output"

ocispec "github.com/opencontainers/image-spec/specs-go/v1"
"oras.land/oras-go/v2"
"oras.land/oras-go/v2/content"
Expand Down Expand Up @@ -70,9 +69,17 @@ func (ph *TTYPushHandler) UpdateCopyOptions(opts *oras.CopyGraphOptions, fetcher
}
opts.PostCopy = func(ctx context.Context, desc ocispec.Descriptor) error {
committed.Store(desc.Digest.String(), desc.Annotations[ocispec.AnnotationTitle])
return output.PrintSuccessorStatus(ctx, desc, fetcher, committed, func(d ocispec.Descriptor) error {
return ph.tracked.Prompt(d, PushPromptSkipped)
})
successors, err := graph.FilteredSuccessors(ctx, desc, fetcher, DeduplicatedFilter(committed))
if err != nil {
return err

Check warning on line 74 in cmd/oras/internal/display/status/tty.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/tty.go#L74

Added line #L74 was not covered by tests
}
for _, successor := range successors {
err = ph.tracked.Prompt(successor, PushPromptSkipped)
if err != nil {
return err

Check warning on line 79 in cmd/oras/internal/display/status/tty.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/tty.go#L79

Added line #L79 was not covered by tests
}
}
return nil
}
}

Expand Down
115 changes: 115 additions & 0 deletions cmd/oras/internal/display/status/tty_console_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
//go:build freebsd || linux || netbsd || openbsd || solaris

/*
Copyright The ORAS Authors.
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.
*/

package status

import (
"context"
"oras.land/oras-go/v2"
"oras.land/oras-go/v2/content/memory"
"oras.land/oras/cmd/oras/internal/display/status/track"
"oras.land/oras/internal/testutils"
"testing"
)

func TestTTYPushHandler_TrackTarget(t *testing.T) {
// prepare pty
_, slave, err := testutils.NewPty()
if err != nil {
t.Fatal(err)
}
defer slave.Close()
ph := NewTTYPushHandler(slave)
store := memory.New()
// test
_, fn, err := ph.TrackTarget(store)
if err != nil {
t.Fatal("TrackTarget() should not return an error")
}
defer func() {
if err := fn(); err != nil {
t.Fatal(err)
}
}()
if ttyPushHandler, ok := ph.(*TTYPushHandler); !ok {
t.Fatalf("TrackTarget() should return a *TTYPushHandler, got %T", ttyPushHandler)
}
}

func TestTTYPushHandler_UpdateCopyOptions(t *testing.T) {
// prepare pty
pty, slave, err := testutils.NewPty()
if err != nil {
t.Fatal(err)
}
defer slave.Close()
ph := NewTTYPushHandler(slave)
gt, _, err := ph.TrackTarget(memory.New())
if err != nil {
t.Fatalf("TrackTarget() should not return an error: %v", err)
}
// test
opts := oras.CopyGraphOptions{}
ph.UpdateCopyOptions(&opts, memStore)
if err := oras.CopyGraph(context.Background(), memStore, gt, manifestDesc, opts); err != nil {
t.Fatalf("CopyGraph() should not return an error: %v", err)
}
if err := oras.CopyGraph(context.Background(), memStore, gt, manifestDesc, opts); err != nil {
t.Fatalf("CopyGraph() should not return an error: %v", err)
}
if tracked, ok := gt.(track.GraphTarget); !ok {
t.Fatalf("TrackTarget() should return a *track.GraphTarget, got %T", tracked)
} else {
_ = tracked.Close()
}
// validate
if err = testutils.MatchPty(pty, slave, "Exists", manifestDesc.MediaType, "100.00%", manifestDesc.Digest.String()); err != nil {
t.Fatal(err)
}
}

func Test_TTYPullHandler_TrackTarget(t *testing.T) {
src := memory.New()
t.Run("has TTY", func(t *testing.T) {
_, device, err := testutils.NewPty()
if err != nil {
t.Fatal(err)
}
defer device.Close()
ph := NewTTYPullHandler(device)
got, fn, err := ph.TrackTarget(src)
if err != nil {
t.Fatal(err)
}
defer func() {
if err := fn(); err != nil {
t.Fatal(err)
}
}()
if got == src {
t.Fatal("GraphTarget not be modified on TTY")
}
})

t.Run("invalid TTY", func(t *testing.T) {
ph := NewTTYPullHandler(nil)

if _, _, err := ph.TrackTarget(src); err == nil {
t.Fatal("expected error for no tty but got nil")
}
})
}
Loading

0 comments on commit 5d5d6ee

Please sign in to comment.