Skip to content

Commit

Permalink
Enhance Maven errors
Browse files Browse the repository at this point in the history
  • Loading branch information
4ernovm authored and emilwareus committed Dec 12, 2023
1 parent 254a8e0 commit 61d2c6a
Show file tree
Hide file tree
Showing 2 changed files with 140 additions and 18 deletions.
109 changes: 100 additions & 9 deletions internal/resolution/pm/maven/job.go
Original file line number Diff line number Diff line change
@@ -1,14 +1,19 @@
package maven

import (
"path/filepath"

"github.com/debricked/cli/internal/resolution/job"
"github.com/debricked/cli/internal/resolution/pm/util"
"path/filepath"
"regexp"
"strings"
)

const (
lockFileExtension = "maven.debricked.lock"
lockFileExtension = "maven.debricked.lock"
nonParseablePomErrRegex = "Non-parseable POM (.*)"
networkUnreachableErrRegex = "Failed to retrieve plugin descriptor"
invalidVersionErrRegex = "('[\\w\\.]+' for [\\w\\.:-]+ must not contain any of these characters .* but found .)"
dependenciesResolveErrRegex = "(Could not resolve dependencies for project .*)\\("
)

type Job struct {
Expand All @@ -27,18 +32,104 @@ func (j *Job) Run() {
workingDirectory := filepath.Dir(filepath.Clean(j.GetFile()))
cmd, err := j.cmdFactory.MakeDependencyTreeCmd(workingDirectory)
if err != nil {
j.Errors().Critical(util.NewPMJobError(err.Error()))
j.handleError(util.NewPMJobError(err.Error()))

return
}
j.SendStatus("creating dependency graph")

status := "creating dependency graph"
j.SendStatus(status)
var output []byte
output, err = cmd.Output()
if err != nil {
if output == nil {
j.Errors().Critical(util.NewPMJobError(err.Error()))
} else {
j.Errors().Critical(util.NewPMJobError(string(output)))
errContent := err.Error()
if output != nil {
errContent = string(output)
}

cmdErr := util.NewPMJobError(errContent)
cmdErr.SetStatus(status)

j.handleError(cmdErr)
}
}

func (j *Job) handleError(cmdErr job.IError) {
expressions := []string{
nonParseablePomErrRegex,
networkUnreachableErrRegex,
invalidVersionErrRegex,
dependenciesResolveErrRegex,
}

for _, expression := range expressions {
regex := regexp.MustCompile(expression)

if regex.MatchString(cmdErr.Error()) {
cmdErr = j.addDocumentation(expression, regex, cmdErr)
j.Errors().Critical(cmdErr)
return
}
}

j.Errors().Critical(cmdErr)
}

func (j *Job) addDocumentation(expr string, regex *regexp.Regexp, cmdErr job.IError) job.IError {
switch {
case expr == nonParseablePomErrRegex:
matches := regex.FindAllStringSubmatch(cmdErr.Error(), 1)
message := "the POM file for errors"
if len(matches) > 0 && len(matches[0]) > 1 {
message = matches[0][1]
}

cmdErr.SetDocumentation(
strings.Join(
[]string{
"Failed to build Maven dependency tree.",
"Your POM file is not valid.",
"Please check",
message,
}, " "),
)
case expr == networkUnreachableErrRegex:
cmdErr.SetDocumentation(
strings.Join(
[]string{
"We weren't able to retrieve one or more plugin descriptor(s).",
"Please check your Internet connection and try again.",
}, " "),
)
case expr == invalidVersionErrRegex:
matches := regex.FindAllStringSubmatch(cmdErr.Error(), 1)
message := "the POM file for errors"
if len(matches) > 0 && len(matches[0]) > 1 {
message = matches[0][1]
}

cmdErr.SetDocumentation(
strings.Join(
[]string{
"There is an error in dependencies:",
message,
}, " "),
)
case expr == dependenciesResolveErrRegex:
matches := regex.FindAllStringSubmatch(cmdErr.Error(), 1)
message := "the POM file for errors"
if len(matches) > 0 && len(matches[0]) > 1 {
message = matches[0][1]
}

cmdErr.SetDocumentation(
strings.Join(
[]string{
message,
"\nTry to run `mvn dependency:tree -e` to get more details",
}, " "),
)
}

return cmdErr
}
49 changes: 40 additions & 9 deletions internal/resolution/pm/maven/job_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,15 +17,46 @@ func TestNewJob(t *testing.T) {
}

func TestRunCmdErr(t *testing.T) {
cmdErr := errors.New("cmd-error")
j := NewJob("file", testdata.CmdFactoryMock{Err: cmdErr})

go jobTestdata.WaitStatus(j)

j.Run()

assert.Len(t, j.Errors().GetAll(), 1)
assert.Contains(t, j.Errors().GetAll(), util.NewPMJobError(cmdErr.Error()))
cases := []struct {
error string
doc string
}{
{
error: "cmd-error",
doc: "No specific documentation for this problem yet, please report it to us! :)",
},
{
error: " |[FATAL] Non-parseable POM /home/asus/Projects/playground/maven-project/pom.xml: end tag name </target> must be the same as start tag <source> from line 37 (position: TEXT seen ...<source>1.6</target>... @37:31) @ /home/asus/Projects/playground/maven-project/pom.xml, line 37, column 31\n",
doc: "Failed to build Maven dependency tree. Your POM file is not valid. Please check /home/asus/Projects/playground/maven-project/pom.xml: end tag name </target> must be the same as start tag <source> from line 37 (position: TEXT seen ...<source>1.6</target>... @37:31) @ /home/asus/Projects/playground/maven-project/pom.xml, line 37, column 31",
},
{
error: " |[WARNING] Failed to retrieve plugin descriptor for org.apache.maven.plugins:maven-compiler-plugin:2.3.2: Plugin org.apache.maven.plugins:maven-compiler-plugin:2.3.2 or one of its dependencies could not be resolved: Failed to read artifact descriptor for org.apache.maven.plugins:maven-compiler-plugin:jar:2.3.2\n",
doc: "We weren't able to retrieve one or more plugin descriptor(s). Please check your Internet connection and try again.",
},
{
error: " |[ERROR] 'dependencies.dependency.version' for org.hamcrest:hamcrest-library:jar must not contain any of these characters \\/:\"<>|?* but found * @ com.example.maven-project:maven-project:1.0-SNAPSHOT, /home/asus/Projects/playground/maven-project/pom.xml, line 196, column 18\n",
doc: "There is an error in dependencies: 'dependencies.dependency.version' for org.hamcrest:hamcrest-library:jar must not contain any of these characters \\/:\"<>|?* but found *",
},
{
error: " |[ERROR] Failed to execute goal on project jackpot: Could not resolve dependencies for project com.jeteo:jackpot:war:1.0-SNAPSHOT: The following artifacts could not be resolved: javax.servlet:com.springsource.javax.servlet:jar:2.5.0, javax.servlet:com.springsource.javax.servlet.jsp.jstl:jar:1.2.0 (http://repository.springsource.com/maven/bundles/release) -> [Help 1]\n",
doc: "Could not resolve dependencies for project com.jeteo:jackpot:war:1.0-SNAPSHOT: The following artifacts could not be resolved: javax.servlet:com.springsource.javax.servlet:jar:2.5.0, javax.servlet:com.springsource.javax.servlet.jsp.jstl:jar:1.2.0 \nTry to run `mvn dependency:tree -e` to get more details",
},
}

for _, c := range cases {
expectedError := util.NewPMJobError(c.error)
expectedError.SetDocumentation(c.doc)

cmdErr := errors.New(c.error)
j := NewJob("file", testdata.CmdFactoryMock{Err: cmdErr})

go jobTestdata.WaitStatus(j)

j.Run()

assert.Len(t, j.Errors().GetAll(), 1)
assert.Contains(t, j.Errors().GetAll(), expectedError)
}
}

func TestRunCmdOutputErr(t *testing.T) {
Expand Down

0 comments on commit 61d2c6a

Please sign in to comment.