Skip to content

Commit

Permalink
feat(cli): TestRunner cli client (#2951)
Browse files Browse the repository at this point in the history
  • Loading branch information
schoren committed Jul 17, 2023
1 parent 837a6b5 commit 1c7b6a8
Show file tree
Hide file tree
Showing 7 changed files with 362 additions and 0 deletions.
22 changes: 22 additions & 0 deletions cli/cmd/resources.go
Expand Up @@ -206,6 +206,28 @@ var (
resourcemanager.WithResourceType("DataStore"),
),
).
Register(
resourcemanager.NewClient(
httpClient, cliLogger,
"testrunner", "testrunners",
resourcemanager.WithTableConfig(resourcemanager.TableConfig{
Cells: []resourcemanager.TableCellConfig{
{Header: "ID", Path: "spec.id"},
{Header: "NAME", Path: "spec.name"},
{Header: "REQUIRED GATES", Path: "spec.gates"},
},
ItemModifier: func(item *gabs.Container) error {
gates := []string{}
for _, gate := range item.Path("spec.requiredGates").Children() {
gates = append(gates, "- "+gate.Data().(string))
}
item.SetP(strings.Join(gates, "\n"), "spec.gates")
return nil
},
}),
resourcemanager.WithResourceType("TestRunner"),
),
).
Register(environmentClient).
Register(transactionClient).
Register(testClient)
Expand Down
@@ -0,0 +1,48 @@
package testrunner

