upload_test.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. package xfer
  2. import (
  3. "errors"
  4. "sync/atomic"
  5. "testing"
  6. "time"
  7. "github.com/docker/distribution"
  8. "github.com/docker/docker/layer"
  9. "github.com/docker/docker/pkg/progress"
  10. "golang.org/x/net/context"
  11. )
  12. const maxUploadConcurrency = 3
  13. type mockUploadDescriptor struct {
  14. currentUploads *int32
  15. diffID layer.DiffID
  16. simulateRetries int
  17. }
  18. // Key returns the key used to deduplicate downloads.
  19. func (u *mockUploadDescriptor) Key() string {
  20. return u.diffID.String()
  21. }
  22. // ID returns the ID for display purposes.
  23. func (u *mockUploadDescriptor) ID() string {
  24. return u.diffID.String()
  25. }
  26. // DiffID should return the DiffID for this layer.
  27. func (u *mockUploadDescriptor) DiffID() layer.DiffID {
  28. return u.diffID
  29. }
  30. // SetRemoteDescriptor is not used in the mock.
  31. func (u *mockUploadDescriptor) SetRemoteDescriptor(remoteDescriptor distribution.Descriptor) {
  32. }
  33. // Upload is called to perform the upload.
  34. func (u *mockUploadDescriptor) Upload(ctx context.Context, progressOutput progress.Output) (distribution.Descriptor, error) {
  35. if u.currentUploads != nil {
  36. defer atomic.AddInt32(u.currentUploads, -1)
  37. if atomic.AddInt32(u.currentUploads, 1) > maxUploadConcurrency {
  38. return distribution.Descriptor{}, errors.New("concurrency limit exceeded")
  39. }
  40. }
  41. // Sleep a bit to simulate a time-consuming upload.
  42. for i := int64(0); i <= 10; i++ {
  43. select {
  44. case <-ctx.Done():
  45. return distribution.Descriptor{}, ctx.Err()
  46. case <-time.After(10 * time.Millisecond):
  47. progressOutput.WriteProgress(progress.Progress{ID: u.ID(), Current: i, Total: 10})
  48. }
  49. }
  50. if u.simulateRetries != 0 {
  51. u.simulateRetries--
  52. return distribution.Descriptor{}, errors.New("simulating retry")
  53. }
  54. return distribution.Descriptor{}, nil
  55. }
  56. func uploadDescriptors(currentUploads *int32) []UploadDescriptor {
  57. return []UploadDescriptor{
  58. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:cbbf2f9a99b47fc460d422812b6a5adff7dfee951d8fa2e4a98caa0382cfbdbf"), 0},
  59. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:1515325234325236634634608943609283523908626098235490238423902343"), 0},
  60. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:6929356290463485374960346430698374523437683470934634534953453453"), 0},
  61. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:cbbf2f9a99b47fc460d422812b6a5adff7dfee951d8fa2e4a98caa0382cfbdbf"), 0},
  62. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:8159352387436803946235346346368745389534789534897538734598734987"), 1},
  63. &mockUploadDescriptor{currentUploads, layer.DiffID("sha256:4637863963478346897346987346987346789346789364879364897364987346"), 0},
  64. }
  65. }
  66. func TestSuccessfulUpload(t *testing.T) {
  67. lum := NewLayerUploadManager(maxUploadConcurrency, func(m *LayerUploadManager) { m.waitDuration = time.Millisecond })
  68. progressChan := make(chan progress.Progress)
  69. progressDone := make(chan struct{})
  70. receivedProgress := make(map[string]int64)
  71. go func() {
  72. for p := range progressChan {
  73. receivedProgress[p.ID] = p.Current
  74. }
  75. close(progressDone)
  76. }()
  77. var currentUploads int32
  78. descriptors := uploadDescriptors(&currentUploads)
  79. err := lum.Upload(context.Background(), descriptors, progress.ChanOutput(progressChan))
  80. if err != nil {
  81. t.Fatalf("upload error: %v", err)
  82. }
  83. close(progressChan)
  84. <-progressDone
  85. }
  86. func TestCancelledUpload(t *testing.T) {
  87. lum := NewLayerUploadManager(maxUploadConcurrency, func(m *LayerUploadManager) { m.waitDuration = time.Millisecond })
  88. progressChan := make(chan progress.Progress)
  89. progressDone := make(chan struct{})
  90. go func() {
  91. for range progressChan {
  92. }
  93. close(progressDone)
  94. }()
  95. ctx, cancel := context.WithCancel(context.Background())
  96. go func() {
  97. <-time.After(time.Millisecond)
  98. cancel()
  99. }()
  100. descriptors := uploadDescriptors(nil)
  101. err := lum.Upload(ctx, descriptors, progress.ChanOutput(progressChan))
  102. if err != context.Canceled {
  103. t.Fatal("expected upload to be cancelled")
  104. }
  105. close(progressChan)
  106. <-progressDone
  107. }