Skip to content

Commit

Permalink
test: Add tests for edge cases
Browse files Browse the repository at this point in the history
Signed-off-by: Mike Sul <[email protected]>
  • Loading branch information
mike-sul committed Sep 12, 2024
1 parent 5a6df29 commit 347f29a
Show file tree
Hide file tree
Showing 5 changed files with 157 additions and 0 deletions.
4 changes: 4 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -13,5 +13,9 @@ jobs:
- uses: actions/checkout@v3
- name: build
run: docker compose --env-file=test/compose/.env.test -f test/compose/docker-compose.yml run composectl make
- name: prepare test env
run: docker compose --env-file=test/compose/.env.test -f test/compose/docker-compose.yml run composectl test/fixtures/images/build-images.sh
- name: test
run: docker compose --env-file=test/compose/.env.test -f test/compose/docker-compose.yml run composectl go test -v ./...
- name: teardown test env
run: docker compose --env-file=test/compose/.env.test -f test/compose/docker-compose.yml down
28 changes: 28 additions & 0 deletions test/fixtures/images/build-images.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
#!/bin/sh

DIR="$(dirname "$(realpath "$0")")"
REGISTRY_URL="registry:5000"
IMAGE_NAME="factory/runner-image"
IMAGE_TAG="v0.1"
IMAGE_URI="${REGISTRY_URL}/${IMAGE_NAME}:${IMAGE_TAG}"

# Function to check if the image exists in the registry
check_image() {
local response=$(curl -s -o /dev/null -w "%{http_code}" \
"https://${REGISTRY_URL}/v2/${IMAGE_NAME}/manifests/${IMAGE_TAG}")

if [[ "$response" == "200" || "$response" == "302" ]]; then
return 0
else
return 1
fi
}

# Using the function in an if statement with negation
if ! check_image; then
docker buildx build -t "${IMAGE_URI}" --push $DIR/runner
# Remove image from a local dockerd store, the tests need it to be in the registry
docker image rm ${IMAGE_URI}
else
echo "Image ${IMAGE_URI} exists in the registry."
fi
8 changes: 8 additions & 0 deletions test/fixtures/images/runner/Dockerfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
FROM golang:alpine AS builder
WORKDIR /app
COPY forever.go .
RUN CGO_ENABLED=0 go build -ldflags="-w -s" -o forever forever.go

FROM scratch
COPY --from=builder /app/forever /forever
CMD ["/forever"]
11 changes: 11 additions & 0 deletions test/fixtures/images/runner/forever.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package main

import (
"time"
)

func main() {
for {
time.Sleep(1 * time.Minute) // Sleep for 1 hour in each iteration
}
}
106 changes: 106 additions & 0 deletions test/integration/edge_cases_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,106 @@
package main

import (
"encoding/json"
"fmt"
composectl "github.com/foundriesio/composeapp/cmd/composectl/cmd"
"os"
"os/exec"
"path"
"testing"
)

func TestAppImageMultiUse(t *testing.T) {
appName := "app1"
appBaseUri := "registry:5000/factory/" + appName
appComposeDef := `
services:
srvs-01:
image: registry:5000/factory/runner-image:v0.1
ports:
- 8080:80
srvs-02:
image: registry:5000/factory/runner-image:v0.1
`
appDir := path.Join(t.TempDir(), appName)
digestFile := path.Join(t.TempDir(), "app.sha256")
err := os.MkdirAll(appDir, 0o755)
if err != nil {
t.Fatal(err)
}
err = os.WriteFile(path.Join(appDir, "docker-compose.yml"), []byte(appComposeDef), 0o640)
if err != nil {
t.Fatal(err)
}

runCmd := func(t *testing.T, args ...string) []byte {
c := exec.Command(composeExec, args...)
c.Dir = appDir
output, err := c.CombinedOutput()
if err != nil {
t.Errorf("failed to run `%s` command: %s\n", args[0], output)
}
return output
}

var appUri string

removeApp := func(t *testing.T) {
t.Run("remove app", func(t *testing.T) {
runCmd(t, "rm", appUri)
})
}
uninstallApp := func(t *testing.T) {
t.Run("uninstall app", func(t *testing.T) {
runCmd(t, "uninstall", appName)
})
}
stopApp := func(t *testing.T) {
t.Run("stop app", func(t *testing.T) {
runCmd(t, "stop", appName)
})
}

t.Run("publish app", func(t *testing.T) {
runCmd(t, "publish", "-d", digestFile, appBaseUri+":asdsa", "amd64")
if b, err := os.ReadFile(digestFile); err == nil {
appUri = appBaseUri + "@" + string(b)
} else {
t.Errorf("failed to read the published app digest: %s\n", err)
}
fmt.Printf("published app uri: %s\n", appUri)
})

t.Run("pull app", func(t *testing.T) {
runCmd(t, "pull", appUri, "-u", "90")
})
defer removeApp(t)

t.Run("install app", func(t *testing.T) {
runCmd(t, "install", appUri)
})
defer uninstallApp(t)

t.Run("run app", func(t *testing.T) {
runCmd(t, "run", appName)
})
defer stopApp(t)

t.Run("check if running", func(t *testing.T) {
output := runCmd(t, "ps", appUri, "--format", "json")
var psOutput map[string]composectl.App
if err := json.Unmarshal(output, &psOutput); err != nil {
t.Errorf("failed to unmarshal app ps output: %s\n", err)
}
if len(psOutput) != 1 {
t.Errorf("expected one element in ps output, got: %d\n", len(psOutput))
}
appStatus, ok := psOutput[appUri]
if !ok {
t.Errorf("no app URI in the ps output: %+v\n", psOutput)
}
if appStatus.State != "running" {
t.Errorf("app is not running, its state: %+s\n", appStatus.State)
}
})
}

0 comments on commit 347f29a

Please sign in to comment.