Skip to content

Commit

Permalink
build pipeline changes and more tests
Browse files Browse the repository at this point in the history
  • Loading branch information
patilpankaj212 committed Feb 18, 2021
1 parent 456cb31 commit 2434d9e
Show file tree
Hide file tree
Showing 10 changed files with 117 additions and 38 deletions.
3 changes: 3 additions & 0 deletions .github/workflows/gobuild.yml
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,9 @@ jobs:
- name: Run unit tests
run: make unit-tests

- name: Run e2e tests
run: make e2e-tests

- name: Upload coverage to Codecov
uses: codecov/codecov-action@v1

Expand Down
2 changes: 1 addition & 1 deletion Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ cicd: validate build test docker-build


# run all unit and integration tests
test: unit-tests
test: unit-tests e2e-tests


# run all validation tests
Expand Down
2 changes: 1 addition & 1 deletion scripts/generate-coverage.sh
Original file line number Diff line number Diff line change
Expand Up @@ -4,5 +4,5 @@ set -o errexit
set -o nounset
set -o pipefail

go test -v -coverpkg=./... -coverprofile=coverage.out ./...
go test -v -coverpkg=./pkg/... -coverprofile=coverage.out ./pkg/...
go tool cover -func coverage.out
2 changes: 2 additions & 0 deletions test/e2e/init/config/invalid_path.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
[policy]
path = "invalid/path"
3 changes: 3 additions & 0 deletions test/e2e/init/config/valid_config.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
[policy]
repo_url = "https://github.com/accurics/KaiMonkey.git"
branch = "master"
2 changes: 1 addition & 1 deletion test/e2e/init/golden/init_help.txt
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,4 @@ Global Flags:
-c, --config-path string config file path
-l, --log-level string log level (debug, info, warn, error, panic, fatal) (default "info")
-x, --log-type string log output type (console, json) (default "console")
-o, --output string output type (human, json, yaml, xml) (default "human")
-o, --output string output type (human, json, yaml, xml, junit-xml) (default "human")
3 changes: 3 additions & 0 deletions test/e2e/init/ignore.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
package init