import (
"fmt"
"testing"

"github.com/kubeshop/tracetest/cli-e2etest/environment"
"github.com/kubeshop/tracetest/cli-e2etest/helpers"
"github.com/kubeshop/tracetest/cli-e2etest/testscenarios/types"
"github.com/kubeshop/tracetest/cli-e2etest/tracetestcli"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func TestApplyTestRunner(t *testing.T) {
// instantiate require with testing helper
require := require.New(t)
assert := assert.New(t)

// setup isolated e2e environment
env := environment.CreateAndStart(t)
defer env.Close(t)

cliConfig := env.GetCLIConfigPath(t)

// Given I am a Tracetest CLI user
// And I have my server recently created

// When I try to set up a new testRunner
// Then it should be applied with success
testRunnerPath := env.GetTestResourcePath(t, "new-testrunner")

result := tracetestcli.Exec(t, fmt.Sprintf("apply testrunner --file %s", testRunnerPath), tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

// When I try to get a testRunner again
// Then it should return the testRunner applied on the last step, with analytics disabled
result = tracetestcli.Exec(t, "get testrunner --id current", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

testRunner := helpers.UnmarshalYAML[types.TestRunnerResource](t, result.StdOut)
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
}
@@ -0,0 +1,30 @@
package testrunner

import (
"testing"

"github.com/kubeshop/tracetest/cli-e2etest/environment"
"github.com/kubeshop/tracetest/cli-e2etest/helpers"
"github.com/kubeshop/tracetest/cli-e2etest/tracetestcli"
"github.com/stretchr/testify/require"
)

func TestDeleteTestRunner(t *testing.T) {
// instantiate require with testing helper
require := require.New(t)

// setup isolated e2e environment
env := environment.CreateAndStart(t)
defer env.Close(t)

cliConfig := env.GetCLIConfigPath(t)

// Given I am a Tracetest CLI user
// And I have my server recently created

// When I try to delete the testrunner
// Then it should return a error message, showing that we cannot delete a testrunner
result := tracetestcli.Exec(t, "delete testrunner --id current", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 1)
require.Contains(result.StdErr, "resource TestRunner does not support the action")
}
115 changes: 115 additions & 0 deletions testing/cli-e2etest/testscenarios/testrunner/get_testrunner_test.go
@@ -0,0 +1,115 @@
package testrunner

import (
"fmt"
"testing"

"github.com/kubeshop/tracetest/cli-e2etest/environment"
"github.com/kubeshop/tracetest/cli-e2etest/helpers"
"github.com/kubeshop/tracetest/cli-e2etest/testscenarios/types"
"github.com/kubeshop/tracetest/cli-e2etest/tracetestcli"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func addGetTestRunnerPreReqs(t *testing.T, env environment.Manager) {
cliConfig := env.GetCLIConfigPath(t)

// When I try to set up a testrunner
// Then it should be applied with success
testRunnerPath := env.GetTestResourcePath(t, "new-testrunner")

result := tracetestcli.Exec(t, fmt.Sprintf("apply testrunner --file %s", testRunnerPath), tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)
}

func TestGetTestRunner(t *testing.T) {
// instantiate require with testing helper
require := require.New(t)
assert := assert.New(t)

env := environment.CreateAndStart(t)
defer env.Close(t)

cliConfig := env.GetCLIConfigPath(t)

t.Run("get with no testrunner initialized", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And no testrunner previously registered

// When I try to get a testrunner on yaml mode
// Then it should print a YAML with the default testrunner
result := tracetestcli.Exec(t, "get testrunner --id current --output yaml", tracetestcli.WithCLIConfig(cliConfig))
require.Equal(0, result.ExitCode)

testRunner := helpers.UnmarshalYAML[types.TestRunnerResource](t, result.StdOut)
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
})

addGetTestRunnerPreReqs(t, env)

t.Run("get with YAML format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I have a testrunner already set

// When I try to get a testrunner on yaml mode
// Then it should print a YAML
result := tracetestcli.Exec(t, "get testrunner --id current --output yaml", tracetestcli.WithCLIConfig(cliConfig))
require.Equal(0, result.ExitCode)

testRunner := helpers.UnmarshalYAML[types.TestRunnerResource](t, result.StdOut)
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
})

t.Run("get with JSON format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I have a testrunner already set

// When I try to get a testrunner on json mode
// Then it should print a json
result := tracetestcli.Exec(t, "get testrunner --id current --output json", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

testRunner := helpers.UnmarshalJSON[types.TestRunnerResource](t, result.StdOut)
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
})

t.Run("get with pretty format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I have a testrunner already set

// When I try to get a testrunner on pretty mode
// Then it should print a table with 4 lines printed: header, separator, a testrunner item and empty line
result := tracetestcli.Exec(t, "get testrunner --id current --output pretty", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

parsedTable := helpers.UnmarshalTable(t, result.StdOut)
// this output shows one gate per line, so the parser reads that as an entire new row
require.Len(parsedTable, 2)

singleLine := parsedTable[0]
require.Equal("current", singleLine["ID"])
require.Equal("default", singleLine["NAME"])
require.Equal("- analyzer-score", parsedTable[0]["REQUIRED GATES"])
require.Equal("- test-specs", parsedTable[1]["REQUIRED GATES"])
})
}
128 changes: 128 additions & 0 deletions testing/cli-e2etest/testscenarios/testrunner/list_testrunner_test.go
@@ -0,0 +1,128 @@
package testrunner

import (
"fmt"
"strings"
"testing"

"github.com/kubeshop/tracetest/cli-e2etest/environment"
"github.com/kubeshop/tracetest/cli-e2etest/helpers"
"github.com/kubeshop/tracetest/cli-e2etest/testscenarios/types"
"github.com/kubeshop/tracetest/cli-e2etest/tracetestcli"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func addListConfigPreReqs(t *testing.T, env environment.Manager) {
cliConfig := env.GetCLIConfigPath(t)

// When I try to set up a new testrunner
// Then it should be applied with success
testRunnerPath := env.GetTestResourcePath(t, "new-testrunner")

result := tracetestcli.Exec(t, fmt.Sprintf("apply testrunner --file %s", testRunnerPath), tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)
}

func TestListConfig(t *testing.T) {
// instantiate require with testing helper
require := require.New(t)
assert := assert.New(t)

env := environment.CreateAndStart(t)
defer env.Close(t)

cliConfig := env.GetCLIConfigPath(t)

t.Run("list with no testrunner initialized", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created

// When I try to list testrunner on pretty mode and there is no testrunner previously registered
// Then it should print an empty table
// Then it should print a table with 5 lines printed: header, separator, the default testrunner item, an entire line for the second gate and empty line
result := tracetestcli.Exec(t, "list testrunner --sortBy name --output pretty", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)
require.Contains(result.StdOut, "current")

lines := strings.Split(result.StdOut, "\n")
require.Len(lines, 5)
})

addListConfigPreReqs(t, env)

t.Run("list with invalid sortBy field", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I already have a testrunner created

// When I try to list a testrunner by an invalid field
// Then I should receive an error
result := tracetestcli.Exec(t, "list testrunner --sortBy id --output yaml", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 1)
require.Contains(result.StdErr, "invalid sort field: id") // TODO: think on how to improve this error handling
})

t.Run("list with YAML format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I already have a testrunner created

// When I try to list testrunner again on yaml mode
// Then it should print a YAML list with one item
result := tracetestcli.Exec(t, "list testrunner --sortBy name --output yaml", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

testRunner := helpers.UnmarshalYAML[types.TestRunnerResource](t, result.StdOut)
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
})

t.Run("list with JSON format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I already have a testrunner created

// When I try to list testrunner again on json mode
// Then it should print a JSON list with one item
result := tracetestcli.Exec(t, "list testrunner --sortBy name --output json", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

testRunnerList := helpers.UnmarshalJSON[types.ResourceList[types.TestRunnerResource]](t, result.StdOut)
require.Len(testRunnerList.Items, 1)
require.Equal(len(testRunnerList.Items), testRunnerList.Count)

testRunner := testRunnerList.Items[0]
assert.Equal("TestRunner", testRunner.Type)
assert.Equal("current", testRunner.Spec.ID)
assert.Equal("default", testRunner.Spec.Name)
require.Len(testRunner.Spec.RequiredGates, 2)
assert.Equal("analyzer-score", testRunner.Spec.RequiredGates[0])
assert.Equal("test-specs", testRunner.Spec.RequiredGates[1])
})

t.Run("list with pretty format", func(t *testing.T) {
// Given I am a Tracetest CLI user
// And I have my server recently created
// And I already have a testrunner created

// When I try to list testrunner again on pretty mode
// Then it should print a table with 4 lines printed: header, separator, testrunner item and empty line
result := tracetestcli.Exec(t, "list testrunner --sortBy name --output pretty", tracetestcli.WithCLIConfig(cliConfig))
helpers.RequireExitCodeEqual(t, result, 0)

parsedTable := helpers.UnmarshalTable(t, result.StdOut)
// this output shows one gate per line, so the parser reads that as an entire new row
require.Len(parsedTable, 2)

singleLine := parsedTable[0]
require.Equal("current", singleLine["ID"])
require.Equal("default", singleLine["NAME"])
require.Equal("- analyzer-score", parsedTable[0]["REQUIRED GATES"])
require.Equal("- test-specs", parsedTable[1]["REQUIRED GATES"])
})
}
@@ -0,0 +1,7 @@
type: TestRunner
spec:
id: current
name: default
requiredGates:
- analyzer-score
- test-specs
12 changes: 12 additions & 0 deletions testing/cli-e2etest/testscenarios/types/testrunner.go
@@ -0,0 +1,12 @@
package types

type TestRunnerResource struct {
Type string `json:"type"`
Spec TestRunner `json:"spec"`
}

type TestRunner struct {
ID string `json:"id"`
Name string `json:"name"`
RequiredGates []string `json:"requiredGates"`
}

0 comments on commit 1c7b6a8

Please sign in to comment.