first draft at self refreshing loading bar centralized/controlled by Ui
This commit is contained in:
parent
3940cd9c7f
commit
ddd96c513b
|
@ -9,6 +9,8 @@ import (
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/hashicorp/packer/packer"
|
||||||
)
|
)
|
||||||
|
|
||||||
// PackerKeyEnv is used to specify the key interval (delay) between keystrokes
|
// PackerKeyEnv is used to specify the key interval (delay) between keystrokes
|
||||||
|
@ -41,7 +43,7 @@ func SupportedProtocol(u *url.URL) bool {
|
||||||
|
|
||||||
// build a dummy NewDownloadClient since this is the only place that valid
|
// build a dummy NewDownloadClient since this is the only place that valid
|
||||||
// protocols are actually exposed.
|
// protocols are actually exposed.
|
||||||
cli := NewDownloadClient(&DownloadConfig{}, nil)
|
cli := NewDownloadClient(&DownloadConfig{}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
// Iterate through each downloader to see if a protocol was found.
|
// Iterate through each downloader to see if a protocol was found.
|
||||||
ok := false
|
ok := false
|
||||||
|
@ -173,7 +175,7 @@ func FileExistsLocally(original string) bool {
|
||||||
|
|
||||||
// First create a dummy downloader so we can figure out which
|
// First create a dummy downloader so we can figure out which
|
||||||
// protocol to use.
|
// protocol to use.
|
||||||
cli := NewDownloadClient(&DownloadConfig{}, nil)
|
cli := NewDownloadClient(&DownloadConfig{}, new(packer.NoopProgressBar))
|
||||||
d, ok := cli.config.DownloaderMap[u.Scheme]
|
d, ok := cli.config.DownloaderMap[u.Scheme]
|
||||||
if !ok {
|
if !ok {
|
||||||
return false
|
return false
|
||||||
|
|
|
@ -10,19 +10,17 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"hash"
|
"hash"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
|
||||||
|
|
||||||
// imports related to each Downloader implementation
|
"github.com/hashicorp/packer/packer" // imports related to each Downloader implementation
|
||||||
import (
|
|
||||||
"io"
|
|
||||||
"net/http"
|
|
||||||
"path/filepath"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// DownloadConfig is the configuration given to instantiate a new
|
// DownloadConfig is the configuration given to instantiate a new
|
||||||
|
@ -81,14 +79,9 @@ func HashForType(t string) hash.Hash {
|
||||||
|
|
||||||
// NewDownloadClient returns a new DownloadClient for the given
|
// NewDownloadClient returns a new DownloadClient for the given
|
||||||
// configuration.
|
// configuration.
|
||||||
func NewDownloadClient(c *DownloadConfig, bar ProgressBar) *DownloadClient {
|
func NewDownloadClient(c *DownloadConfig, bar packer.ProgressBar) *DownloadClient {
|
||||||
const mtu = 1500 /* ethernet */ - 20 /* ipv4 */ - 20 /* tcp */
|
const mtu = 1500 /* ethernet */ - 20 /* ipv4 */ - 20 /* tcp */
|
||||||
|
|
||||||
// If bar is nil, then use a dummy progress bar that doesn't do anything
|
|
||||||
if bar == nil {
|
|
||||||
bar = GetDummyProgressBar()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create downloader map if it hasn't been specified already.
|
// Create downloader map if it hasn't been specified already.
|
||||||
if c.DownloaderMap == nil {
|
if c.DownloaderMap == nil {
|
||||||
c.DownloaderMap = map[string]Downloader{
|
c.DownloaderMap = map[string]Downloader{
|
||||||
|
@ -237,7 +230,7 @@ type HTTPDownloader struct {
|
||||||
total uint64
|
total uint64
|
||||||
userAgent string
|
userAgent string
|
||||||
|
|
||||||
progress ProgressBar
|
progress packer.ProgressBar
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *HTTPDownloader) Cancel() {
|
func (d *HTTPDownloader) Cancel() {
|
||||||
|
@ -331,9 +324,10 @@ func (d *HTTPDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
d.total = d.current + uint64(resp.ContentLength)
|
d.total = d.current + uint64(resp.ContentLength)
|
||||||
|
|
||||||
bar := d.progress
|
bar := d.progress
|
||||||
bar.SetTotal64(int64(d.total))
|
log.Printf("this %#v", bar)
|
||||||
progressBar := bar.Start()
|
log.Printf("that")
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Start(d.total)
|
||||||
|
bar.Set(d.current)
|
||||||
|
|
||||||
var buffer [4096]byte
|
var buffer [4096]byte
|
||||||
for {
|
for {
|
||||||
|
@ -343,7 +337,7 @@ func (d *HTTPDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
d.current += uint64(n)
|
d.current += uint64(n)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
|
|
||||||
if _, werr := dst.Write(buffer[:n]); werr != nil {
|
if _, werr := dst.Write(buffer[:n]); werr != nil {
|
||||||
return werr
|
return werr
|
||||||
|
@ -353,7 +347,7 @@ func (d *HTTPDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
progressBar.Finish()
|
bar.Finish()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -374,7 +368,7 @@ type FileDownloader struct {
|
||||||
current uint64
|
current uint64
|
||||||
total uint64
|
total uint64
|
||||||
|
|
||||||
progress ProgressBar
|
progress packer.ProgressBar
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *FileDownloader) Progress() uint64 {
|
func (d *FileDownloader) Progress() uint64 {
|
||||||
|
@ -466,9 +460,9 @@ func (d *FileDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
d.total = uint64(fi.Size())
|
d.total = uint64(fi.Size())
|
||||||
|
|
||||||
bar := d.progress
|
bar := d.progress
|
||||||
bar.SetTotal64(int64(d.total))
|
|
||||||
progressBar := bar.Start()
|
bar.Start(d.total)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
|
|
||||||
// no bufferSize specified, so copy synchronously.
|
// no bufferSize specified, so copy synchronously.
|
||||||
if d.bufferSize == nil {
|
if d.bufferSize == nil {
|
||||||
|
@ -477,7 +471,7 @@ func (d *FileDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
d.active = false
|
d.active = false
|
||||||
|
|
||||||
d.current += uint64(n)
|
d.current += uint64(n)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
|
|
||||||
// use a goro in case someone else wants to enable cancel/resume
|
// use a goro in case someone else wants to enable cancel/resume
|
||||||
} else {
|
} else {
|
||||||
|
@ -490,7 +484,7 @@ func (d *FileDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
d.current += uint64(n)
|
d.current += uint64(n)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
}
|
}
|
||||||
d.active = false
|
d.active = false
|
||||||
e <- err
|
e <- err
|
||||||
|
@ -499,7 +493,7 @@ func (d *FileDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
// ...and we spin until it's done
|
// ...and we spin until it's done
|
||||||
err = <-errch
|
err = <-errch
|
||||||
}
|
}
|
||||||
progressBar.Finish()
|
bar.Finish()
|
||||||
f.Close()
|
f.Close()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -513,7 +507,7 @@ type SMBDownloader struct {
|
||||||
current uint64
|
current uint64
|
||||||
total uint64
|
total uint64
|
||||||
|
|
||||||
progress ProgressBar
|
progress packer.ProgressBar
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *SMBDownloader) Progress() uint64 {
|
func (d *SMBDownloader) Progress() uint64 {
|
||||||
|
@ -587,9 +581,8 @@ func (d *SMBDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
d.total = uint64(fi.Size())
|
d.total = uint64(fi.Size())
|
||||||
|
|
||||||
bar := d.progress
|
bar := d.progress
|
||||||
bar.SetTotal64(int64(d.total))
|
|
||||||
progressBar := bar.Start()
|
bar.Start(d.current)
|
||||||
progressBar.Set64(int64(d.current))
|
|
||||||
|
|
||||||
// no bufferSize specified, so copy synchronously.
|
// no bufferSize specified, so copy synchronously.
|
||||||
if d.bufferSize == nil {
|
if d.bufferSize == nil {
|
||||||
|
@ -598,7 +591,7 @@ func (d *SMBDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
d.active = false
|
d.active = false
|
||||||
|
|
||||||
d.current += uint64(n)
|
d.current += uint64(n)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
|
|
||||||
// use a goro in case someone else wants to enable cancel/resume
|
// use a goro in case someone else wants to enable cancel/resume
|
||||||
} else {
|
} else {
|
||||||
|
@ -611,7 +604,7 @@ func (d *SMBDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
d.current += uint64(n)
|
d.current += uint64(n)
|
||||||
progressBar.Set64(int64(d.current))
|
bar.Set(d.current)
|
||||||
}
|
}
|
||||||
d.active = false
|
d.active = false
|
||||||
e <- err
|
e <- err
|
||||||
|
@ -620,7 +613,7 @@ func (d *SMBDownloader) Download(dst *os.File, src *url.URL) error {
|
||||||
// ...and as usual we spin until it's done
|
// ...and as usual we spin until it's done
|
||||||
err = <-errch
|
err = <-errch
|
||||||
}
|
}
|
||||||
progressBar.Finish()
|
bar.Finish()
|
||||||
f.Close()
|
f.Close()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,8 @@ import (
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/hashicorp/packer/packer"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestDownloadClientVerifyChecksum(t *testing.T) {
|
func TestDownloadClientVerifyChecksum(t *testing.T) {
|
||||||
|
@ -36,7 +38,7 @@ func TestDownloadClientVerifyChecksum(t *testing.T) {
|
||||||
Checksum: checksum,
|
Checksum: checksum,
|
||||||
}
|
}
|
||||||
|
|
||||||
d := NewDownloadClient(config, nil)
|
d := NewDownloadClient(config, new(packer.NoopProgressBar))
|
||||||
result, err := d.VerifyChecksum(tf.Name())
|
result, err := d.VerifyChecksum(tf.Name())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Verify err: %s", err)
|
t.Fatalf("Verify err: %s", err)
|
||||||
|
@ -59,7 +61,7 @@ func TestDownloadClient_basic(t *testing.T) {
|
||||||
Url: ts.URL + "/basic.txt",
|
Url: ts.URL + "/basic.txt",
|
||||||
TargetPath: tf.Name(),
|
TargetPath: tf.Name(),
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
path, err := client.Get()
|
path, err := client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -95,7 +97,7 @@ func TestDownloadClient_checksumBad(t *testing.T) {
|
||||||
Hash: HashForType("md5"),
|
Hash: HashForType("md5"),
|
||||||
Checksum: checksum,
|
Checksum: checksum,
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
if _, err := client.Get(); err == nil {
|
if _, err := client.Get(); err == nil {
|
||||||
t.Fatal("should error")
|
t.Fatal("should error")
|
||||||
|
@ -121,7 +123,7 @@ func TestDownloadClient_checksumGood(t *testing.T) {
|
||||||
Hash: HashForType("md5"),
|
Hash: HashForType("md5"),
|
||||||
Checksum: checksum,
|
Checksum: checksum,
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
path, err := client.Get()
|
path, err := client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -153,7 +155,7 @@ func TestDownloadClient_checksumNoDownload(t *testing.T) {
|
||||||
Hash: HashForType("md5"),
|
Hash: HashForType("md5"),
|
||||||
Checksum: checksum,
|
Checksum: checksum,
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
path, err := client.Get()
|
path, err := client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("err: %s", err)
|
t.Fatalf("err: %s", err)
|
||||||
|
@ -183,7 +185,7 @@ func TestDownloadClient_notFound(t *testing.T) {
|
||||||
client := NewDownloadClient(&DownloadConfig{
|
client := NewDownloadClient(&DownloadConfig{
|
||||||
Url: ts.URL + "/not-found.txt",
|
Url: ts.URL + "/not-found.txt",
|
||||||
TargetPath: tf.Name(),
|
TargetPath: tf.Name(),
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
if _, err := client.Get(); err == nil {
|
if _, err := client.Get(); err == nil {
|
||||||
t.Fatal("should error")
|
t.Fatal("should error")
|
||||||
|
@ -211,7 +213,7 @@ func TestDownloadClient_resume(t *testing.T) {
|
||||||
Url: ts.URL,
|
Url: ts.URL,
|
||||||
TargetPath: tf.Name(),
|
TargetPath: tf.Name(),
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}, nil)
|
}, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
path, err := client.Get()
|
path, err := client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -273,7 +275,7 @@ func TestDownloadClient_usesDefaultUserAgent(t *testing.T) {
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
client := NewDownloadClient(config, nil)
|
client := NewDownloadClient(config, new(packer.NoopProgressBar))
|
||||||
_, err = client.Get()
|
_, err = client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -306,7 +308,7 @@ func TestDownloadClient_setsUserAgent(t *testing.T) {
|
||||||
CopyFile: true,
|
CopyFile: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
client := NewDownloadClient(config, nil)
|
client := NewDownloadClient(config, new(packer.NoopProgressBar))
|
||||||
_, err = client.Get()
|
_, err = client.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -405,7 +407,7 @@ func TestDownloadFileUrl(t *testing.T) {
|
||||||
CopyFile: false,
|
CopyFile: false,
|
||||||
}
|
}
|
||||||
|
|
||||||
client := NewDownloadClient(config, nil)
|
client := NewDownloadClient(config, new(packer.NoopProgressBar))
|
||||||
|
|
||||||
// Verify that we fail to match the checksum
|
// Verify that we fail to match the checksum
|
||||||
_, err = client.Get()
|
_, err = client.Get()
|
||||||
|
@ -436,7 +438,7 @@ func SimulateFileUriDownload(t *testing.T, uri string) (string, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// go go go
|
// go go go
|
||||||
client := NewDownloadClient(config, nil)
|
client := NewDownloadClient(config, new(packer.NoopProgressBar))
|
||||||
path, err := client.Get()
|
path, err := client.Get()
|
||||||
|
|
||||||
// ignore any non-important checksum errors if it's not a unc path
|
// ignore any non-important checksum errors if it's not a unc path
|
||||||
|
|
|
@ -2,13 +2,14 @@ package common
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"reflect"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/cheggaaa/pb"
|
"github.com/cheggaaa/pb"
|
||||||
"github.com/hashicorp/packer/helper/multistep"
|
"github.com/hashicorp/packer/helper/multistep"
|
||||||
"github.com/hashicorp/packer/packer"
|
"github.com/hashicorp/packer/packer"
|
||||||
"github.com/hashicorp/packer/packer/rpc"
|
"github.com/hashicorp/packer/packer/rpc"
|
||||||
"log"
|
|
||||||
"reflect"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// This is the arrow from packer/ui.go -> TargetedUI.prefixLines
|
// This is the arrow from packer/ui.go -> TargetedUI.prefixLines
|
||||||
|
|
|
@ -64,7 +64,7 @@ func (s *StepDownload) Run(_ context.Context, state multistep.StateBag) multiste
|
||||||
ui.Say(fmt.Sprintf("Retrieving %s", s.Description))
|
ui.Say(fmt.Sprintf("Retrieving %s", s.Description))
|
||||||
|
|
||||||
// Get a progress bar from the ui so we can hand it off to the download client
|
// Get a progress bar from the ui so we can hand it off to the download client
|
||||||
bar := GetProgressBar(ui, GetPackerConfigFromStateBag(state))
|
bar := ui.ProgressBar()
|
||||||
|
|
||||||
// First try to use any already downloaded file
|
// First try to use any already downloaded file
|
||||||
// If it fails, proceed to regular download logic
|
// If it fails, proceed to regular download logic
|
||||||
|
@ -144,7 +144,8 @@ func (s *StepDownload) download(config *DownloadConfig, state multistep.StateBag
|
||||||
ui := state.Get("ui").(packer.Ui)
|
ui := state.Get("ui").(packer.Ui)
|
||||||
|
|
||||||
// Get a progress bar and hand it off to the download client
|
// Get a progress bar and hand it off to the download client
|
||||||
bar := GetProgressBar(ui, GetPackerConfigFromStateBag(state))
|
bar := ui.ProgressBar()
|
||||||
|
log.Printf("new progress bar: %#v, %t", bar, bar == nil)
|
||||||
|
|
||||||
// Create download client with config and progress bar
|
// Create download client with config and progress bar
|
||||||
download := NewDownloadClient(config, bar)
|
download := NewDownloadClient(config, bar)
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
package packer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/cheggaaa/pb"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ProgressBar allows to graphically display
|
||||||
|
// a self refreshing progress bar.
|
||||||
|
// No-op When in machine readable mode.
|
||||||
|
type ProgressBar interface {
|
||||||
|
Start(total uint64)
|
||||||
|
Set(current uint64)
|
||||||
|
Finish()
|
||||||
|
}
|
||||||
|
|
||||||
|
type BasicProgressBar struct {
|
||||||
|
*pb.ProgressBar
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bpb *BasicProgressBar) Start(total uint64) {
|
||||||
|
bpb.SetTotal64(int64(total))
|
||||||
|
bpb.ProgressBar.Start()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bpb *BasicProgressBar) Set(current uint64) {
|
||||||
|
bpb.ProgressBar.Set64(int64(current))
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ ProgressBar = new(BasicProgressBar)
|
||||||
|
|
||||||
|
// NoopProgressBar is a silent progress bar
|
||||||
|
type NoopProgressBar struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bpb *NoopProgressBar) Start(_ uint64) {}
|
||||||
|
func (bpb *NoopProgressBar) Set(_ uint64) {}
|
||||||
|
func (bpb *NoopProgressBar) Finish() {}
|
||||||
|
|
||||||
|
var _ ProgressBar = new(NoopProgressBar)
|
|
@ -114,7 +114,8 @@ func (s *Server) RegisterProvisioner(p packer.Provisioner) {
|
||||||
|
|
||||||
func (s *Server) RegisterUi(ui packer.Ui) {
|
func (s *Server) RegisterUi(ui packer.Ui) {
|
||||||
s.server.RegisterName(DefaultUiEndpoint, &UiServer{
|
s.server.RegisterName(DefaultUiEndpoint, &UiServer{
|
||||||
ui: ui,
|
ui: ui,
|
||||||
|
register: s.server.RegisterName,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@ package rpc
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"log"
|
"log"
|
||||||
|
"math/rand"
|
||||||
"net/rpc"
|
"net/rpc"
|
||||||
|
|
||||||
"github.com/hashicorp/packer/packer"
|
"github.com/hashicorp/packer/packer"
|
||||||
|
@ -14,10 +15,13 @@ type Ui struct {
|
||||||
endpoint string
|
endpoint string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _ packer.Ui = new(Ui)
|
||||||
|
|
||||||
// UiServer wraps a packer.Ui implementation and makes it exportable
|
// UiServer wraps a packer.Ui implementation and makes it exportable
|
||||||
// as part of a Golang RPC server.
|
// as part of a Golang RPC server.
|
||||||
type UiServer struct {
|
type UiServer struct {
|
||||||
ui packer.Ui
|
ui packer.Ui
|
||||||
|
register func(name string, rcvr interface{}) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// The arguments sent to Ui.Machine
|
// The arguments sent to Ui.Machine
|
||||||
|
@ -60,6 +64,38 @@ func (u *Ui) Say(message string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *Ui) ProgressBar() packer.ProgressBar {
|
||||||
|
var callMeMaybe string
|
||||||
|
if err := u.client.Call("Ui.ProgressBar", nil, &callMeMaybe); err != nil {
|
||||||
|
log.Printf("Error in Ui RPC call: %s", err)
|
||||||
|
return new(packer.NoopProgressBar)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &RemoteProgressBarClient{
|
||||||
|
id: callMeMaybe,
|
||||||
|
client: u.client,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type RemoteProgressBarClient struct {
|
||||||
|
id string
|
||||||
|
client *rpc.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ packer.ProgressBar = new(RemoteProgressBarClient)
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarClient) Start(total uint64) {
|
||||||
|
pb.client.Call(pb.id+".Start", total, new(interface{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarClient) Set(current uint64) {
|
||||||
|
pb.client.Call(pb.id+".Set", current, new(interface{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarClient) Finish() {
|
||||||
|
pb.client.Call(pb.id+".Finish", nil, new(interface{}))
|
||||||
|
}
|
||||||
|
|
||||||
func (u *UiServer) Ask(query string, reply *string) (err error) {
|
func (u *UiServer) Ask(query string, reply *string) (err error) {
|
||||||
*reply, err = u.ui.Ask(query)
|
*reply, err = u.ui.Ask(query)
|
||||||
return
|
return
|
||||||
|
@ -91,3 +127,47 @@ func (u *UiServer) Say(message *string, reply *interface{}) error {
|
||||||
*reply = nil
|
*reply = nil
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func RandStringBytes(n int) string {
|
||||||
|
const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
||||||
|
|
||||||
|
b := make([]byte, n)
|
||||||
|
for i := range b {
|
||||||
|
b[i] = letterBytes[rand.Intn(len(letterBytes))]
|
||||||
|
}
|
||||||
|
return string(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *UiServer) ProgressBar(_ *string, reply *interface{}) error {
|
||||||
|
bar := u.ui.ProgressBar()
|
||||||
|
|
||||||
|
callbackName := RandStringBytes(6)
|
||||||
|
|
||||||
|
log.Printf("registering progressbar %s", callbackName)
|
||||||
|
err := u.register(callbackName, &RemoteProgressBarServer{bar})
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("failed to register a new progress bar rpc server, %s", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*reply = callbackName
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type RemoteProgressBarServer struct {
|
||||||
|
pb packer.ProgressBar
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarServer) Finish(_ string, _ *interface{}) error {
|
||||||
|
pb.pb.Finish()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarServer) Start(total uint64, _ *interface{}) error {
|
||||||
|
pb.pb.Start(total)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pb *RemoteProgressBarServer) Set(current uint64, _ *interface{}) error {
|
||||||
|
pb.pb.Set(current)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
29
packer/ui.go
29
packer/ui.go
|
@ -15,6 +15,8 @@ import (
|
||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
"unicode"
|
"unicode"
|
||||||
|
|
||||||
|
"github.com/cheggaaa/pb"
|
||||||
)
|
)
|
||||||
|
|
||||||
type UiColor uint
|
type UiColor uint
|
||||||
|
@ -37,6 +39,7 @@ type Ui interface {
|
||||||
Message(string)
|
Message(string)
|
||||||
Error(string)
|
Error(string)
|
||||||
Machine(string, ...string)
|
Machine(string, ...string)
|
||||||
|
ProgressBar() ProgressBar
|
||||||
}
|
}
|
||||||
|
|
||||||
// ColoredUi is a UI that is colored using terminal colors.
|
// ColoredUi is a UI that is colored using terminal colors.
|
||||||
|
@ -46,6 +49,8 @@ type ColoredUi struct {
|
||||||
Ui Ui
|
Ui Ui
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _ Ui = new(ColoredUi)
|
||||||
|
|
||||||
// TargetedUI is a UI that wraps another UI implementation and modifies
|
// TargetedUI is a UI that wraps another UI implementation and modifies
|
||||||
// the output to indicate a specific target. Specifically, all Say output
|
// the output to indicate a specific target. Specifically, all Say output
|
||||||
// is prefixed with the target name. Message output is not prefixed but
|
// is prefixed with the target name. Message output is not prefixed but
|
||||||
|
@ -56,6 +61,8 @@ type TargetedUI struct {
|
||||||
Ui Ui
|
Ui Ui
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _ Ui = new(TargetedUI)
|
||||||
|
|
||||||
// The BasicUI is a UI that reads and writes from a standard Go reader
|
// The BasicUI is a UI that reads and writes from a standard Go reader
|
||||||
// and writer. It is safe to be called from multiple goroutines. Machine
|
// and writer. It is safe to be called from multiple goroutines. Machine
|
||||||
// readable output is simply logged for this UI.
|
// readable output is simply logged for this UI.
|
||||||
|
@ -68,12 +75,21 @@ type BasicUi struct {
|
||||||
scanner *bufio.Scanner
|
scanner *bufio.Scanner
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _ Ui = new(BasicUi)
|
||||||
|
|
||||||
|
func (bu *BasicUi) ProgressBar() ProgressBar {
|
||||||
|
log.Printf("hehey !")
|
||||||
|
return &BasicProgressBar{ProgressBar: pb.New(0)}
|
||||||
|
}
|
||||||
|
|
||||||
// MachineReadableUi is a UI that only outputs machine-readable output
|
// MachineReadableUi is a UI that only outputs machine-readable output
|
||||||
// to the given Writer.
|
// to the given Writer.
|
||||||
type MachineReadableUi struct {
|
type MachineReadableUi struct {
|
||||||
Writer io.Writer
|
Writer io.Writer
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _ Ui = new(MachineReadableUi)
|
||||||
|
|
||||||
func (u *ColoredUi) Ask(query string) (string, error) {
|
func (u *ColoredUi) Ask(query string) (string, error) {
|
||||||
return u.Ui.Ask(u.colorize(query, u.Color, true))
|
return u.Ui.Ask(u.colorize(query, u.Color, true))
|
||||||
}
|
}
|
||||||
|
@ -100,6 +116,10 @@ func (u *ColoredUi) Machine(t string, args ...string) {
|
||||||
u.Ui.Machine(t, args...)
|
u.Ui.Machine(t, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *ColoredUi) ProgressBar() ProgressBar {
|
||||||
|
return u.Ui.ProgressBar() //TODO(adrien): color me
|
||||||
|
}
|
||||||
|
|
||||||
func (u *ColoredUi) colorize(message string, color UiColor, bold bool) string {
|
func (u *ColoredUi) colorize(message string, color UiColor, bold bool) string {
|
||||||
if !u.supportsColors() {
|
if !u.supportsColors() {
|
||||||
return message
|
return message
|
||||||
|
@ -153,6 +173,10 @@ func (u *TargetedUI) Machine(t string, args ...string) {
|
||||||
u.Ui.Machine(fmt.Sprintf("%s,%s", u.Target, t), args...)
|
u.Ui.Machine(fmt.Sprintf("%s,%s", u.Target, t), args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *TargetedUI) ProgressBar() ProgressBar {
|
||||||
|
return u.Ui.ProgressBar()
|
||||||
|
}
|
||||||
|
|
||||||
func (u *TargetedUI) prefixLines(arrow bool, message string) string {
|
func (u *TargetedUI) prefixLines(arrow bool, message string) string {
|
||||||
arrowText := "==>"
|
arrowText := "==>"
|
||||||
if !arrow {
|
if !arrow {
|
||||||
|
@ -305,3 +329,8 @@ func (u *MachineReadableUi) Machine(category string, args ...string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *MachineReadableUi) ProgressBar() ProgressBar {
|
||||||
|
panic("MachineReadableUi")
|
||||||
|
return nil // no-op
|
||||||
|
}
|
||||||
|
|
|
@ -612,3 +612,8 @@ func (ui *Ui) Machine(t string, args ...string) {
|
||||||
ui.ui.Machine(t, args...)
|
ui.ui.Machine(t, args...)
|
||||||
<-ui.sem
|
<-ui.sem
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ui *Ui) ProgressBar() packer.ProgressBar {
|
||||||
|
panic("to implement")
|
||||||
|
return nil // TODO
|
||||||
|
}
|
||||||
|
|
|
@ -127,12 +127,12 @@ func (p *Provisioner) ProvisionDownload(ui packer.Ui, comm packer.Communicator)
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
|
|
||||||
// Get a default progress bar
|
// Get a default progress bar
|
||||||
pb := common.GetProgressBar(ui, &p.config.PackerConfig)
|
pb := packer.NoopProgressBar{}
|
||||||
bar := pb.Start()
|
pb.Start(0)
|
||||||
defer bar.Finish()
|
defer pb.Finish()
|
||||||
|
|
||||||
// Create MultiWriter for the current progress
|
// Create MultiWriter for the current progress
|
||||||
pf := io.MultiWriter(f, bar)
|
pf := io.MultiWriter(f)
|
||||||
|
|
||||||
// Download the file
|
// Download the file
|
||||||
if err = comm.Download(src, pf); err != nil {
|
if err = comm.Download(src, pf); err != nil {
|
||||||
|
@ -176,8 +176,8 @@ func (p *Provisioner) ProvisionUpload(ui packer.Ui, comm packer.Communicator) er
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get a default progress bar
|
// Get a default progress bar
|
||||||
pb := common.GetProgressBar(ui, &p.config.PackerConfig)
|
bar := ui.ProgressBar()
|
||||||
bar := pb.Start()
|
bar.Start(0)
|
||||||
defer bar.Finish()
|
defer bar.Finish()
|
||||||
|
|
||||||
// Create ProxyReader for the current progress
|
// Create ProxyReader for the current progress
|
||||||
|
|
Loading…
Reference in New Issue