// This file is added to ignore 'no non-test Go files' issue
107 changes: 81 additions & 26 deletions test/e2e/init/init_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import (
"io"
"os"
"path/filepath"
"time"

"github.com/accurics/terrascan/test/helper"
. "github.com/onsi/ginkgo"
Expand All @@ -18,10 +19,12 @@ const (
)

var (
initCommand string = "init"
defaultPolicyRepoPath string = os.Getenv("HOME") + "/.terrascan"
terrascanGitURL string = "https://github.com/accurics/terrascan.git"
terrascanDefaultBranch string = "master"
terrascanConfigEnvName string = "TERRASCAN_CONFIG"
kaiMoneyGitURL string = "https://github.com/accurics/KaiMonkey.git"
)

var _ = Describe("Init", func() {
Expand All @@ -46,10 +49,9 @@ var _ = Describe("Init", func() {
})

Describe("terrascan init is run", func() {
When("terrascan init is run without any flags", func() {
When("without any flags", func() {
It("should download policies and exit with status code 0", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init")
Eventually(session, initCommandTimeout).Should(gexec.Exit(0))
session = runInitCommand(terrascanBinaryPath, outWriter, errWriter, 0)
Expect(outWriter).Should(gbytes.Say(""))
})

Expand All @@ -61,19 +63,10 @@ var _ = Describe("Init", func() {
var repo *git.Repository
var err error
It("should be a valid git repo", func() {
repo, err = git.PlainOpen(defaultPolicyRepoPath)
Expect(err).NotTo(HaveOccurred())
Expect(repo).NotTo(BeNil())
repo = openGitRepo(defaultPolicyRepoPath)
})
It("should be terrascan git repo", func() {
remote, err := repo.Remote("origin")
Expect(err).NotTo(HaveOccurred())
Expect(remote).NotTo(BeNil())
remoteConfig := remote.Config()
Expect(remoteConfig).NotTo(BeNil())
err = remoteConfig.Validate()
Expect(err).NotTo(HaveOccurred())
Expect(remoteConfig.URLs[0]).To(BeEquivalentTo(terrascanGitURL))
validateGitRepo(repo, terrascanGitURL)
})
It("master branch should be present", func() {
_, err = repo.Branch(terrascanDefaultBranch)
Expand All @@ -84,14 +77,14 @@ var _ = Describe("Init", func() {

When("terrascan init is run with -h flag", func() {
It("should print help", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init", "-h")
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, initCommand, "-h")
goldenFileAbsPath, err := filepath.Abs("golden/init_help.txt")
Expect(err).NotTo(HaveOccurred())
helper.CompareActualWithGolden(session, goldenFileAbsPath, true)
})

It("should exit with status code 0", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init", "-h")
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, initCommand, "-h")
Eventually(session).Should(gexec.Exit(0))
})
})
Expand Down Expand Up @@ -128,9 +121,8 @@ var _ = Describe("Init", func() {
os.Setenv(terrascanConfigEnvName, "")
})
It("should error out and exit with status code 1", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init")
Eventually(session, initCommandTimeout).Should(gexec.Exit(1))
helper.ContainsErrorSubString(session, `failed to download policies. error: 'Get "https://repository/url/info/refs?service=git-upload-pack": dial tcp: lookup repository on 8.8.8.8:53: no such host'`)
session = runInitCommand(terrascanBinaryPath, outWriter, errWriter, 1)
helper.ContainsErrorSubString(session, `failed to download policies. error: 'Get "https://repository/url/info/refs?service=git-upload-pack": dial tcp:`)
})
})
When("the config file has invalid branch name", func() {
Expand All @@ -141,9 +133,8 @@ var _ = Describe("Init", func() {
os.Setenv(terrascanConfigEnvName, "")
})
It("should error out and exit with status code 1", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init")
Eventually(session, initCommandTimeout).Should(gexec.Exit(1))
helper.ContainsErrorSubString(session, `failed to checkout branch 'invalid-branch'. error: 'reference not found'`)
session = runInitCommand(terrascanBinaryPath, outWriter, errWriter, 1)
helper.ContainsErrorSubString(session, `failed to initialize terrascan. error : failed to checkout git branch 'invalid-branch'. error: 'reference not found'`)
})
})
When("the config file has invalid rego subdir", func() {
Expand All @@ -154,8 +145,7 @@ var _ = Describe("Init", func() {
os.Setenv(terrascanConfigEnvName, "")
})
It("should error out and exit with status code 1", func() {
session = helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, "init")
Eventually(session, initCommandTimeout).Should(gexec.Exit(1))
session = runInitCommand(terrascanBinaryPath, outWriter, errWriter, 1)
helper.ContainsErrorSubString(session, "invalid/path: no such file or directory")
})
})
Expand All @@ -166,9 +156,74 @@ var _ = Describe("Init", func() {
JustAfterEach(func() {
os.Setenv(terrascanConfigEnvName, "")
})
It("should error out and exit with status code 1", func() {
Skip("Skipping invalid path test until discussion with team")
It("should should download policies and exit with status code 0", func() {
runInitCommand(terrascanBinaryPath, outWriter, errWriter, 0)
})
})
Context("the config file has valid data", func() {
When("config file has different git repo and branch", func() {
JustBeforeEach(func() {
os.Setenv(terrascanConfigEnvName, "config/valid_config.toml")
})
JustAfterEach(func() {
os.Setenv(terrascanConfigEnvName, "")
})
It("init should download the repo provided in the config file", func() {
runInitCommand(terrascanBinaryPath, outWriter, errWriter, 0)
})
Context("Kai Monkey git repo is downloaded", func() {
It("should validate Kai Monkey repo in the policy path", func() {
repo := openGitRepo(defaultPolicyRepoPath)
validateGitRepo(repo, kaiMoneyGitURL)
})
})
})
})
})

Describe("terrascan init is run multiple times", func() {
Context("init clones the git repo to a temp dir, deletes policy path and renames tempdir to policy path", func() {
Context("running init the first time", func() {
var modifiedTime time.Time
It("should download policies at the default policy path", func() {
runInitCommand(terrascanBinaryPath, outWriter, errWriter, 0)
fi, err := os.Stat(defaultPolicyRepoPath)
Expect(err).ToNot(HaveOccurred())
modifiedTime = fi.ModTime()
})
Context("running init the second time", func() {
It("should download policies again at the default policy path", func() {
runInitCommand(terrascanBinaryPath, outWriter, errWriter, 0)
fi, err := os.Stat(defaultPolicyRepoPath)
Expect(err).ToNot(HaveOccurred())
Expect(fi.ModTime()).To(BeTemporally(">", modifiedTime))
})
})
})
})
})
})

func runInitCommand(terrascanBinaryPath string, outWriter, errWriter io.Writer, exitCode int) *gexec.Session {
session := helper.RunCommand(terrascanBinaryPath, outWriter, errWriter, initCommand)
Eventually(session, initCommandTimeout).Should(gexec.Exit(exitCode))
return session
}

func openGitRepo(repoPath string) *git.Repository {
repo, err := git.PlainOpen(repoPath)
Expect(err).NotTo(HaveOccurred())
Expect(repo).NotTo(BeNil())
return repo
}

func validateGitRepo(repo *git.Repository, gitURL string) {
remote, err := repo.Remote("origin")
Expect(err).NotTo(HaveOccurred())
Expect(remote).NotTo(BeNil())
remoteConfig := remote.Config()
Expect(remoteConfig).NotTo(BeNil())
err = remoteConfig.Validate()
Expect(err).NotTo(HaveOccurred())
Expect(remoteConfig.URLs[0]).To(BeEquivalentTo(gitURL))
}
18 changes: 9 additions & 9 deletions test/helper/helper.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,33 +6,33 @@ import (
"os"
"os/exec"

. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/onsi/ginkgo"
"github.com/onsi/gomega"
"github.com/onsi/gomega/gexec"
)

// CompareActualWithGolden compares
func CompareActualWithGolden(session *gexec.Session, goldenFileAbsPath string, isStdOut bool) {
fileData, err := ioutil.ReadFile(goldenFileAbsPath)
Expect(err).NotTo(HaveOccurred())
gomega.Expect(err).NotTo(gomega.HaveOccurred())
if isStdOut {
Expect(string(session.Wait().Out.Contents())).Should(BeIdenticalTo(string(fileData)))
gomega.Expect(string(session.Wait().Out.Contents())).Should(gomega.BeIdenticalTo(string(fileData)))
} else {
Expect(string(session.Wait().Err.Contents())).Should(BeIdenticalTo(string(fileData)))
gomega.Expect(string(session.Wait().Err.Contents())).Should(gomega.BeIdenticalTo(string(fileData)))
}
}

// ContainsErrorSubString will assert if error string is part of error output
func ContainsErrorSubString(session *gexec.Session, errSubString string) {
Expect(string(session.Wait().Err.Contents())).Should(ContainSubstring(errSubString))
gomega.Expect(string(session.Wait().Err.Contents())).Should(gomega.ContainSubstring(errSubString))
}

// GetTerrascanBinaryPath returns the terrascan binary path
func GetTerrascanBinaryPath() string {
terrascanBinaryPath := os.Getenv("TERRASCAN_BIN_PATH")
Describe("terrascan binary path should be set for executing tests", func() {
ginkgo.Describe("terrascan binary path should be set for executing tests", func() {
if terrascanBinaryPath == "" {
Fail("ensure that TERRASCAN_BIN_PATH is set")
ginkgo.Fail("ensure that TERRASCAN_BIN_PATH is set")
}
})
return terrascanBinaryPath
Expand All @@ -42,6 +42,6 @@ func GetTerrascanBinaryPath() string {
func RunCommand(path string, outWriter, errWriter io.Writer, args ...string) *gexec.Session {
cmd := exec.Command(path, args...)
session, err := gexec.Start(cmd, outWriter, errWriter)
Expect(err).NotTo(HaveOccurred())
gomega.Expect(err).NotTo(gomega.HaveOccurred())
return session
}
13 changes: 13 additions & 0 deletions tools/tools.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
// +build tools

package tools

import (
// used only for testing
_ "github.com/onsi/ginkgo/ginkgo"
// used only for testing
_ "github.com/onsi/gomega"
)

// This file imports packages that are used when running go generate, or used
// during the development process but not otherwise depended on by built code.

0 comments on commit 2434d9e

Please sign in to comment.