Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: simplify track reader manager and messenger use #1474

Draft
wants to merge 3 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
71 changes: 44 additions & 27 deletions cmd/oras/internal/display/status/console/console.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,9 @@
package console

import (
"os"

"github.com/containerd/console"
containerd "github.com/containerd/console"
"github.com/morikuni/aec"
"os"
)

const (
Expand All @@ -33,60 +32,78 @@
Restore = "\0338"
)

// Console is a wrapper around containerd's console.Console and ANSI escape
// codes.
type Console struct {
console.Console
// Console is a wrapper around containerd's Console and ANSI escape codes.
type Console interface {
containerd.Console
GetHeightWidth() (height, width int)
Save()
NewRow()
OutputTo(upCnt uint, str string)
Restore()
}

type console struct {
containerd.Console
}

// NewConsole generates a console from a file.
func NewConsole(f *os.File) (Console, error) {
if f != nil && f.Name() == os.DevNull {
return NewDiscardConsole(f), nil
}
c, err := containerd.ConsoleFromFile(f)
if err != nil {
return nil, err
}
return &console{c}, nil
}

// Size returns the width and height of the console.
// If the console size cannot be determined, returns a default value of 80x10.
func (c *Console) Size() (width, height int) {
width = MinWidth
height = MinHeight
size, err := c.Console.Size()
if err == nil {
if size.Height > MinHeight {
height = int(size.Height)
func (c *console) Size() (size containerd.WinSize, err error) {
size, err = c.Console.Size()
if err != nil {
size.Height = MinHeight
size.Width = MinWidth

Check warning on line 67 in cmd/oras/internal/display/status/console/console.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/console/console.go#L66-L67

Added lines #L66 - L67 were not covered by tests
} else {
if size.Height < MinHeight {
size.Height = MinHeight
}
if size.Width > MinWidth {
width = int(size.Width)
if size.Width < MinWidth {
size.Width = MinWidth
}
}
return
}

// New generates a Console from a file.
func New(f *os.File) (*Console, error) {
c, err := console.ConsoleFromFile(f)
if err != nil {
return nil, err
}
return &Console{c}, nil
// GetHeightWidth returns the width and height of the console.
func (c *console) GetHeightWidth() (height, width int) {
windowSize, _ := c.Size()
return int(windowSize.Height), int(windowSize.Width)
}

// Save saves the current cursor position.
func (c *Console) Save() {
func (c *console) Save() {
_, _ = c.Write([]byte(aec.Hide.Apply(Save)))
}

// NewRow allocates a horizontal space to the output area with scroll if needed.
func (c *Console) NewRow() {
func (c *console) NewRow() {
_, _ = c.Write([]byte(Restore))
_, _ = c.Write([]byte("\n"))
_, _ = c.Write([]byte(Save))
}

// OutputTo outputs a string to a specific line.
func (c *Console) OutputTo(upCnt uint, str string) {
func (c *console) OutputTo(upCnt uint, str string) {
_, _ = c.Write([]byte(Restore))
_, _ = c.Write([]byte(aec.PreviousLine(upCnt).Apply(str)))
_, _ = c.Write([]byte("\n"))
_, _ = c.Write([]byte(aec.EraseLine(aec.EraseModes.Tail).String()))
}

// Restore restores the saved cursor position.
func (c *Console) Restore() {
func (c *console) Restore() {
// cannot use aec.Restore since DEC has better compatibility than SCO
_, _ = c.Write([]byte(Restore))
_, _ = c.Write([]byte(aec.Column(0).
Expand Down
20 changes: 10 additions & 10 deletions cmd/oras/internal/display/status/console/console_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ package console
import (
"testing"

"github.com/containerd/console"
containerd "github.com/containerd/console"
)

func validateSize(t *testing.T, gotWidth, gotHeight, wantWidth, wantHeight int) {
Expand All @@ -34,30 +34,30 @@ func validateSize(t *testing.T, gotWidth, gotHeight, wantWidth, wantHeight int)
}

func TestConsole_Size(t *testing.T) {
pty, _, err := console.NewPty()
pty, _, err := containerd.NewPty()
if err != nil {
t.Fatal(err)
}
c := &Console{
c := &console{
Console: pty,
}

// minimal width and height
gotWidth, gotHeight := c.Size()
gotHeight, gotWidth := c.GetHeightWidth()
validateSize(t, gotWidth, gotHeight, MinWidth, MinHeight)

// zero width
_ = pty.Resize(console.WinSize{Width: 0, Height: MinHeight})
gotWidth, gotHeight = c.Size()
_ = pty.Resize(containerd.WinSize{Width: 0, Height: MinHeight})
gotHeight, gotWidth = c.GetHeightWidth()
validateSize(t, gotWidth, gotHeight, MinWidth, MinHeight)

// zero height
_ = pty.Resize(console.WinSize{Width: MinWidth, Height: 0})
gotWidth, gotHeight = c.Size()
_ = pty.Resize(containerd.WinSize{Width: MinWidth, Height: 0})
gotHeight, gotWidth = c.GetHeightWidth()
validateSize(t, gotWidth, gotHeight, MinWidth, MinHeight)

// valid zero and height
_ = pty.Resize(console.WinSize{Width: 200, Height: 100})
gotWidth, gotHeight = c.Size()
_ = pty.Resize(containerd.WinSize{Width: 200, Height: 100})
gotHeight, gotWidth = c.GetHeightWidth()
validateSize(t, gotWidth, gotHeight, 200, 100)
}
85 changes: 85 additions & 0 deletions cmd/oras/internal/display/status/console/discard.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
/*
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 console

import (
"os"

containerd "github.com/containerd/console"
)

type discardConsole struct {
*os.File
}

func NewDiscardConsole(f *os.File) Console {
dc := discardConsole{
File: f,
}
return &dc
}

// Fd returns its file descriptor
func (mc *discardConsole) Fd() uintptr {
return os.Stderr.Fd()

Check warning on line 37 in cmd/oras/internal/display/status/console/discard.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/console/discard.go#L36-L37

Added lines #L36 - L37 were not covered by tests
}

// Name returns its file name
func (mc *discardConsole) Name() string {
return mc.File.Name()

Check warning on line 42 in cmd/oras/internal/display/status/console/discard.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/console/discard.go#L41-L42

Added lines #L41 - L42 were not covered by tests
}

func (mc *discardConsole) Resize(_ containerd.WinSize) error {
return nil
}

func (mc *discardConsole) ResizeFrom(containerd.Console) error {
return nil
}
func (mc *discardConsole) SetRaw() error {
return nil
}
func (mc *discardConsole) DisableEcho() error {
return nil
}
func (mc *discardConsole) Reset() error {
return nil
}
func (mc *discardConsole) Size() (containerd.WinSize, error) {
ws := containerd.WinSize{
Width: 80,
Height: 24,
}
return ws, nil
}

// GetHeightWidth returns the width and height of the console.
func (mc *discardConsole) GetHeightWidth() (height, width int) {
windowSize, _ := mc.Size()
return int(windowSize.Height), int(windowSize.Width)
}

func (mc *discardConsole) Save() {
}

func (mc *discardConsole) NewRow() {
}

func (mc *discardConsole) OutputTo(_ uint, _ string) {

Check warning on line 81 in cmd/oras/internal/display/status/console/discard.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/console/discard.go#L81

Added line #L81 was not covered by tests
}

func (mc *discardConsole) Restore() {

Check warning on line 84 in cmd/oras/internal/display/status/console/discard.go

View check run for this annotation

Codecov / codecov/patch

cmd/oras/internal/display/status/console/discard.go#L84

Added line #L84 was not covered by tests
}
65 changes: 65 additions & 0 deletions cmd/oras/internal/display/status/console/discard_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/*
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 console

import (
"os"
"testing"

containerd "github.com/containerd/console"
)

func TestConsole_New(t *testing.T) {
mockFile, err := os.OpenFile(os.DevNull, os.O_RDWR, 0666)
if err != nil {
t.Fatalf("Unexpected error %v", err)
}

sut, err := NewConsole(mockFile)
if err != nil {
t.Errorf("Unexpected error %v", err)
}

if err = sut.Resize(containerd.WinSize{}); err != nil {
t.Errorf("Unexpected erro for Resize: %v", err)
}
if err = sut.ResizeFrom(nil); err != nil {
t.Errorf("Unexpected erro for Resize: %v", err)
}
if err = sut.SetRaw(); err != nil {
t.Errorf("Unexpected erro for Resize: %v", err)
}
if err = sut.DisableEcho(); err != nil {
t.Errorf("Unexpected erro for Resize: %v", err)
}
if err = sut.Reset(); err != nil {
t.Errorf("Unexpected erro for Resize: %v", err)
}
windowSize, _ := sut.Size()
if windowSize.Height != 24 {
t.Errorf("Expected size 24 actual %d", windowSize.Height)
}
if windowSize.Width != 80 {
t.Errorf("Expected size 80 actual %d", windowSize.Width)
}
h, w := sut.GetHeightWidth()
if h != 24 {
t.Errorf("Expected size 24 actual %d", h)
}
if w != 80 {
t.Errorf("Expected size 80 actual %d", w)
}
}
18 changes: 11 additions & 7 deletions cmd/oras/internal/display/status/progress/manager.go
Original file line number Diff line number Diff line change
Expand Up @@ -44,20 +44,24 @@ type Manager interface {
type manager struct {
status []*status
statusLock sync.RWMutex
console *console.Console
console console.Console
actionPrompt string
donePrompt string
updating sync.WaitGroup
renderDone chan struct{}
renderClosed chan struct{}
}

// NewManager initialized a new progress manager.
func NewManager(f *os.File) (Manager, error) {
c, err := console.New(f)
func NewManager(actionPrompt string, donePrompt string, tty *os.File) (Manager, error) {
c, err := console.NewConsole(tty)
if err != nil {
return nil, err
}
m := &manager{
console: c,
actionPrompt: actionPrompt,
donePrompt: donePrompt,
renderDone: make(chan struct{}),
renderClosed: make(chan struct{}),
}
Expand Down Expand Up @@ -88,7 +92,7 @@ func (m *manager) render() {
m.statusLock.RLock()
defer m.statusLock.RUnlock()
// todo: update size in another routine
width, height := m.console.Size()
height, width := m.console.GetHeightWidth()
lineCount := len(m.status) * 2
offset := 0
if lineCount > height {
Expand Down Expand Up @@ -131,15 +135,15 @@ func (m *manager) SendAndStop(desc ocispec.Descriptor, prompt string) error {
}

func (m *manager) statusChan(s *status) *Messenger {
ch := make(chan *status, BufferSize)
messenger := NewMessenger(m.actionPrompt, m.donePrompt)
m.updating.Add(1)
go func() {
defer m.updating.Done()
for newStatus := range ch {
for newStatus := range messenger.ch {
s.update(newStatus)
}
}()
return &Messenger{ch: ch}
return messenger
}

// Close stops all status and waits for updating and rendering.
Expand Down
Loading
Loading