package pkg_test import ( "crypto/sha512" "net/http" "reflect" "testing" "testing/fstest" "hakurei.app/container/check" "hakurei.app/internal/pkg" ) func TestHTTPGet(t *testing.T) { t.Parallel() const testdata = "\x7f\xe1\x69\xa2\xdd\x63\x96\x26\x83\x79\x61\x8b\xf0\x3f\xd5\x16\x9a\x39\x3a\xdb\xcf\xb1\xbc\x8d\x33\xff\x75\xee\x62\x56\xa9\xf0\x27\xac\x13\x94\x69" testdataChecksum := func() pkg.Checksum { h := sha512.New384() h.Write([]byte(testdata)) return (pkg.Checksum)(h.Sum(nil)) }() var transport http.Transport client := http.Client{Transport: &transport} transport.RegisterProtocol("file", http.NewFileTransportFS(fstest.MapFS{ "testdata": {Data: []byte(testdata), Mode: 0400}, })) checkWithCache(t, []cacheTestCase{ {"direct", nil, func(t *testing.T, base *check.Absolute, c *pkg.Cache) { f := pkg.NewHTTPGet( t.Context(), &client, "file:///testdata", testdataChecksum, ) wantIdent := pkg.KindHTTPGet.Ident([]byte("file:///testdata")) if got, err := f.Data(); err != nil { t.Fatalf("Data: error = %v", err) } else if string(got) != testdata { t.Fatalf("Data: %x, want %x", got, testdata) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdent)) } // check direct validation f = pkg.NewHTTPGet( t.Context(), &client, "file:///testdata", pkg.Checksum{}, ) wantErrMismatch := &pkg.ChecksumMismatchError{ Got: testdataChecksum, } if _, err := f.Data(); !reflect.DeepEqual(err, wantErrMismatch) { t.Fatalf("Data: error = %#v, want %#v", err, wantErrMismatch) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdent)) } // check direct response error f = pkg.NewHTTPGet( t.Context(), &client, "file:///nonexistent", pkg.Checksum{}, ) wantIdentNonexistent := pkg.KindHTTPGet.Ident([]byte("file:///nonexistent")) wantErrNotFound := pkg.ResponseStatusError(http.StatusNotFound) if _, err := f.Data(); !reflect.DeepEqual(err, wantErrNotFound) { t.Fatalf("Data: error = %#v, want %#v", err, wantErrNotFound) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdentNonexistent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdentNonexistent)) } }, pkg.MustDecode("E4vEZKhCcL2gPZ2Tt59FS3lDng-d_2SKa2i5G_RbDfwGn6EemptFaGLPUDiOa94C")}, {"cure", nil, func(t *testing.T, base *check.Absolute, c *pkg.Cache) { f := pkg.NewHTTPGet( t.Context(), &client, "file:///testdata", testdataChecksum, ) wantIdent := pkg.KindHTTPGet.Ident([]byte("file:///testdata")) wantPathname := base.Append( "identifier", pkg.Encode(wantIdent), ) if pathname, checksum, err := c.Cure(f); err != nil { t.Fatalf("Cure: error = %v", err) } else if !pathname.Is(wantPathname) { t.Fatalf("Cure: %q, want %q", pathname, wantPathname) } else if checksum != testdataChecksum { t.Fatalf("Cure: %x, want %x", checksum, testdataChecksum) } if got, err := f.Data(); err != nil { t.Fatalf("Data: error = %v", err) } else if string(got) != testdata { t.Fatalf("Data: %x, want %x", got, testdata) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdent)) } // check load from cache f = pkg.NewHTTPGet( t.Context(), &client, "file:///testdata", testdataChecksum, ) if got, err := f.Data(); err != nil { t.Fatalf("Data: error = %v", err) } else if string(got) != testdata { t.Fatalf("Data: %x, want %x", got, testdata) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdent)) } // check error passthrough f = pkg.NewHTTPGet( t.Context(), &client, "file:///nonexistent", pkg.Checksum{}, ) wantIdentNonexistent := pkg.KindHTTPGet.Ident([]byte("file:///nonexistent")) wantErrNotFound := pkg.ResponseStatusError(http.StatusNotFound) if _, _, err := c.Cure(f); !reflect.DeepEqual(err, wantErrNotFound) { t.Fatalf("Pathname: error = %#v, want %#v", err, wantErrNotFound) } else if gotIdent := pkg.Ident(f); gotIdent != wantIdentNonexistent { t.Fatalf("Ident: %s, want %s", pkg.Encode(gotIdent), pkg.Encode(wantIdentNonexistent)) } }, pkg.MustDecode("bqtn69RkV5E7V7GhhgCFjcvbxmaqrO8DywamM4Tyjf10F6EJBHjXiIa_tFRtF4iN")}, }) }