|
|
@ -20,6 +20,7 @@ import (
|
|
|
|
"encoding/hex"
|
|
|
|
"encoding/hex"
|
|
|
|
"os"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"path/filepath"
|
|
|
|
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
@ -485,3 +486,102 @@ func TestDownloadToCache(t *testing.T) {
|
|
|
|
c.Keyring = ""
|
|
|
|
c.Keyring = ""
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
func TestParallelDownloadTo(t *testing.T) {
|
|
|
|
|
|
|
|
// Set up a simple test server with a chart
|
|
|
|
|
|
|
|
srv := repotest.NewTempServer(t, repotest.WithChartSourceGlob("testdata/*.tgz"))
|
|
|
|
|
|
|
|
defer srv.Stop()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if err := srv.CreateIndex(); err != nil {
|
|
|
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
dest := t.TempDir()
|
|
|
|
|
|
|
|
cacheDir := t.TempDir()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
c := ChartDownloader{
|
|
|
|
|
|
|
|
Out: os.Stderr,
|
|
|
|
|
|
|
|
RepositoryConfig: repoConfig,
|
|
|
|
|
|
|
|
RepositoryCache: repoCache,
|
|
|
|
|
|
|
|
ContentCache: cacheDir,
|
|
|
|
|
|
|
|
Cache: &DiskCache{Root: cacheDir},
|
|
|
|
|
|
|
|
Getters: getter.All(&cli.EnvSettings{
|
|
|
|
|
|
|
|
RepositoryConfig: repoConfig,
|
|
|
|
|
|
|
|
RepositoryCache: repoCache,
|
|
|
|
|
|
|
|
ContentCache: cacheDir,
|
|
|
|
|
|
|
|
}),
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Use a direct URL to bypass repository lookup
|
|
|
|
|
|
|
|
chartURL := srv.URL() + "/local-subchart-0.1.0.tgz"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Number of parallel downloads to attempt
|
|
|
|
|
|
|
|
numDownloads := 10
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
errors := make([]error, numDownloads)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Launch multiple goroutines to download the same chart simultaneously
|
|
|
|
|
|
|
|
for i := 0; i < numDownloads; i++ {
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
|
|
|
|
go func(index int) {
|
|
|
|
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
_, _, err := c.DownloadTo(chartURL, "", dest)
|
|
|
|
|
|
|
|
errors[index] = err
|
|
|
|
|
|
|
|
}(i)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Check if any download failed
|
|
|
|
|
|
|
|
failedCount := 0
|
|
|
|
|
|
|
|
for i, err := range errors {
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
t.Logf("Download %d failed: %v", i, err)
|
|
|
|
|
|
|
|
failedCount++
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// With the file locking fix, all parallel downloads should succeed
|
|
|
|
|
|
|
|
if failedCount > 0 {
|
|
|
|
|
|
|
|
t.Errorf("Parallel downloads failed: %d out of %d downloads failed due to concurrent file access", failedCount, numDownloads)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Verify the file exists and is valid
|
|
|
|
|
|
|
|
expectedFile := filepath.Join(dest, "local-subchart-0.1.0.tgz")
|
|
|
|
|
|
|
|
info, err := os.Stat(expectedFile)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
t.Errorf("Expected file %s does not exist: %v", expectedFile, err)
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// Verify the file is not empty
|
|
|
|
|
|
|
|
if info.Size() == 0 {
|
|
|
|
|
|
|
|
t.Errorf("Downloaded file %s is empty (0 bytes)", expectedFile)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Verify the file has the expected size (should match the source file)
|
|
|
|
|
|
|
|
sourceFile := "testdata/local-subchart-0.1.0.tgz"
|
|
|
|
|
|
|
|
sourceInfo, err := os.Stat(sourceFile)
|
|
|
|
|
|
|
|
if err == nil && info.Size() != sourceInfo.Size() {
|
|
|
|
|
|
|
|
t.Errorf("Downloaded file size (%d bytes) doesn't match source file size (%d bytes)",
|
|
|
|
|
|
|
|
info.Size(), sourceInfo.Size())
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Verify it's a valid tar.gz file by checking the magic bytes
|
|
|
|
|
|
|
|
file, err := os.Open(expectedFile)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
// gzip magic bytes are 0x1f 0x8b
|
|
|
|
|
|
|
|
magic := make([]byte, 2)
|
|
|
|
|
|
|
|
if n, err := file.Read(magic); err == nil && n == 2 {
|
|
|
|
|
|
|
|
if magic[0] != 0x1f || magic[1] != 0x8b {
|
|
|
|
|
|
|
|
t.Errorf("Downloaded file is not a valid gzip file (magic bytes: %x)", magic)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Verify no lock file was left behind
|
|
|
|
|
|
|
|
lockFile := expectedFile + ".lock"
|
|
|
|
|
|
|
|
if _, err := os.Stat(lockFile); err == nil {
|
|
|
|
|
|
|
|
t.Errorf("Lock file %s was not cleaned up", lockFile)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|