From 289135415589f937afce23d66b913abba0aa1183 Mon Sep 17 00:00:00 2001 From: ClaytonNorthey92 Date: Sun, 16 Feb 2020 11:26:01 -0800 Subject: [PATCH] \#1500 - prototype for using testcontainers-go for dependencies --- .drone.yml | 21 ---- e2etests/all_test.go | 175 ++++++++++++++++++++++++++++++++ e2etests/run_athens.go | 63 ++++++++++++ go.mod | 5 +- go.sum | 38 +++++++ pkg/stash/with_redis_test.go | 26 ++++- pkg/storage/minio/minio_test.go | 62 ++++++++--- pkg/storage/mongo/mongo_test.go | 45 ++++++-- pkg/storage/s3/s3_test.go | 47 +++++++-- 9 files changed, 424 insertions(+), 58 deletions(-) create mode 100644 e2etests/all_test.go create mode 100644 e2etests/run_athens.go diff --git a/.drone.yml b/.drone.yml index 9beb1aaf30..15f678c168 100644 --- a/.drone.yml +++ b/.drone.yml @@ -43,9 +43,6 @@ steps: - GOPATH=~/emptygopath GOPROXY=http://localhost:3000 go build environment: GO111MODULE: on - ATHENS_MONGO_STORAGE_URL: mongodb://mongo:27017 - ATHENS_MINIO_ENDPOINT: minio:9000 - REDIS_TEST_ENDPOINT: redis:6379 GCS_SERVICE_ACCOUNT: from_secret: GCS_SERVICE_ACCOUNT GCS_PROJECT_ID: @@ -136,24 +133,6 @@ steps: # Here we can add any backend storage that can be tested. services: -- name: mongo - image: mongo - ports: - - 27017 -- name: minio - image: minio/minio:latest - command: - - server - - /data - ports: - - 9000 - environment: - MINIO_ACCESS_KEY: minio - MINIO_SECRET_KEY: minio123 -- name: redis - image: redis - ports: - - 6379 - name: athens-proxy image: gomods/athens:canary pull: always diff --git a/e2etests/all_test.go b/e2etests/all_test.go new file mode 100644 index 0000000000..35b8dd89f9 --- /dev/null +++ b/e2etests/all_test.go @@ -0,0 +1,175 @@ +// +build e2etests + +package e2etests + +import ( + "context" + "encoding/json" + "fmt" + "io/ioutil" + "net/http" + "os" + "os/exec" + "path/filepath" + "testing" + + "github.com/gobuffalo/envy" + "github.com/stretchr/testify/suite" + "github.com/testcontainers/testcontainers-go" +) + +type E2eSuite struct { + suite.Suite + goBinaryPath string + env []string + tmpDir string + sampleRepoPath string + stopAthens context.CancelFunc + athensContainer testcontainers.Container +} + +type catalogRes struct { + Modules []struct { + Module string `json:"module"` + Version string `json:"version"` + } `json:"modules"` +} + +func (m *E2eSuite) SetupSuite() { + var err error + m.tmpDir, err = ioutil.TempDir("/tmp", "gopath") + if err != nil { + m.Fail("Failed to make temp dir", err) + } + + m.sampleRepoPath, err = ioutil.TempDir("/tmp", "repopath") + if err != nil { + m.Fail("Failed to make temp dir for sample repo", err) + } + + m.goBinaryPath = envy.Get("GO_BINARY_PATH", "go") + + athensBin, err := buildAthens(m.goBinaryPath, m.tmpDir, m.env) + if err != nil { + m.Fail("Failed to build athens ", err) + } + // ignoring error as if no athens is running it fails. + + ctx := context.Background() + ctx, m.stopAthens = context.WithCancel(ctx) + + if m.athensContainer != nil { + err = stopAthens(ctx, m.athensContainer) + if err != nil { + m.Fail(err.Error()) + } + } + + c, err := runAthensAndWait(ctx, athensBin, m.getEnv()) + if err != nil { + m.Fail(err.Error()) + } + + m.athensContainer = c + + setupTestRepo(m.sampleRepoPath, "https://github.com/athens-artifacts/happy-path.git") +} + +func (m *E2eSuite) TearDownSuite() { + if m.athensContainer == nil { + err := stopAthens(context.Background(), m.athensContainer) + if err != nil { + m.Fail(err.Error()) + } + } + m.stopAthens() +} + +func TestE2E(t *testing.T) { + suite.Run(t, &E2eSuite{}) +} + +func (m *E2eSuite) SetupTest() { + chmodR(m.tmpDir, 0777) + err := cleanGoCache(m.getEnv()) + if err != nil { + m.Fail("Failed to clear go cache", err) + } +} + +func (m *E2eSuite) TestNoGoProxy() { + cmd := exec.Command("go", "run", ".") + cmd.Env = m.env + cmd.Dir = m.sampleRepoPath + + err := cmd.Run() + if err != nil { + m.Fail("go run failed on test repo", err) + } +} + +func (m *E2eSuite) TestGoProxy() { + + // TESTCONTAINERS - sub for docker container + ep, err := m.athensContainer.Endpoint(context.Background(), "http") + if err != nil { + m.Fail(err.Error()) + } + + cmd := exec.Command("go", "run", ".") + cmd.Env = m.getEnvGoProxy(ep) + cmd.Dir = m.sampleRepoPath + err = cmd.Run() + + if err != nil { + m.Fail("go run failed on test repo", err) + } + + resp, err := http.Get(fmt.Sprintf("%s/catalog", ep)) + if err != nil { + m.Fail("failed to read catalog", err) + } + + var catalog catalogRes + err = json.NewDecoder(resp.Body).Decode(&catalog) + if err != nil { + m.Fail("failed to decode catalog res", err) + } + m.Assert().Equal(len(catalog.Modules), 1) + m.Assert().Equal(catalog.Modules[0].Module, "github.com/athens-artifacts/no-tags") +} + +func (m *E2eSuite) TestWrongGoProxy() { + cmd := exec.Command("go", "run", ".") + cmd.Env = m.getEnvWrongGoProxy(m.tmpDir) + cmd.Dir = m.sampleRepoPath + err := cmd.Run() + m.Assert().NotNil(err, "Wrong proxy should fail") +} + +func (m *E2eSuite) getEnv() []string { + res := []string{ + fmt.Sprintf("GOPATH=%s", m.tmpDir), + "GO111MODULE=on", + fmt.Sprintf("PATH=%s", os.Getenv("PATH")), + fmt.Sprintf("GOCACHE=%s", filepath.Join(m.tmpDir, "cache")), + } + return res +} + +func (m *E2eSuite) getEnvGoProxy(goproxyOverride string) []string { + goProxy := "http://localhost:3000" + if goproxyOverride != "" { + goProxy = goproxyOverride + } + + res := m.getEnv() + res = append(res, fmt.Sprintf("GOPROXY=%s", goProxy)) + return res +} + +func (m *E2eSuite) getEnvWrongGoProxy(gopath string) []string { + res := m.getEnv() + res = append(res, "GOPROXY=http://localhost:3001") + return res +} diff --git a/e2etests/run_athens.go b/e2etests/run_athens.go new file mode 100644 index 0000000000..076a6caa77 --- /dev/null +++ b/e2etests/run_athens.go @@ -0,0 +1,63 @@ +// +build e2etests + +package e2etests + +import ( + "context" + "fmt" + "os/exec" + "path" + "path/filepath" + "time" + + testcontainers "github.com/testcontainers/testcontainers-go" + "github.com/testcontainers/testcontainers-go/wait" +) + +func buildAthens(goBin, destPath string, env []string) (string, error) { + target := path.Join(destPath, "athens-proxy") + binFolder, err := filepath.Abs("../cmd/proxy") + if err != nil { + return "", fmt.Errorf("Failed to get athens source path %v", err) + } + + cmd := exec.Command(goBin, "build", "-o", target, binFolder) + cmd.Env = env + output, err := cmd.CombinedOutput() + if err != nil { + return "", fmt.Errorf("Failed to build athens: %v - %s", err, string(output)) + } + return target, nil +} + +func stopAthens(ctx context.Context, athensContainer testcontainers.Container) error { + err := athensContainer.Terminate(ctx) + if err != nil { + return err + } + + return nil +} + +func runAthensAndWait(ctx context.Context, athensBin string, env []string) (testcontainers.Container, error) { + req := testcontainers.ContainerRequest{ + FromDockerfile: testcontainers.FromDockerfile{ + Dockerfile: "cmd/proxy/Dockerfile", + Context: "./..", + }, + ExposedPorts: []string{"3000/tcp"}, + WaitingFor: wait.ForLog("Starting application at port :3000").WithStartupTimeout(time.Minute * 2), + Cmd: []string{"athens-proxy", "-config_file=/config/config.toml"}, + AutoRemove: true, + } + + c, err := testcontainers.GenericContainer(ctx, testcontainers.GenericContainerRequest{ + ContainerRequest: req, + Started: true, + }) + if err != nil { + return nil, fmt.Errorf("Failed to run athens: %s", err) + } + + return c, nil +} diff --git a/go.mod b/go.mod index 5d00f562a4..b1307f8a24 100644 --- a/go.mod +++ b/go.mod @@ -18,7 +18,7 @@ require ( github.com/fatih/color v1.7.0 github.com/go-playground/locales v0.12.1 // indirect github.com/go-playground/universal-translator v0.16.0 // indirect - github.com/go-redis/redis v6.15.2+incompatible + github.com/go-redis/redis v6.15.7+incompatible github.com/gobuffalo/envy v1.6.7 github.com/gobuffalo/httptest v1.0.4 github.com/golang/snappy v0.0.1 // indirect @@ -34,12 +34,11 @@ require ( github.com/minio/minio-go/v6 v6.0.43 github.com/mitchellh/go-homedir v1.1.0 github.com/philhofer/fwd v1.0.0 // indirect - github.com/pkg/errors v0.8.1 // indirect github.com/sirupsen/logrus v1.4.2 github.com/spf13/afero v1.1.2 - github.com/spf13/pflag v1.0.3 // indirect github.com/stretchr/testify v1.3.0 github.com/technosophos/moniker v0.0.0-20180509230615-a5dbd03a2245 + github.com/testcontainers/testcontainers-go v0.2.0 github.com/tidwall/pretty v0.0.0-20180105212114-65a9db5fad51 // indirect github.com/tinylib/msgp v1.0.2 // indirect github.com/unrolled/secure v0.0.0-20181221173256-0d6b5bb13069 diff --git a/go.sum b/go.sum index 5685aacfce..3feb4c2781 100644 --- a/go.sum +++ b/go.sum @@ -11,6 +11,7 @@ github.com/Azure/azure-pipeline-go v0.2.1 h1:OLBdZJ3yvOn2MezlWvbrBMTEUQC72zAftRZ github.com/Azure/azure-pipeline-go v0.2.1/go.mod h1:UGSo8XybXnIGZ3epmeBw7Jdz+HiUVpqIlpz/HKHylF4= github.com/Azure/azure-storage-blob-go v0.7.0 h1:MuueVOYkufCxJw5YZzF842DY2MBsp+hLuh2apKY0mck= github.com/Azure/azure-storage-blob-go v0.7.0/go.mod h1:f9YQKtsG1nMisotuTPpO0tjNuEjKRYAcJU8/ydDI++4= +github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= github.com/Azure/go-autorest/autorest v0.9.0 h1:MRvx8gncNaXJqOoLmhNjUAKh33JJF8LyxPhomEtOsjs= github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI= github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0= @@ -33,6 +34,8 @@ github.com/DataDog/datadog-go v0.0.0-20180822151419-281ae9f2d895 h1:dmc/C8bpE5Vk github.com/DataDog/datadog-go v0.0.0-20180822151419-281ae9f2d895/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= github.com/DataDog/opencensus-go-exporter-datadog v0.0.0-20180917103902-e6c7f767dc57 h1:V1H8VVVxLALfaPLvAFCPoa0AN5nVPAqEu2UvH+QP3Vc= github.com/DataDog/opencensus-go-exporter-datadog v0.0.0-20180917103902-e6c7f767dc57/go.mod h1:gMGUEe16aZh0QN941HgDjwrdjU4iTthPoz2/AtDRADE= +github.com/Microsoft/go-winio v0.4.11/go.mod h1:VhR8bwka0BXejwEJY73c50VrPtXAaKcyvVC4A4RozmA= +github.com/Microsoft/hcsshim v0.8.6/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg= github.com/Shopify/sarama v1.19.0/go.mod h1:FVkBWblsNy7DGZRfXLU0O9RCGt5g3g3yEuWXgklEdEo= github.com/Shopify/toxiproxy v2.1.4+incompatible/go.mod h1:OXgGpZ6Cli1/URJOF1DMxUHB2q5Ap20/P/eIdh4G0pI= github.com/agext/levenshtein v1.2.1 h1:QmvMAjj2aEICytGiWzmxoE0x2KZvE0fvmqMOfy2tjT8= @@ -54,9 +57,13 @@ github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kB github.com/bsm/go-vlq v0.0.0-20150828105119-ec6e8d4f5f4e/go.mod h1:N+BjUcTjSxc2mtRGSCPsat1kze3CUtvJN3/jTXlp29k= github.com/bsm/redis-lock v8.0.0+incompatible h1:QgB0J2pNG8hUfndTIvpPh38F5XsUTTvO7x8Sls++9Mk= github.com/bsm/redis-lock v8.0.0+incompatible/go.mod h1:8dGkQ5GimBCahwF2R67tqGCJbyDZSp0gzO7wq3pDrik= +github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= +github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= github.com/codegangsta/negroni v1.0.0 h1:+aYywywx4bnKXWvoWtRfJ91vC59NbEhEY03sZjQhbVY= github.com/codegangsta/negroni v1.0.0/go.mod h1:v0y3T5G7Y1UlFfyxFn/QLRU4a2EuNau2iZY63YTKWo0= +github.com/containerd/continuity v0.0.0-20190426062206-aaeac12a7ffc h1:TP+534wVlf61smEIq1nwLLAjQVEK2EADoW3CX9AuT+8= +github.com/containerd/continuity v0.0.0-20190426062206-aaeac12a7ffc/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y= github.com/coreos/go-semver v0.2.0 h1:3Jm3tLmsgAYcjC+4Up7hJrFBPr+n7rAqYeSw/SZazuY= github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7 h1:u9SHYsPQNyt5tgDm3YN7+9dYrpK96E5wFilTFWIDZOM= @@ -68,6 +75,14 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/dgrijalva/jwt-go v3.2.0+incompatible h1:7qlOGliEKZXTDg6OTjfoBKDXWrumCAMpl/TFQ4/5kLM= github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= +github.com/docker/distribution v2.7.1-0.20190205005809-0d3efadf0154+incompatible h1:dvc1KSkIYTVjZgHf/CTC2diTYC8PzhaA5sFISRfNVrE= +github.com/docker/distribution v2.7.1-0.20190205005809-0d3efadf0154+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= +github.com/docker/docker v0.7.3-0.20190506211059-b20a14b54661 h1:ZuxGvIvF01nfc/G9RJ5Q7Va1zQE2WJyG18Zv3DqCEf4= +github.com/docker/docker v0.7.3-0.20190506211059-b20a14b54661/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= +github.com/docker/go-connections v0.4.0 h1:El9xVISelRB7BuFusrZozjnkIM5YnzCViNKohAFqRJQ= +github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec= +github.com/docker/go-units v0.3.3 h1:Xk8S3Xj5sLGlG5g67hJmYMmUgXv5N4PhkjJHHqrwnTk= +github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo= github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= @@ -90,6 +105,10 @@ github.com/go-playground/universal-translator v0.16.0 h1:X++omBR/4cE2MNg91AoC3rm github.com/go-playground/universal-translator v0.16.0/go.mod h1:1AnU7NaIRDWWzGEKwgtJRd2xk99HeFyHw3yid4rvQIY= github.com/go-redis/redis v6.15.2+incompatible h1:9SpNVG76gr6InJGxoZ6IuuxaCOQwDAhzyXg+Bs+0Sb4= github.com/go-redis/redis v6.15.2+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA= +github.com/go-redis/redis v6.15.7+incompatible h1:3skhDh95XQMpnqeqNftPkQD9jL9e5e36z/1SUm6dy1U= +github.com/go-redis/redis v6.15.7+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA= +github.com/go-sql-driver/mysql v1.5.0 h1:ozyZYNQW3x3HtqT1jira07DN2PArx2v7/mN66gGcHOs= +github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= github.com/go-stack/stack v1.8.0 h1:5SgMzNM5HxrEjV0ww2lTmX6E2Izsfxas4+YHWRs3Lsk= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= github.com/go-test/deep v1.0.1 h1:UQhStjbkDClarlmv0am7OXXO4/GaPdCGiUiMTvi28sg= @@ -204,14 +223,23 @@ github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7 h1:DpOJ2HYzCv8LZP15IdmG+YdwD2luVPHITV96TkirNBM= github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= +github.com/morikuni/aec v0.0.0-20170113033406-39771216ff4c/go.mod h1:BbKIizmSmc5MMPqRYbxO4ZU0S0+P200+tUnFx7PXmsc= github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.7.0 h1:WSHQ+IS43OoUrWtD1/bbclrwK8TTH5hzp+umCiuxHgs= github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.8.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/gomega v1.4.2/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= github.com/onsi/gomega v1.4.3 h1:RE1xgDvH7imwFD45h+u2SgIfERHlS2yNG4DObb5BSKU= github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= +github.com/onsi/gomega v1.5.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= +github.com/opencontainers/go-digest v1.0.0-rc1 h1:WzifXhOVOEOuFYOJAW6aQqW0TooG2iki3E3Ii+WN7gQ= +github.com/opencontainers/go-digest v1.0.0-rc1/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s= +github.com/opencontainers/image-spec v1.0.1 h1:JMemWkRwHx4Zj+fVxWoMCFm/8sYGGrUVojFA6h/TRcI= +github.com/opencontainers/image-spec v1.0.1/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= +github.com/opencontainers/runc v0.1.1 h1:GlxAyO6x8rfZYN9Tt0Kti5a/cP41iuiO2yYT0IJGY8Y= +github.com/opencontainers/runc v0.1.1/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U= github.com/openzipkin/zipkin-go v0.1.6/go.mod h1:QgAqvLzwWbR/WpD4A3cGpPtJrZXNIiJc5AZX7/PBEpw= github.com/philhofer/fwd v1.0.0 h1:UbZqGr5Y38ApvM/V/jEljVxwocdweyH+vmYvRPBnbqQ= github.com/philhofer/fwd v1.0.0/go.mod h1:gk3iGcWd9+svBvR0sR+KPcfE+RNWozjowpeBVG3ZVNU= @@ -220,6 +248,8 @@ github.com/pkg/errors v0.8.0 h1:WdK/asTD0HN+q6hsWO3/vpuAkAr+tw6aNJNDFFf0+qw= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= +github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/prometheus/client_golang v0.8.0/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= @@ -241,6 +271,8 @@ github.com/prometheus/procfs v0.0.0-20181204211112-1dc9a6cbc91a/go.mod h1:c3At6R github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1 h1:/K3IL0Z1quvmJ7X0A1AwNEK7CRkVK3YwfOU/QAL4WGg= github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= +github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww= +github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/sergi/go-diff v1.0.0 h1:Kpca3qRNrduNnOQeazBd0ysaKrUJiIuISHxogkT9RPQ= github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= github.com/sirupsen/logrus v1.0.5/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc= @@ -268,6 +300,8 @@ github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0 github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/technosophos/moniker v0.0.0-20180509230615-a5dbd03a2245 h1:DNVk+NIkGS0RbLkjQOLCJb/759yfCysThkMbl7EXxyY= github.com/technosophos/moniker v0.0.0-20180509230615-a5dbd03a2245/go.mod h1:O1c8HleITsZqzNZDjSNzirUGsMT0oGu9LhHKoJrqO+A= +github.com/testcontainers/testcontainers-go v0.2.0 h1:7PED4vniZMXNkPln4zg9U9ZDp4MmD/LuX0LGYTnByE0= +github.com/testcontainers/testcontainers-go v0.2.0/go.mod h1:5aBi+1PJmFixVc3b349A7NrhyTRYkMDpZEtT5MFxCs8= github.com/tidwall/pretty v0.0.0-20180105212114-65a9db5fad51 h1:BP2bjP495BBPaBcS5rmqviTfrOkN5rO5ceKAMRZCRFc= github.com/tidwall/pretty v0.0.0-20180105212114-65a9db5fad51/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= github.com/tinylib/msgp v1.0.2 h1:DfdQrzQa7Yh2es9SuLkixqxuXS2SxsdYn0KbdrOGWD8= @@ -347,6 +381,7 @@ golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5h golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181122145206-62eef0e2fa9b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181228144115-9a3f9b0469bb/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -358,6 +393,8 @@ golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2 h1:+DCIGbF/swA92ohVg0//6X2IVY3KZs6p9mix0ziNYJM= golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/tools v0.0.0-20180810170437-e96c4e24768d/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= @@ -403,6 +440,7 @@ gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWD gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gotest.tools v0.0.0-20181223230014-1083505acf35/go.mod h1:R//lfYlUuTOTfblYI3lGoAAAebUdzjvbmQsuB7Ykd90= honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= howett.net/plist v0.0.0-20181124034731-591f970eefbb/go.mod h1:vMygbs4qMhSZSc4lCUl2OEE+rDiIIJAIdR4m7MiMcm0= diff --git a/pkg/stash/with_redis_test.go b/pkg/stash/with_redis_test.go index 8ec50a0e59..8d403b8cdd 100644 --- a/pkg/stash/with_redis_test.go +++ b/pkg/stash/with_redis_test.go @@ -3,7 +3,6 @@ package stash import ( "context" "fmt" - "os" "strings" "sync" "testing" @@ -11,6 +10,8 @@ import ( "github.com/gomods/athens/pkg/storage" "github.com/gomods/athens/pkg/storage/mem" + testcontainers "github.com/testcontainers/testcontainers-go" + "github.com/testcontainers/testcontainers-go/wait" "golang.org/x/sync/errgroup" ) @@ -18,10 +19,27 @@ import ( // response. We can ensure that because only the first response does not return an error // and therefore all 5 responses should have no error. func TestWithRedisLock(t *testing.T) { - endpoint := os.Getenv("REDIS_TEST_ENDPOINT") - if len(endpoint) == 0 { - t.SkipNow() + req := testcontainers.ContainerRequest{ + Image: "redis:alpine", + ExposedPorts: []string{"6379/tcp"}, + WaitingFor: wait.ForLog("Ready to accept connections").WithStartupTimeout(time.Minute * 1), } + c, err := testcontainers.GenericContainer(context.Background(), testcontainers.GenericContainerRequest{ + ContainerRequest: req, + Started: true, + }) + + if err != nil { + t.Fatal(err) + } + + defer c.Terminate(context.Background()) + + endpoint, err := c.Endpoint(context.Background(), "") + if err != nil { + t.Fatal(err) + } + strg, err := mem.NewStorage() if err != nil { t.Fatal(err) diff --git a/pkg/storage/minio/minio_test.go b/pkg/storage/minio/minio_test.go index 85c87473f0..5994927d75 100644 --- a/pkg/storage/minio/minio_test.go +++ b/pkg/storage/minio/minio_test.go @@ -1,13 +1,47 @@ package minio import ( + "context" "os" "testing" + "time" "github.com/gomods/athens/pkg/config" "github.com/gomods/athens/pkg/storage/compliance" + testcontainers "github.com/testcontainers/testcontainers-go" + "github.com/testcontainers/testcontainers-go/wait" ) +var minioContainer testcontainers.Container + +func TestMain(m *testing.M) { + ctx := context.Background() + req := testcontainers.ContainerRequest{ + Image: "minio/minio:latest", + ExposedPorts: []string{"9000/tcp"}, + WaitingFor: wait.ForLog("Endpoint:").WithStartupTimeout(time.Minute * 1), + Cmd: []string{"server", "/data"}, + Env: map[string]string{ + "MINIO_ACCESS_KEY": "minio", + "MINIO_SECRET_KEY": "minio123", + }, + } + + c, err := testcontainers.GenericContainer(ctx, testcontainers.GenericContainerRequest{ + ContainerRequest: req, + Started: true, + }) + + if err != nil { + panic(err.Error()) + } + + minioContainer = c + defer c.Terminate(ctx) + os.Exit(m.Run()) + +} + func TestBackend(t *testing.T) { backend := getStorage(t) compliance.RunTests(t, backend, backend.clear) @@ -15,10 +49,6 @@ func TestBackend(t *testing.T) { // TestNewStorageExists tests the logic around MakeBucket and BucketExists func TestNewStorageExists(t *testing.T) { - url := os.Getenv("ATHENS_MINIO_ENDPOINT") - if url == "" { - t.SkipNow() - } tests := []struct { name string @@ -30,7 +60,7 @@ func TestNewStorageExists(t *testing.T) { for _, test := range tests { backend, err := NewStorage(&config.MinioConfig{ - Endpoint: url, + Endpoint: getURL(t), Key: "minio", Secret: "minio123", Bucket: test.name, @@ -51,10 +81,7 @@ func TestNewStorageExists(t *testing.T) { // To ensure both paths are tested, there is a strict path error using the // "_" and a non strict error using less than 3 characters func TestNewStorageError(t *testing.T) { - url := os.Getenv("ATHENS_MINIO_ENDPOINT") - if url == "" { - t.SkipNow() - } + url := getURL(t) // "_" is not allowed in a bucket name // bucket name must be bigger than 3 @@ -91,14 +118,23 @@ func (s *storageImpl) clear() error { return nil } +func getURL(t testing.TB) string { + ep, err := minioContainer.Endpoint(context.Background(), "") + if err != nil { + t.Fatal(err) + } + + return ep +} + func getStorage(t testing.TB) *storageImpl { - url := os.Getenv("ATHENS_MINIO_ENDPOINT") - if url == "" { - t.SkipNow() + ep, err := minioContainer.Endpoint(context.Background(), "") + if err != nil { + t.Fatal(err) } backend, err := NewStorage(&config.MinioConfig{ - Endpoint: url, + Endpoint: ep, Key: "minio", Secret: "minio123", Bucket: "gomods", diff --git a/pkg/storage/mongo/mongo_test.go b/pkg/storage/mongo/mongo_test.go index a7473d094f..42425941c7 100644 --- a/pkg/storage/mongo/mongo_test.go +++ b/pkg/storage/mongo/mongo_test.go @@ -6,15 +6,42 @@ import ( "io/ioutil" "os" "testing" + "time" "github.com/gomods/athens/pkg/config" "github.com/gomods/athens/pkg/errors" "github.com/gomods/athens/pkg/storage" "github.com/gomods/athens/pkg/storage/compliance" - "github.com/stretchr/testify/require" + testcontainers "github.com/testcontainers/testcontainers-go" + "github.com/testcontainers/testcontainers-go/wait" ) +var mongoContainer testcontainers.Container + +func TestMain(m *testing.M) { + ctx := context.Background() + req := testcontainers.ContainerRequest{ + Image: "mongo:3.6", + ExposedPorts: []string{"27017/tcp"}, + WaitingFor: wait.ForLog("waiting for connections on port 27017").WithStartupTimeout(time.Minute * 1), + } + + c, err := testcontainers.GenericContainer(ctx, testcontainers.GenericContainerRequest{ + ContainerRequest: req, + Started: true, + }) + + if err != nil { + os.Exit(1) + } + + mongoContainer = c + defer c.Terminate(ctx) + os.Exit(m.Run()) + +} + func TestBackend(t *testing.T) { backend := getStorage(t) compliance.RunTests(t, backend, backend.clear) @@ -31,13 +58,12 @@ func BenchmarkBackend(b *testing.B) { } func getStorage(tb testing.TB) *ModuleStore { - url := os.Getenv("ATHENS_MONGO_STORAGE_URL") - - if url == "" { - tb.SkipNow() + ep, err := mongoContainer.Endpoint(context.Background(), "mongodb") + if err != nil { + tb.Fatal(err) } - backend, err := NewStorage(&config.MongoConfig{URL: url}, config.GetTimeoutDuration(300)) + backend, err := NewStorage(&config.MongoConfig{URL: ep}, config.GetTimeoutDuration(300)) require.NoError(tb, err) return backend @@ -98,10 +124,9 @@ func TestQueryKindNotFoundErrorCases(t *testing.T) { } } func TestNewStorageWithDefaultOverrides(t *testing.T) { - url := os.Getenv("ATHENS_MONGO_STORAGE_URL") - - if url == "" { - t.SkipNow() + url, err := mongoContainer.Endpoint(context.Background(), "mongodb") + if err != nil { + t.Fatal(err) } testCases := []struct { diff --git a/pkg/storage/s3/s3_test.go b/pkg/storage/s3/s3_test.go index f24b40e825..1388f54b1d 100644 --- a/pkg/storage/s3/s3_test.go +++ b/pkg/storage/s3/s3_test.go @@ -4,14 +4,47 @@ import ( "context" "os" "testing" + "time" "github.com/aws/aws-sdk-go/aws" "github.com/aws/aws-sdk-go/aws/awserr" "github.com/aws/aws-sdk-go/service/s3" "github.com/gomods/athens/pkg/config" "github.com/gomods/athens/pkg/storage/compliance" + testcontainers "github.com/testcontainers/testcontainers-go" + "github.com/testcontainers/testcontainers-go/wait" ) +var minioContainer testcontainers.Container + +func TestMain(m *testing.M) { + ctx := context.Background() + req := testcontainers.ContainerRequest{ + Image: "minio/minio:latest", + ExposedPorts: []string{"9000/tcp"}, + WaitingFor: wait.ForLog("Endpoint:").WithStartupTimeout(time.Minute * 1), + Cmd: []string{"server", "/data"}, + Env: map[string]string{ + "MINIO_ACCESS_KEY": "minio", + "MINIO_SECRET_KEY": "minio123", + }, + } + + c, err := testcontainers.GenericContainer(ctx, testcontainers.GenericContainerRequest{ + ContainerRequest: req, + Started: true, + }) + + if err != nil { + panic(err.Error()) + } + + minioContainer = c + defer c.Terminate(ctx) + os.Exit(m.Run()) + +} + func TestBackend(t *testing.T) { backend := getStorage(t) compliance.RunTests(t, backend, backend.clear) @@ -69,9 +102,9 @@ func (s *Storage) createBucket() error { } func getStorage(t testing.TB) *Storage { - url := os.Getenv("ATHENS_MINIO_ENDPOINT") - if url == "" { - t.SkipNow() + url, err := minioContainer.Endpoint(context.Background(), "") + if err != nil { + t.Fatal(err) } options := func(conf *aws.Config) { @@ -80,10 +113,10 @@ func getStorage(t testing.TB) *Storage { } backend, err := New( &config.S3Config{ - Key: "minio", - Secret: "minio123", - Bucket: "gomodsaws", - Region: "us-west-1", + Key: "minio", + Secret: "minio123", + Bucket: "gomodsaws", + Region: "us-west-1", ForcePathStyle: true, }, config.GetTimeoutDuration(300),