Skip to content

Commit

Permalink
Add Java-/Typescript linting capabilities (#1636)
Browse files Browse the repository at this point in the history
This change adds linting support for Java-/TypeScript projects based on the checkDefaultLint and checkUserLint steps from Cloud SDK Pipeline.

Following options are supported:

1. Define a linting script named ci-lint in the package.json file(s) of the project
2. Provide a configuration for ESLint as part of the project
3. If none of the above: ESLint is executed with a general purpose configuration.

Co-authored-by: Daniel Kurzynski <[email protected]>
  • Loading branch information
KevinHudemann and daniel-kurzynski authored Jun 22, 2020
1 parent 7c0c524 commit 60fa1d5
Show file tree
Hide file tree
Showing 13 changed files with 694 additions and 13 deletions.
168 changes: 168 additions & 0 deletions cmd/npmExecuteLint.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
package cmd

import (
"fmt"
"github.com/SAP/jenkins-library/pkg/command"
piperhttp "github.com/SAP/jenkins-library/pkg/http"
"github.com/SAP/jenkins-library/pkg/log"
"github.com/SAP/jenkins-library/pkg/npm"
"github.com/SAP/jenkins-library/pkg/piperutils"
"github.com/SAP/jenkins-library/pkg/telemetry"
"io/ioutil"
"net/http"
"os"
"path/filepath"
"strconv"
"strings"
)

type lintUtils interface {
Glob(pattern string) (matches []string, err error)

getExecRunner() execRunner
getGeneralPurposeConfig(configURL string)
}

type lintUtilsBundle struct {
*piperutils.Files
execRunner *command.Command
client *piperhttp.Client
}

func newLintUtilsBundle() *lintUtilsBundle {
return &lintUtilsBundle{
Files: &piperutils.Files{},
client: &piperhttp.Client{},
}
}

func (u *lintUtilsBundle) getExecRunner() execRunner {
if u.execRunner == nil {
u.execRunner = &command.Command{}
u.execRunner.Stdout(log.Writer())
u.execRunner.Stderr(log.Writer())
}
return u.execRunner
}

func (u *lintUtilsBundle) getGeneralPurposeConfig(configURL string) {
response, err := u.client.SendRequest(http.MethodGet, configURL, nil, nil, nil)
if err != nil {
log.Entry().Warnf("failed to download general purpose configuration: %v", err)
return
}

defer response.Body.Close()

content, err := ioutil.ReadAll(response.Body)
if err != nil {
log.Entry().Warnf("error while reading the general purpose configuration: %v", err)
return
}

err = u.FileWrite(filepath.Join(".pipeline", ".eslintrc.json"), content, os.ModePerm)
if err != nil {
log.Entry().Warnf("failed to write .eslintrc.json file to .pipeline/: %w", err)
}
}

func npmExecuteLint(config npmExecuteLintOptions, telemetryData *telemetry.CustomData) {
utils := newLintUtilsBundle()
npmExecutorOptions := npm.ExecutorOptions{DefaultNpmRegistry: config.DefaultNpmRegistry, SapNpmRegistry: config.SapNpmRegistry, ExecRunner: utils.getExecRunner()}
npmExecutor := npm.NewExecutor(npmExecutorOptions)

err := runNpmExecuteLint(npmExecutor, utils, &config)
if err != nil {
log.Entry().WithError(err).Fatal("step execution failed")
}
}

func runNpmExecuteLint(npmExecutor npm.Executor, utils lintUtils, config *npmExecuteLintOptions) error {
packageJSONFiles := npmExecutor.FindPackageJSONFiles()
packagesWithCiLint, _ := npmExecutor.FindPackageJSONFilesWithScript(packageJSONFiles, "ci-lint")

if len(packagesWithCiLint) > 0 {
err := runCiLint(npmExecutor, config.FailOnError)
if err != nil {
return err
}
} else {
err := runDefaultLint(npmExecutor, utils, config.FailOnError)
if err != nil {
return err
}
}
return nil
}

func runCiLint(npmExecutor npm.Executor, failOnError bool) error {
runScripts := []string{"ci-lint"}
runOptions := []string{"--silent"}

err := npmExecutor.RunScriptsInAllPackages(runScripts, runOptions, false)
if err != nil {
if failOnError {
return fmt.Errorf("ci-lint script execution failed with error: %w. This might be the result of severe linting findings, or some other issue while executing the script. Please examine the linting results in the UI, the cilint.xml file, if available, or the log above. ", err)
}
}
return nil
}

func runDefaultLint(npmExecutor npm.Executor, utils lintUtils, failOnError bool) error {
execRunner := utils.getExecRunner()
eslintConfigs := findEslintConfigs(utils)

err := npmExecutor.SetNpmRegistries()
if err != nil {
log.Entry().Warnf("failed to set npm registries before running default lint: %v", err)
}

// If the user has ESLint configs in the project we use them to lint existing JS files. In this case we do not lint other types of files,
// i.e., .jsx, .ts, .tsx, since we can not be sure that the provided config enables parsing of these file types.
if len(eslintConfigs) > 0 {
for i, config := range eslintConfigs {
dir := filepath.Dir(config)
if dir == "." {
err = execRunner.RunExecutable("npx", "eslint", ".", "-f", "checkstyle", "-o", "./"+strconv.Itoa(i)+"_defaultlint.xml", "--ignore-pattern", "node_modules/", "--ignore-pattern", ".eslintrc.js")
} else {
lintPattern := dir + "/**/*.js"
err = execRunner.RunExecutable("npx", "eslint", lintPattern, "-f", "checkstyle", "-o", "./"+strconv.Itoa(i)+"_defaultlint.xml", "--ignore-pattern", "node_modules/", "--ignore-pattern", ".eslintrc.js")
}
if err != nil {
if failOnError {
return fmt.Errorf("Lint execution failed. This might be the result of severe linting findings, problems with the provided ESLint configuration (%s), or another issue. Please examine the linting results in the UI or in %s, if available, or the log above. ", config, strconv.Itoa(i)+"_defaultlint.xml")
}
}
}
} else {
// install dependencies manually, since npx cannot resolve the dependencies required for general purpose
// ESLint config, e.g., TypeScript ESLint plugin
log.Entry().Info("Run ESLint with general purpose config")
utils.getGeneralPurposeConfig("https://raw.githubusercontent.com/SAP/jenkins-library/master/resources/.eslintrc.json")

// Ignore possible errors when invoking ESLint to not fail the pipeline based on linting results
_ = execRunner.RunExecutable("npm", "install", "eslint@^7.0.0", "typescript@^3.7.4", "@typescript-eslint/parser@^3.0.0", "@typescript-eslint/eslint-plugin@^3.0.0")
_ = execRunner.RunExecutable("npx", "--no-install", "eslint", ".", "--ext", ".js,.jsx,.ts,.tsx", "-c", ".pipeline/.eslintrc.json", "-f", "checkstyle", "-o", "./defaultlint.xml", "--ignore-pattern", ".eslintrc.js")
}
return nil
}

func findEslintConfigs(utils lintUtils) []string {
unfilteredListOfEslintConfigs, _ := utils.Glob("**/.eslintrc.*")

var eslintConfigs []string

for _, config := range unfilteredListOfEslintConfigs {
if strings.Contains(config, "node_modules") {
continue
}

if strings.HasPrefix(config, ".pipeline"+string(os.PathSeparator)) {
continue
}

eslintConfigs = append(eslintConfigs, config)
log.Entry().Info("Discovered ESLint config " + config)
}
return eslintConfigs
}
122 changes: 122 additions & 0 deletions cmd/npmExecuteLint_generated.go

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

16 changes: 16 additions & 0 deletions cmd/npmExecuteLint_generated_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package cmd

import (
"testing"

"github.com/stretchr/testify/assert"
)

func TestNpmExecuteLintCommand(t *testing.T) {

testCmd := NpmExecuteLintCommand()

// only high level testing performed - details are tested in step generation procudure
assert.Equal(t, "npmExecuteLint", testCmd.Use, "command name incorrect")

}
Loading

0 comments on commit 60fa1d5

Please sign in to comment.