Skip to content

Commit

Permalink
Merge pull request #161 from symflower/158-coverage-failing-tests
Browse files Browse the repository at this point in the history
Do not ignore coverage count if there are failing tests
  • Loading branch information
bauersimon authored Jun 5, 2024
2 parents fbe850a + 30113f2 commit efe1ea3
Show file tree
Hide file tree
Showing 9 changed files with 125 additions and 32 deletions.
3 changes: 2 additions & 1 deletion evaluate/repository.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,8 @@ func Repository(logger *log.Logger, resultPath string, model evalmodel.Model, la
repositoryAssessment.Add(assessments)
repositoryAssessment.Award(metrics.AssessmentKeyResponseNoError)

coverage, err := language.Execute(log, testDataPath)
coverage, ps, err := language.Execute(log, testDataPath)
problems = append(problems, ps...)
if err != nil {
problems = append(problems, pkgerrors.WithMessage(err, filePath))

Expand Down
31 changes: 22 additions & 9 deletions language/golang/language.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import (
"os"
"path/filepath"
"regexp"
"strconv"
"strings"

pkgerrors "github.com/pkg/errors"
Expand Down Expand Up @@ -74,14 +75,10 @@ func (l *Language) TestFramework() (testFramework string) {
return ""
}

var languageGoNoTestsMatch = regexp.MustCompile(`(?m)^DONE (\d+) tests.*in (.+?)$`)
var languageGoCoverageMatch = regexp.MustCompile(`(?m)^coverage: (\d+\.?\d+)% of statements`)
var languageGoNoCoverageMatch = regexp.MustCompile(`(?m)^coverage: \[no statements\]$`)
var languageGoTestsErrorMatch = regexp.MustCompile(`DONE (\d+) tests, (\d+) failure`)

// Execute invokes the language specific testing on the given repository.
func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage uint64, err error) {
coverageFilePath := filepath.Join(repositoryPath, "coverage.json")

func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage uint64, problems []error, err error) {
commandOutput, err := util.CommandWithResult(context.Background(), logger, &util.Command{
Command: []string{
"go",
Expand All @@ -92,9 +89,10 @@ func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage
Directory: repositoryPath,
})
if err != nil {
return 0, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
return 0, problems, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
}

coverageFilePath := filepath.Join(repositoryPath, "coverage.json")
commandOutput, err = util.CommandWithResult(context.Background(), logger, &util.Command{
Command: []string{
tools.SymflowerPath, "test",
Expand All @@ -106,8 +104,23 @@ func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage
Directory: repositoryPath,
})
if err != nil {
return 0, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
testSummary := languageGoTestsErrorMatch.FindStringSubmatch(commandOutput)
if len(testSummary) > 0 {
if failureCount, e := strconv.Atoi(testSummary[2]); e != nil {
return 0, problems, pkgerrors.WithStack(e)
} else if failureCount > 0 {
// If there are test failures, then this is just a soft error since we still are able to receive coverage data.
problems = append(problems, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput))
}
} else {
return 0, problems, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
}
}

coverage, err = language.CoverageObjectCountOfFile(coverageFilePath)
if err != nil {
return 0, problems, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
}

return language.CoverageObjectCountOfFile(coverageFilePath)
return coverage, problems, nil
}
42 changes: 37 additions & 5 deletions language/golang/language_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -64,9 +64,10 @@ func TestLanguageExecute(t *testing.T) {
RepositoryPath string
RepositoryChange func(t *testing.T, repositoryPath string)

ExpectedCoverage uint64
ExpectedError error
ExpectedErrorText string
ExpectedCoverage uint64
ExpectedProblemTexts []string
ExpectedError error
ExpectedErrorText string
}

validate := func(t *testing.T, tc *testCase) {
Expand All @@ -89,7 +90,12 @@ func TestLanguageExecute(t *testing.T) {
if tc.Language == nil {
tc.Language = &Language{}
}
actualCoverage, actualError := tc.Language.Execute(logger, repositoryPath)
actualCoverage, actualProblems, actualError := tc.Language.Execute(logger, repositoryPath)

require.Equal(t, len(tc.ExpectedProblemTexts), len(actualProblems), "the number of expected problems need to match the number of actual problems")
for i, expectedProblemText := range tc.ExpectedProblemTexts {
assert.ErrorContains(t, actualProblems[i], expectedProblemText)
}

if tc.ExpectedError != nil {
assert.ErrorIs(t, actualError, tc.ExpectedError)
Expand All @@ -107,7 +113,8 @@ func TestLanguageExecute(t *testing.T) {

RepositoryPath: filepath.Join("..", "..", "testdata", "golang", "plain"),

ExpectedCoverage: 0,
ExpectedCoverage: 0,
ExpectedErrorText: "exit status 1",
})

t.Run("With test file", func(t *testing.T) {
Expand All @@ -132,6 +139,31 @@ func TestLanguageExecute(t *testing.T) {
ExpectedCoverage: 1,
})

validate(t, &testCase{
Name: "Failing tests",

RepositoryPath: filepath.Join("..", "..", "testdata", "golang", "light"),
RepositoryChange: func(t *testing.T, repositoryPath string) {
require.NoError(t, os.WriteFile(filepath.Join(repositoryPath, "simpleIfElse_test.go"), []byte(bytesutil.StringTrimIndentations(`
package light
import (
"testing"
)
func TestSimpleIfElse(t *testing.T) {
simpleIfElse(1) // Get some coverage...
t.Fail() // ...and then fail.
}
`)), 0660))
},

ExpectedCoverage: 1,
ExpectedProblemTexts: []string{
"exit status 1", // Test execution fails.
},
})

validate(t, &testCase{
Name: "Syntax error",

Expand Down
11 changes: 8 additions & 3 deletions language/java/language.go
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,7 @@ func (l *Language) TestFramework() (testFramework string) {
var languageJavaCoverageMatch = regexp.MustCompile(`Total coverage (.+?)%`)

// Execute invokes the language specific testing on the given repository.
func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage uint64, err error) {
func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage uint64, problems []error, err error) {
coverageFilePath := filepath.Join(repositoryPath, "coverage.json")
commandOutput, err := util.CommandWithResult(context.Background(), logger, &util.Command{
Command: []string{
Expand All @@ -102,8 +102,13 @@ func (l *Language) Execute(logger *log.Logger, repositoryPath string) (coverage
Directory: repositoryPath,
})
if err != nil {
return 0, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
return 0, nil, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
}

return language.CoverageObjectCountOfFile(coverageFilePath)
coverage, err = language.CoverageObjectCountOfFile(coverageFilePath)
if err != nil {
return 0, nil, pkgerrors.WithMessage(pkgerrors.WithStack(err), commandOutput)
}

return coverage, nil, nil
}
42 changes: 37 additions & 5 deletions language/java/language_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -130,9 +130,10 @@ func TestLanguageExecute(t *testing.T) {
RepositoryPath string
RepositoryChange func(t *testing.T, repositoryPath string)

ExpectedCoverage uint64
ExpectedError error
ExpectedErrorText string
ExpectedCoverage uint64
ExpectedProblemTexts []string
ExpectedError error
ExpectedErrorText string
}

validate := func(t *testing.T, tc *testCase) {
Expand All @@ -155,7 +156,12 @@ func TestLanguageExecute(t *testing.T) {
if tc.Language == nil {
tc.Language = &Language{}
}
actualCoverage, actualError := tc.Language.Execute(logger, repositoryPath)
actualCoverage, actualProblems, actualError := tc.Language.Execute(logger, repositoryPath)

require.Equal(t, len(tc.ExpectedProblemTexts), len(actualProblems), "the number of expected problems need to match the number of actual problems")
for i, expectedProblemText := range tc.ExpectedProblemTexts {
assert.ErrorContains(t, actualProblems[i], expectedProblemText)
}

if tc.ExpectedError != nil {
assert.ErrorIs(t, actualError, tc.ExpectedError)
Expand All @@ -173,7 +179,8 @@ func TestLanguageExecute(t *testing.T) {

RepositoryPath: filepath.Join("..", "..", "testdata", "java", "plain"),

ExpectedCoverage: 0, // TODO Let the test case identify and error that there are no test files (needs to be implemented in `symflower test`). https://github.com/symflower/eval-dev-quality/issues/35
ExpectedCoverage: 0,
ExpectedErrorText: "exit status 1",
})

t.Run("With test file", func(t *testing.T) {
Expand Down Expand Up @@ -201,6 +208,31 @@ func TestLanguageExecute(t *testing.T) {
ExpectedCoverage: 1,
})

validate(t, &testCase{
Name: "Failing tests",

RepositoryPath: filepath.Join("..", "..", "testdata", "java", "light"),
RepositoryChange: func(t *testing.T, repositoryPath string) {
javaTestFilePath := filepath.Join(repositoryPath, "src/test/java/com/eval/SimpleIfElseSymflowerTest.java")
require.NoError(t, os.MkdirAll(filepath.Dir(javaTestFilePath), 0755))
require.NoError(t, os.WriteFile(javaTestFilePath, []byte(bytesutil.StringTrimIndentations(`
package com.eval;
import org.junit.jupiter.api.*;
public class SimpleIfElseSymflowerTest {
@Test
public void simpleIfElse() {
int actual = SimpleIfElse.simpleIfElse(1); // Get some coverage...
Assertions.assertEquals(true, false); // ... and then fail.
}
}
`)), 0660))
},

ExpectedCoverage: 3,
})

validate(t, &testCase{
Name: "Syntax error",

Expand Down
2 changes: 1 addition & 1 deletion language/language.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ type Language interface {
TestFramework() (testFramework string)

// Execute invokes the language specific testing on the given repository.
Execute(logger *log.Logger, repositoryPath string) (coverage uint64, err error)
Execute(logger *log.Logger, repositoryPath string) (coverage uint64, problems []error, err error)
}

// Languages holds a register of all languages.
Expand Down
21 changes: 15 additions & 6 deletions language/testing/Language_mock_gen.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 2 additions & 1 deletion model/symflower/symflower_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,8 +62,9 @@ func TestModelGenerateTestsForFile(t *testing.T) {
}
metricstesting.AssertAssessmentsEqual(t, tc.ExpectedAssessment, actualAssessment)

actualCoverage, err := tc.Language.Execute(logger, repositoryPath)
actualCoverage, actualProblems, err := tc.Language.Execute(logger, repositoryPath)
require.NoError(t, err)
require.Empty(t, actualProblems)
assert.Equal(t, tc.ExpectedCoverage, actualCoverage)
})
}
Expand Down
2 changes: 1 addition & 1 deletion tools/symflower.go
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ func (*symflower) CheckVersion(logger *log.Logger, binaryPath string) (err error
}

// SymflowerVersionRequired holds the version of Symflower required for this revision of the evaluation.
const SymflowerVersionRequired = "36800"
const SymflowerVersionRequired = "37153"

// RequiredVersion returns the required version of the tool.
func (*symflower) RequiredVersion() string {
Expand Down

0 comments on commit efe1ea3

Please sign in to comment.