236 lines
5.8 KiB
Go
236 lines
5.8 KiB
Go
package cache
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func newTestServer(t *testing.T, handler http.HandlerFunc) *httptest.Server {
|
|
t.Helper()
|
|
svr := httptest.NewServer(handler)
|
|
t.Cleanup(func() { svr.Close() })
|
|
return svr
|
|
}
|
|
|
|
func newTestCache(t *testing.T, mirrorURLs []string) *Cache {
|
|
t.Helper()
|
|
mirroredRepos := []string{"core", "extra"}
|
|
c, err := NewCache(t.TempDir(), mirrorURLs, mirroredRepos)
|
|
if err != nil {
|
|
t.Fatalf("failed to create cache: %v", err)
|
|
}
|
|
c.client.Timeout = 500 * time.Millisecond
|
|
return c
|
|
}
|
|
|
|
func TestCacheHit(t *testing.T) {
|
|
const expected = "This is fake file contents"
|
|
|
|
c := newTestCache(t, []string{"http://example.com/"})
|
|
tmpFileName := "fakeFile"
|
|
err := c.cr.WriteFile(tmpFileName, []byte(expected), 0644)
|
|
if err != nil {
|
|
t.Fatalf("failed to create tempfile: %v", err)
|
|
}
|
|
|
|
cachedFile, err := c.Fetch(tmpFileName)
|
|
if err != nil {
|
|
t.Fatalf("failed to fetch file: %v", err)
|
|
}
|
|
//nolint:errcheck //ephemeral no need to check
|
|
defer cachedFile.Reader.Close()
|
|
|
|
if cachedFile.Filename != tmpFileName {
|
|
t.Errorf("expected filename %s got %s", tmpFileName, cachedFile.Filename)
|
|
}
|
|
|
|
if int64(len(expected)) != cachedFile.Size {
|
|
t.Errorf("expected %d got %d", len(expected), cachedFile.Size)
|
|
}
|
|
|
|
data, err := io.ReadAll(cachedFile.Reader)
|
|
if err != nil {
|
|
t.Fatalf("error reading file back: %v", err)
|
|
}
|
|
//nolint:errcheck //ephemeral no need to check
|
|
defer cachedFile.Reader.Close()
|
|
|
|
if !bytes.Equal(data, []byte(expected)) {
|
|
t.Errorf("expected file to contain %s got %s", expected, data)
|
|
}
|
|
}
|
|
|
|
func TestCacheMissExists(t *testing.T) {
|
|
const expected = "This is fake file contents"
|
|
|
|
svr := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
//nolint:errcheck //ephemeral no need to check
|
|
fmt.Fprint(w, expected)
|
|
}))
|
|
|
|
c := newTestCache(t, []string{svr.URL + "/"})
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
if err != nil {
|
|
t.Fatalf("Fetch failed %v", err)
|
|
}
|
|
|
|
data, err := c.cr.ReadFile("fakefile")
|
|
if err != nil {
|
|
t.Fatalf("Error reading file back: %v", err)
|
|
}
|
|
if !bytes.Equal(data, []byte(expected)) {
|
|
t.Errorf("expected file to contain %s got %s", expected, data)
|
|
}
|
|
}
|
|
|
|
func TestFetchNotFound(t *testing.T) {
|
|
svr := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
}))
|
|
|
|
c := newTestCache(t, []string{svr.URL + "/"})
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
var upstreamErr *UpstreamError
|
|
if !errors.As(err, &upstreamErr) {
|
|
t.Fatalf("expected UpstreamError got %v", err)
|
|
}
|
|
if upstreamErr.StatusCode != http.StatusNotFound {
|
|
t.Errorf("expected 404 got %d", upstreamErr.StatusCode)
|
|
}
|
|
}
|
|
|
|
func TestFetchSrvError(t *testing.T) {
|
|
svr := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
}))
|
|
|
|
c := newTestCache(t, []string{svr.URL + "/"})
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
var upstreamErr *UpstreamError
|
|
if !errors.As(err, &upstreamErr) {
|
|
t.Fatalf("expected UpstreamError fot %v", err)
|
|
}
|
|
if upstreamErr.StatusCode != http.StatusInternalServerError {
|
|
t.Errorf("expected 500 got %d", upstreamErr.StatusCode)
|
|
}
|
|
}
|
|
|
|
func TestFetchSrvDead(t *testing.T) {
|
|
svr := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
select {
|
|
case <-r.Context().Done():
|
|
return
|
|
case <-time.After(60 * time.Second):
|
|
//nolint:errcheck //ephemeral no need to check
|
|
fmt.Fprint(w, "too late")
|
|
}
|
|
}))
|
|
defer svr.Close()
|
|
|
|
c := newTestCache(t, []string{svr.URL + "/"})
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
if err == nil {
|
|
t.Fatal("expected err got nil")
|
|
}
|
|
|
|
if _, ok := errors.AsType[*UpstreamError](err); ok {
|
|
t.Error("expected network error not UpstreamError")
|
|
}
|
|
}
|
|
|
|
func TestFetchRetryExists(t *testing.T) {
|
|
const expected = "This is fake file contents"
|
|
|
|
svr1 := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
}))
|
|
|
|
svr2 := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
//nolint:errcheck //ephemeral no need to check
|
|
fmt.Fprint(w, expected)
|
|
}))
|
|
fakeURLs := []string{
|
|
svr1.URL + "/",
|
|
svr2.URL + "/",
|
|
}
|
|
|
|
c := newTestCache(t, fakeURLs)
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
if err != nil {
|
|
t.Fatalf("fetch failed: %v", err)
|
|
}
|
|
|
|
data, err := c.cr.ReadFile("fakefile")
|
|
if err != nil {
|
|
t.Fatalf("error reading file back: %v", err)
|
|
}
|
|
if !bytes.Equal(data, []byte(expected)) {
|
|
t.Errorf("expected file to contain %s got %s", expected, data)
|
|
}
|
|
}
|
|
|
|
func TestFetchRetryNonExist(t *testing.T) {
|
|
|
|
svr1 := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
}))
|
|
|
|
svr2 := newTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
}))
|
|
|
|
fakeURLs := []string{
|
|
svr1.URL + "/",
|
|
svr2.URL + "/",
|
|
}
|
|
|
|
c := newTestCache(t, fakeURLs)
|
|
|
|
_, err := c.Fetch("fakefile")
|
|
var upstreamErr *UpstreamError
|
|
if !errors.As(err, &upstreamErr) {
|
|
t.Errorf("expected UpstreamError got %v", err)
|
|
}
|
|
if upstreamErr.StatusCode != http.StatusNotFound {
|
|
t.Errorf("expected 404 got %d", upstreamErr.StatusCode)
|
|
}
|
|
}
|
|
|
|
func TestCreateSymlinks(t *testing.T) {
|
|
repos := []string{"core", "extra"}
|
|
tmp := t.TempDir()
|
|
cr, err := os.OpenRoot(tmp)
|
|
if err != nil {
|
|
t.Fatalf("unable to create tmp dir: %v", err)
|
|
}
|
|
|
|
if err := checkSymLinks(cr, repos); err != nil {
|
|
t.Fatalf("error creating links: %v", err)
|
|
}
|
|
|
|
for _, repo := range repos {
|
|
lnfile := filepath.Join(repo, "os/x86_64", repo+".db")
|
|
expected := repo + ".db.tar.gz"
|
|
lnval, err := cr.Readlink(lnfile)
|
|
if err != nil {
|
|
t.Errorf("%s has no link: %v", repo, err)
|
|
}
|
|
if lnval != expected {
|
|
t.Errorf("expected %s got %s", expected, lnval)
|
|
}
|
|
}
|
|
}
|