Skip to content

Commit

Permalink
Merge pull request #18 from dailymotion/before_after_feature
Browse files Browse the repository at this point in the history
Implemented setup/teardown feature
ubermensch01 authored Jan 24, 2020
2 parents b778d93 + 25a235b commit e894c24
Showing 12 changed files with 628 additions and 72 deletions.
335 changes: 289 additions & 46 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,88 +1,331 @@
# Allure Golang Integrations
# Allure Golang Integration

allure-go is a Go package that provides support for Allure reports in Golang : https://github.com/allure-framework/allure2
`allure-go` is a Go package that provides all required tools to transform your tests into [Allure](https://github.com/allure-framework/allure2) reports.

## Installing
## Table of Contents

To start using allure-go, install Go and run `go get`:
1. [Installation](#installation)
2. [Usage](#usage)
1. [Specifying `allure-results` folder location](#specifying-allure-results-folder-location)
2. [Optional environment variable](#optional-environment-variable)
3. [Features](#features)
1. [Test Wrapper](#test-wrapper)
2. [Nested Steps](#nested-steps)
3. [Option Slices](#option-slices)
1. [Description](#description)
2. [Action](#action)
3. [Parameter](#parameter)
4. [Parameters](#parameters)
4. [Test-specific options](#test-specific-options)
1. [Lead](#lead)
2. [Owner](#owner)
3. [Epic](#epic)
4. [Severity](#severity)
5. [Story](#story)
6. [Feature](#feature)
4. [Parameters](#parameters-as-a-feature)
5. [Attachments](#attachments)
6. [Setup/Teardown](#setupteardown)
7. [Environment files](#environment-files)
4. [Feature Roadmap](#feature-roadmap)


## Installation
In order to retrieve `allure-go` simply run the following command (assuming `Go` is installed):
```sh
$ go get -u github.com/dailymotion/allure-go
```

This will retrieve the library.

## Usage

### Specifying allure-results folder location
Golang's approach to evaluating test scripts does not allow to establish a single location for test results
programmatically, the following environment variable is required for `allure-go` to work as expected.
In order to specify the location of a report run the following:
```shell script
export ALLURE_RESULTS_PATH=</some/path>
```

`allure-results` folder will be created in that location.

### Optional environment variable
Allure-go will retrieve the absolute path of your testing files (for example, /Users/myuser/Dev/myProject/tests/myfile_test.go) and will display this path in the reports.

First of all, you need to set an environment variable to define the `allure-results` folder location:
To make it cleaner, you can trim prefix the path of your project by defining the `ALLURE_WORKSPACE_PATH` with the value of your project root path :
```shell script
export ALLURE_WORKSPACE_PATH=/another/path
```
export ALLURE_RESULTS_PATH=/some/path

You will now get the relative path of your test files from your project root level.

## Features

### Test Wrapper
`allure-go` test reporting relies on wrapping test scripts inside a call to `allure.Test()` function:
```go
allure.Test(t *testing.T, testOptions ...Option)
```
This should be the path where the `allure-results` folder exists or should be created, not the path of the folder itself.
This function allows adding required modules representing a test along with additional non-essential modules like labels.
For basic usage refer to the following example:
```go
package test

The `allure-results` folder is automatically created if it doesn't exist with `drwxr-xr-x` file system permission.
import (
"fmt"
"github.com/dailymotion/allure-go"
"testing"
)

func TestStep(t *testing.T) {
allure.Test(t, allure.Action(func() {
fmt.Println("This block of code is a test")
}))
}
```

### Using environment file
### Nested Steps
`allure-go` takes advantage of Allure Reports feature of nested steps and allows marking portions of the code as steps
even if said portions are found outside test scripts in functions that are eventually called by a test script.
This is a great feature for enabling levels of detail for test reporting.
```go
package test

Allure reporting tool allows putting run-specific properties in the report provided
by `environment.properties` or `environment.xml` file.
import (
"fmt"
"github.com/dailymotion/allure-go"
"testing"
)

func TestStep(t *testing.T) {
allure.Test(t, allure.Action(func() {
allure.Step(allure.Description("Action"),
allure.Action(func() {
fmt.Println("This block of code is a step")
}))
PerformAction()
}))
}

In order to take advantage of this feature, set an environment variable to specify the location of the file:
func PerformAction() {
allure.Step(allure.Description("Action"),
allure.Action(func() {
fmt.Println("This block of code is also a step")
}))
}
```
export ALLURE_ENVIRONMENT_FILE_PATH=/path/to/file/environment.properties
```

### Examples
### Option Slices
Option slices allow providing as much or as little detail for the test scripts as needed.

To implement this library in your tests, follow the [examples](example/example_test.go).
#### Description
```go
allure.Description("Description of a test or a step")
```
Provides a name for the step and a description for the test. This option is required

#### Action
```go
allure.Action(func() {})
```
This option is a wrapper for the portion of code that is either your test or your step. This option is required.

Execute tests with the usual go test command :
#### Parameter
```go
allure.Parameter(name string, value interface{})
```
go test ./example
This option specifies parameters for the test or step accordingly.
This particular option can be called multiple times to add multiple parameters.

#### Parameters
```go
allure.Parameters(parameters map[string]interface{})
```
This option allows specifying multiple parameters at once.
This option can be called multiple times as well and will not interfere with previous parameter assignments.

This will automatically generate an `allure-results` folder in the path defined by ALLURE_RESULTS_PATH.
### Test-specific options

To see the report in html, generate it with the allure command line :
#### Lead
```go
allure.Lead(lead string)
```
allure serve $ALLURE_RESULTS_PATH/allure-results
This option specifies a lead of the feature.

#### Owner
```go
allure.Owner(owner string)
```
This will automatically generate and open the HTML reports in a browser.
This option specifies an owner of the feature.

The results file are compatible with the [Jenkins plugin](https://wiki.jenkins.io/display/JENKINS/Allure+Plugin).
#### Epic
```go
allure.Epic(epic string)
```
This option specifies an epic this test belongs to.

## Optional environment variable
#### Severity
```go
allure.Severity(severity severity.Severity)
```
This option specifies a severity level of the test script to allow better prioritization.

Allure-go will retrieve the absolute path of your testing files (for example, /Users/myuser/Dev/myProject/tests/myfile_test.go) and will display this path in the reports.
#### Story
```go
allure.Story(story string)
```
This option specifies a story this test script belongs to.
This particular option can be called multiple times to connect this test script to multiple stories.

To make it cleaner, you can trim prefix the path of your project by defining the `ALLURE_WORKSPACE_PATH` with the value of your project root path :
#### Feature
```go
allure.Feature(feature string)
```
export ALLURE_WORKSPACE_PATH=/another/path
This option specifies a feature this test script belongs to.
This particular option can be called multiple times to connect this test script to multiple features.

### Parameters as a feature
`allure-go` allows specifying parameters on both test and step level to further improvement informativeness of your
test scripts and bring it one step closer to being documentation.
In order to specify a parameter refer to the following example:
```go
package test

import (
"github.com/dailymotion/allure-go"
"testing"
)

func TestParameter(t *testing.T) {
allure.Test(t,
allure.Description("Test that has parameters"),
allure.Parameter("testParameter", "test"),
allure.Action(func() {
allure.Step(allure.Description("Step with parameters"),
allure.Action(func() {}),
allure.Parameter("stepParameter", "step"))
}))
}
```
Allure parameters integrate neatly with Golang's parameterized tests too:
```go
package test

You will now get the relative path of your test files from your project root level.
import (
"github.com/dailymotion/allure-go"
"testing"
)

func TestParameterized(t *testing.T) {
for i := 0; i < 5; i++ {
t.Run("", func(t *testing.T) {
allure.Test(t,
allure.Description("Test with parameters"),
allure.Parameter("counter", i),
allure.Action(func() {}))
})
}
}
```

### Attachments
`allure-go` providing an ability to attach content of various MIME types to test scripts and steps.
The most common MIME types are available as constants in `allure-go`
(`image/png`, `application/json`, `text/plain` and `video/mpeg`).
In order to attach content to a test or step refer to the following example:
```go
package test

## Features available
import (
"errors"
"github.com/dailymotion/allure-go"
"log"
"testing"
)

This project is still in progress. Here is the list of available features :
- It is possible to create a test object in the report with the `allure.Test()` method.
- It is possible to create a step object inside a test object of the report with the `allure.Step()` method.
- It is possible to pass parameters to a test object or a step object with the `allure.TestWithParameters()` and `allure.StepWithParameters()` methods.
- It is possible to add attachments to a test object or a step object with the `allure.AddAttachment()` method (depending if it is called inside an `allure.Test()` wrapper or an `allure.Step()` wrapper).
func TestImageAttachmentToStep(t *testing.T) {
allure.Test(t,
allure.Description("testing image attachment"),
allure.Action(func() {
allure.Step(allure.Description("adding an image attachment"),
allure.Action(func() {
err := allure.AddAttachment("image", allure.ImagePng, []byte("<byte array representing an image>"))
if err != nil {
log.Println(err)
}
}))
}))
}
```

### Setup/Teardown
Golang does not directly follow the setup/teardown approach of other languages like Java, C# and Python.
This does not prevent us from logically separating said phases in test scripts and taking
advantage of separating these phases in the test reports too.
`allure.BeforeTest` and `allure.AfterTest` functions have to be called in sequence with the test wrapper.

`allure.BeforeTest` goes first, then `allure.Test` and finally `allure.AfterTest`.
This is done to allow various logical conclusions, such as upon failure of `allure.BeforeTest` actual test body
will be skipped, etc.

Please refer to the following example of setup/teardown usage:
```go
package test

import (
"github.com/dailymotion/allure-go"
"testing"
)
func TestAllureSetupTeardown(t *testing.T) {
allure.BeforeTest(t,
allure.Description("setup"),
allure.Action(func() {
allure.Step(
allure.Description("Setup step 1"),
allure.Action(func() {}))
}))

allure.Test(t,
allure.Description("actual test"),
allure.Action(func() {
allure.Step(
allure.Description("Test step 1"),
allure.Action(func() {}))
}))

allure.AfterTest(t,
allure.Description("teardown"),
allure.Action(func() {
allure.Step(
allure.Description("Teardown step 1"),
allure.Action(func() {}))
}))
}
```
In case of utilizing setup/teardown in a parameterized test, all of the test phases have to be called
inside `t.Run(...)` call.

### Environment Files
`allure-go` allows taking advantage of the environment file feature of Allure reports. You can specify report-specific
variables that you want to appear in the report such as browser kind and version, OS, environment name, etc.
In order to do that create an `environment.xml` or `environment.properties` file as instructed [here](https://docs.qameta.io/allure/#_environment) and define an
environment variable for `allure-go` to incorporate in the results:
```shell script
export ALLURE_ENVIRONMENT_FILE_PATH=<path to environment file>
```

## Improvements needed
## Feature Roadmap
`allure-go` is still in active development and is not yet stabilized or finalized.

Here is the list of improvements that are needed for existing features :
Here is a list of improvements that are needed for existing features :
- [X] Manage failure at step level : the project must be able to mark a Step as failed (red in the report) when an assertion fails. The assertion stacktrace must appear in the report.
- [ ] Manage error at test and step level : the project must be able to mark a Test and Step as broken (orange in the report) when an error happens. The error stacktrace must appear in the report.
- [ ] Add support of Links
- [X] Manage error at test and step level : the project must be able to mark a Test and Step as broken (orange in the report) when an error happens. The error stacktrace must appear in the report.
- [X] Add support of Links
- [ ] Add support of Unknown status for Test object
- [ ] Add support of Set up in Execution part of a Test
- [ ] Add support of Tear down in Execution part of a Test
- [ ] Add support of Severity
- [X] Add support of Set up in Execution part of a Test
- [X] Add support of Tear down in Execution part of a Test
- [X] Add support of Severity
- [ ] Add support of Flaky tag
- [ ] Add support of Categories
- [ ] Add support of Features
- [X] Add support for environment files
- [X] Add support of Categories
- [X] Add support of Features
- [X] Add support for environment files
- [ ] Add support for history
51 changes: 51 additions & 0 deletions after.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
package allure

import (
"github.com/jtolds/gls"
"log"
"testing"
)

// AfterTest executes a teardown phase of the test and adds parameters to the Allure container object
func AfterTest(t *testing.T, testOptions ...Option) {
after := newAfter()
afterSubContainer := after.Afters[0]
after.UUID = generateUUID()
afterSubContainer.Start = getTimestampMs()

afterSubContainer.Steps = make([]stepObject, 0)
for _, option := range testOptions {
option(afterSubContainer)
}

defer func() {
afterSubContainer.Stop = getTimestampMs()
afterSubContainer.Status = getTestStatus(t)
afterSubContainer.Stage = "finished"
testPhaseObject := getCurrentTestPhaseObject(t)
if testPhaseObject.Test.UUID == "" {
log.Printf("Test's \"%s\" allure teardщwn is being executed before allure test!\n", t.Name())
}

after.Children = append(after.Children, testPhaseObject.Test.UUID)

testPhaseObject.Afters = append(testPhaseObject.Afters, after)
err := after.writeResultsFile()
if err != nil {
log.Println("Failed to write content of result to json file", err)
}
}()
ctxMgr.SetValues(gls.Values{
testResultKey: afterSubContainer,
nodeKey: afterSubContainer,
testInstanceKey: t,
}, afterSubContainer.Action)
}

func newAfter() *container {
return &container{
Children: make([]string, 0),
Links: make([]string, 0),
Afters: []*subContainer{newHelper()},
}
}
10 changes: 5 additions & 5 deletions attachment.go
Original file line number Diff line number Diff line change
@@ -32,16 +32,16 @@ func AddAttachment(name string, mimeType MimeType, content []byte) error {
}

func (a *attachment) writeAttachmentFile() error {
resultsPathEnv := os.Getenv(ResultsPathEnvKey)
resultsPathEnv := os.Getenv(resultsPathEnvKey)
if resultsPathEnv == "" {
log.Printf("%s environment variable cannot be empty\n", ResultsPathEnvKey)
log.Printf("%s environment variable cannot be empty\n", resultsPathEnvKey)
}
if ResultsPath == "" {
ResultsPath = fmt.Sprintf("%s/allure-results", resultsPathEnv)
if resultsPath == "" {
resultsPath = fmt.Sprintf("%s/allure-results", resultsPathEnv)
}

a.Source = fmt.Sprintf("%s-attachment", a.uuid)
err := ioutil.WriteFile(fmt.Sprintf("%s/%s-attachment", ResultsPath, a.uuid), a.content, 0777)
err := ioutil.WriteFile(fmt.Sprintf("%s/%s-attachment", resultsPath, a.uuid), a.content, 0777)
if err != nil {
return errors.Wrap(err, "Failed to write in file")
}
43 changes: 43 additions & 0 deletions before.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package allure

import (
"github.com/jtolds/gls"
"log"
"testing"
)

// BeforeTest executes a setup phase of the test and adds parameters to the Allure container object
func BeforeTest(t *testing.T, testOptions ...Option) {
testPhaseObject := getCurrentTestPhaseObject(t)
if testPhaseObject.Test != nil {
log.Printf("Test's \"%s\" allure setup is being executed after allure test!\n", t.Name())
}
before := newBefore()
testPhaseObject.Befores = append(testPhaseObject.Befores, before)
beforeSubContainer := before.Befores[0]

before.UUID = generateUUID()
beforeSubContainer.Start = getTimestampMs()
beforeSubContainer.Steps = make([]stepObject, 0)
for _, option := range testOptions {
option(beforeSubContainer)
}

defer func() {
beforeSubContainer.Stop = getTimestampMs()
beforeSubContainer.Status = getTestStatus(t)
beforeSubContainer.Stage = "finished"

}()
ctxMgr.SetValues(gls.Values{
testResultKey: beforeSubContainer,
nodeKey: beforeSubContainer,
testInstanceKey: t,
}, beforeSubContainer.Action)
}

func newBefore() *container {
return &container{
Befores: []*subContainer{newHelper()},
}
}
6 changes: 3 additions & 3 deletions error.go
Original file line number Diff line number Diff line change
@@ -34,14 +34,14 @@ func allureError(err error, status string, now bool) {
manipulateOnObjectFromCtx(
testResultKey,
func(testResult interface{}) {
testStatusDetails := testResult.(*result).StatusDetails
testStatusDetails := testResult.(hasStatusDetails).getStatusDetails()
if testStatusDetails == nil {
testStatusDetails = &statusDetails{}
}
testStatusDetails.Trace = filterStackTrace(debug.Stack())
testStatusDetails.Message = err.Error()
testResult.(*result).StatusDetails = testStatusDetails
testResult.(*result).Status = status
testResult.(hasStatusDetails).setStatusDetails(*testStatusDetails)
testResult.(hasStatus).setStatus(status)
})
manipulateOnObjectFromCtx(
nodeKey,
45 changes: 45 additions & 0 deletions example/example_before_after_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
package example

import (
"errors"
"github.com/dailymotion/allure-go"
"testing"
)

func TestAllureSetupTeardown(t *testing.T) {
allure.BeforeTest(t,
allure.Description("setup"),
allure.Action(func() {
allure.Step(
allure.Description("Setup step 1"),
allure.Action(func() {}))
}))

allure.Test(t,
allure.Description("actual test"),
allure.Action(func() {
allure.Step(
allure.Description("Test step 1"),
allure.Action(func() {}))
}))

allure.AfterTest(t,
allure.Description("teardown"),
allure.Action(func() {
allure.Step(
allure.Description("teardown step 1"),
allure.Action(func() {}))
}))
}

func TestAllureSetupFailed(t *testing.T) {
allure.BeforeTest(t,
allure.Description("setup failed"),
allure.Action(func() {
allure.Fail(errors.New("some error"))
}))

allure.Test(t,
allure.Description("actual test"),
allure.Action(func() {}))
}
14 changes: 8 additions & 6 deletions init.go
Original file line number Diff line number Diff line change
@@ -8,20 +8,22 @@ import (
var (
ctxMgr *gls.ContextManager
wsd string
ResultsPath string
CreateFolderOnce sync.Once
CopyEnvFileOnce sync.Once
resultsPath string
createFolderOnce sync.Once
copyEnvFileOnce sync.Once
testPhaseObjects map[string]*testPhaseContainer
)

const (
ResultsPathEnvKey = "ALLURE_RESULTS_PATH"
WsPathEnvKey = "ALLURE_WORKSPACE_PATH"
EnvFileKey = "ALLURE_ENVIRONMENT_FILE_PATH"
resultsPathEnvKey = "ALLURE_RESULTS_PATH"
wsPathEnvKey = "ALLURE_WORKSPACE_PATH"
envFileKey = "ALLURE_ENVIRONMENT_FILE_PATH"
nodeKey = "current_step_container"
testResultKey = "test_result_object"
testInstanceKey = "test_instance"
)

func init() {
ctxMgr = gls.NewContextManager()
testPhaseObjects = make(map[string]*testPhaseContainer)
}
5 changes: 5 additions & 0 deletions interfaces.go
Original file line number Diff line number Diff line change
@@ -26,3 +26,8 @@ type hasAttachments interface {
getAttachments() []attachment
addAttachment(attachment attachment)
}

type hasStatusDetails interface {
getStatusDetails() *statusDetails
setStatusDetails(details statusDetails)
}
16 changes: 8 additions & 8 deletions misc.go
Original file line number Diff line number Diff line change
@@ -64,25 +64,25 @@ func getTimestampMs() int64 {
}

func createFolderIfNotExists() {
resultsPathEnv := os.Getenv(ResultsPathEnvKey)
resultsPathEnv := os.Getenv(resultsPathEnvKey)
if resultsPathEnv == "" {
log.Printf("environment variable %s cannot be empty\n", ResultsPathEnvKey)
log.Printf("environment variable %s cannot be empty\n", resultsPathEnvKey)
}
ResultsPath = fmt.Sprintf("%s/allure-results", resultsPathEnv)
resultsPath = fmt.Sprintf("%s/allure-results", resultsPathEnv)

if _, err := os.Stat(ResultsPath); os.IsNotExist(err) {
err = os.Mkdir(ResultsPath, 0777)
if _, err := os.Stat(resultsPath); os.IsNotExist(err) {
err = os.Mkdir(resultsPath, 0777)
if err != nil {
log.Println(err, "Failed to create allure-results folder")
}
}
}

func copyEnvFileIfExists() {
if envFilePath := os.Getenv(EnvFileKey); envFilePath != "" {
if envFilePath := os.Getenv(envFileKey); envFilePath != "" {
envFilesStrings := strings.Split(envFilePath, "/")
if ResultsPath != "" {
if _, err := copy(envFilePath, ResultsPath+"/"+envFilesStrings[len(envFilesStrings)-1]); err != nil {
if resultsPath != "" {
if _, err := copy(envFilePath, resultsPath+"/"+envFilesStrings[len(envFilesStrings)-1]); err != nil {
log.Println("Could not copy the environment file", err)
}
}
16 changes: 12 additions & 4 deletions result.go
Original file line number Diff line number Diff line change
@@ -86,8 +86,16 @@ func (r *result) getStatus() string {
return r.Status
}

func (r *result) getStatusDetails() *statusDetails {
return r.StatusDetails
}

func (r *result) setStatusDetails(details statusDetails) {
r.StatusDetails = &details
}

func (r *result) setDefaultLabels(t *testing.T) {
wsd := os.Getenv(WsPathEnvKey)
wsd := os.Getenv(wsPathEnvKey)

programCounters := make([]uintptr, 10)
callersCount := runtime.Callers(0, programCounters)
@@ -126,14 +134,14 @@ func (r *result) addLabel(name string, value string) {
}

func (r *result) writeResultsFile() error {
CreateFolderOnce.Do(createFolderIfNotExists)
CopyEnvFileOnce.Do(copyEnvFileIfExists)
createFolderOnce.Do(createFolderIfNotExists)
copyEnvFileOnce.Do(copyEnvFileIfExists)

j, err := json.Marshal(r)
if err != nil {
return errors.Wrap(err, "Failed to marshall result into JSON")
}
err = ioutil.WriteFile(fmt.Sprintf("%s/%s-result.json", ResultsPath, r.UUID), j, 0777)
err = ioutil.WriteFile(fmt.Sprintf("%s/%s-result.json", resultsPath, r.UUID), j, 0777)
if err != nil {
return errors.Wrap(err, "Failed to write in file")
}
13 changes: 13 additions & 0 deletions test.go
Original file line number Diff line number Diff line change
@@ -44,6 +44,7 @@ func Test(t *testing.T, testOptions ...Option) {

defer func() {
panicObject := recover()
getCurrentTestPhaseObject(t).Test = r
r.Stop = getTimestampMs()
if panicObject != nil {
t.Fail()
@@ -62,6 +63,18 @@ func Test(t *testing.T, testOptions ...Option) {
if err != nil {
log.Println("Failed to write content of result to json file", err)
}
setups := getCurrentTestPhaseObject(t).Befores
for _, setup := range setups {
setup.Children = append(setup.Children, r.UUID)
err := setup.writeResultsFile()
if err != nil {
log.Println("Failed to write content of result to json file", err)
}
}

if panicObject != nil {
panic(panicObject)
}
}()
ctxMgr.SetValues(gls.Values{
testResultKey: r,
146 changes: 146 additions & 0 deletions test_phase_container.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
package allure

import (
"encoding/json"
"fmt"
"github.com/pkg/errors"
"io/ioutil"
"testing"
)

type testPhaseContainer struct {
Befores []*container
Test *result
Afters []*container
}

type container struct {
UUID string `json:"uuid"`
Name string `json:"name"`
Children []string `json:"children"`
Description string `json:"description"`
Befores []*subContainer `json:"befores"`
Afters []*subContainer `json:"afters"`
Links []string `json:"links"`
Start int64 `json:"start"`
Stop int64 `json:"stop"`
}

//subContainer defines a step
type subContainer struct {
Name string `json:"name,omitempty"`
Status string `json:"status,omitempty"`
StatusDetails *statusDetails `json:"statusDetails,omitempty"`
Stage string `json:"stage,omitempty"`
Description string `json:"description,omitempty"`
Start int64 `json:"start,omitempty"`
Stop int64 `json:"stop,omitempty"`
Steps []stepObject `json:"steps,omitempty"`
Attachments []attachment `json:"attachments,omitempty"`
Parameters []parameter `json:"parameters,omitempty"`
Action func() `json:"-"`
}

func (sc *subContainer) addLabel(key string, value string) {
panic("implement me")
}

func (sc *subContainer) addDescription(description string) {
sc.Description = description
}

func (sc *subContainer) addParameter(name string, value interface{}) {
sc.Parameters = append(sc.Parameters, parseParameter(name, value))
}

func (sc *subContainer) addParameters(parameters map[string]interface{}) {
for key, value := range parameters {
sc.Parameters = append(sc.Parameters, parseParameter(key, value))
}
}

func (sc *subContainer) addName(name string) {
sc.Name = name
}

func (sc *subContainer) addAction(action func()) {
sc.Action = action
}

func (sc *subContainer) addReason(reason string) {
testStatusDetails := sc.StatusDetails
if testStatusDetails == nil {
testStatusDetails = &statusDetails{}
}
sc.StatusDetails.Message = reason
}

func (sc *subContainer) getAttachments() []attachment {
return sc.Attachments
}

func (sc *subContainer) addAttachment(attachment attachment) {
sc.Attachments = append(sc.Attachments, attachment)
}

func (sc *subContainer) getSteps() []stepObject {
return sc.Steps
}

func (sc *subContainer) addStep(step stepObject) {
sc.Steps = append(sc.Steps, step)
}

func (sc *subContainer) getStatusDetails() *statusDetails {
return sc.StatusDetails
}

func (sc *subContainer) setStatusDetails(details statusDetails) {
sc.StatusDetails = &details
}

func (sc *subContainer) setStatus(status string) {
sc.Status = status
}

func (sc *subContainer) getStatus() string {
return sc.Status
}

func getCurrentTestPhaseObject(t *testing.T) *testPhaseContainer {
var currentPhaseObject *testPhaseContainer
if phaseContainer, ok := testPhaseObjects[t.Name()]; ok {
currentPhaseObject = phaseContainer
} else {
currentPhaseObject = &testPhaseContainer{
Befores: make([]*container, 0),
Afters: make([]*container, 0),
}
testPhaseObjects[t.Name()] = currentPhaseObject
}

return currentPhaseObject
}

func (c container) writeResultsFile() error {
createFolderOnce.Do(createFolderIfNotExists)
copyEnvFileOnce.Do(copyEnvFileIfExists)

j, err := json.Marshal(c)
if err != nil {
return errors.Wrap(err, "Failed to marshall result into JSON")
}
err = ioutil.WriteFile(fmt.Sprintf("%s/%s-container.json", resultsPath, c.UUID), j, 0777)
if err != nil {
return errors.Wrap(err, "Failed to write in file")
}
return nil
}

func newHelper() *subContainer {
return &subContainer{
Steps: make([]stepObject, 0),
Attachments: make([]attachment, 0),
Parameters: make([]parameter, 0),
}
}

0 comments on commit e894c24

Please sign in to comment.