2019-06-04 08:01:43 +02:00
|
|
|
import static com.sap.piper.Prerequisites.checkScript
|
|
|
|
import com.sap.piper.GenerateDocumentation
|
|
|
|
import com.sap.piper.ConfigurationHelper
|
|
|
|
import com.sap.piper.Utils
|
|
|
|
import groovy.transform.Field
|
|
|
|
|
|
|
|
@Field def STEP_NAME = getClass().getName()
|
|
|
|
@Field Set GENERAL_CONFIG_KEYS = [
|
|
|
|
/**
|
|
|
|
* Dockerfile to be used for the assessment.
|
|
|
|
*/
|
|
|
|
'dockerFile',
|
|
|
|
/**
|
|
|
|
* Name of the docker image that should be used, in which node should be installed and configured. Default value is 'hadolint/hadolint:latest-debian'.
|
|
|
|
*/
|
|
|
|
'dockerImage'
|
|
|
|
]
|
|
|
|
@Field Set STEP_CONFIG_KEYS = GENERAL_CONFIG_KEYS.plus([
|
|
|
|
/**
|
|
|
|
* Name of the configuration file used locally within the step. If a file with this name is detected as part of your repo downloading the central configuration via `configurationUrl` will be skipped. If you change the file's name make sure your stashing configuration also reflects this.
|
|
|
|
*/
|
|
|
|
'configurationFile',
|
|
|
|
/**
|
|
|
|
* URL pointing to the .hadolint.yaml exclude configuration to be used for linting. Also have a look at `configurationFile` which could avoid central configuration download in case the file is part of your repository.
|
|
|
|
*/
|
|
|
|
'configurationUrl',
|
|
|
|
/**
|
|
|
|
* Docker options to be set when starting the container.
|
|
|
|
*/
|
|
|
|
'dockerOptions',
|
|
|
|
/**
|
|
|
|
* Quality Gates to fail the build, see [warnings-ng plugin documentation](https://github.com/jenkinsci/warnings-plugin/blob/master/doc/Documentation.md#quality-gate-configuration).
|
|
|
|
*/
|
|
|
|
'qualityGates',
|
|
|
|
/**
|
|
|
|
* Name of the result file used locally within the step.
|
|
|
|
*/
|
2020-02-04 17:36:09 +02:00
|
|
|
'reportFile',
|
|
|
|
/**
|
|
|
|
* Name of the checkstyle report being generated our of the results.
|
|
|
|
*/
|
|
|
|
'reportName'
|
2019-06-04 08:01:43 +02:00
|
|
|
])
|
|
|
|
@Field Set PARAMETER_KEYS = STEP_CONFIG_KEYS
|
|
|
|
/**
|
|
|
|
* Executes the Haskell Dockerfile Linter which is a smarter Dockerfile linter that helps you build [best practice](https://docs.docker.com/develop/develop-images/dockerfile_best-practices/) Docker images.
|
|
|
|
* The linter is parsing the Dockerfile into an abstract syntax tree (AST) and performs rules on top of the AST.
|
|
|
|
*/
|
|
|
|
@GenerateDocumentation
|
|
|
|
void call(Map parameters = [:]) {
|
|
|
|
handlePipelineStepErrors(stepName: STEP_NAME, stepParameters: parameters) {
|
|
|
|
final script = checkScript(this, parameters) ?: this
|
|
|
|
final utils = parameters.juStabUtils ?: new Utils()
|
2020-08-26 15:32:58 +02:00
|
|
|
String stageName = parameters.stageName ?: env.STAGE_NAME
|
2019-06-04 08:01:43 +02:00
|
|
|
|
|
|
|
// load default & individual configuration
|
|
|
|
Map configuration = ConfigurationHelper.newInstance(this)
|
2020-08-26 15:32:58 +02:00
|
|
|
.loadStepDefaults([:], stageName)
|
2019-06-04 08:01:43 +02:00
|
|
|
.mixinGeneralConfig(script.commonPipelineEnvironment, GENERAL_CONFIG_KEYS)
|
|
|
|
.mixinStepConfig(script.commonPipelineEnvironment, STEP_CONFIG_KEYS)
|
2020-08-26 15:32:58 +02:00
|
|
|
.mixinStageConfig(script.commonPipelineEnvironment, stageName, STEP_CONFIG_KEYS)
|
2019-06-04 08:01:43 +02:00
|
|
|
.mixin(parameters, PARAMETER_KEYS)
|
|
|
|
.use()
|
|
|
|
|
|
|
|
new Utils().pushToSWA([
|
|
|
|
step: STEP_NAME,
|
|
|
|
stepParamKey1: 'scriptMissing',
|
|
|
|
stepParam1: parameters?.script == null
|
|
|
|
], configuration)
|
|
|
|
|
|
|
|
def existingStashes = utils.unstashAll(configuration.stashContent)
|
|
|
|
|
|
|
|
if (!fileExists(configuration.dockerFile)) {
|
|
|
|
error "[${STEP_NAME}] Dockerfile '${configuration.dockerFile}' is not found."
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!fileExists(configuration.configurationFile) && configuration.configurationUrl) {
|
|
|
|
sh "curl --fail --location --output ${configuration.configurationFile} ${configuration.configurationUrl}"
|
|
|
|
if(existingStashes) {
|
|
|
|
def stashName = 'hadolintConfiguration'
|
|
|
|
stash name: stashName, includes: configuration.configurationFile
|
|
|
|
existingStashes += stashName
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
def options = [
|
|
|
|
"--config ${configuration.configurationFile}",
|
|
|
|
"--format checkstyle > ${configuration.reportFile}"
|
|
|
|
]
|
|
|
|
|
|
|
|
dockerExecute(
|
|
|
|
script: script,
|
|
|
|
dockerImage: configuration.dockerImage,
|
|
|
|
dockerOptions: configuration.dockerOptions,
|
|
|
|
stashContent: existingStashes
|
|
|
|
) {
|
|
|
|
// HaDoLint status code is ignore, results will be handled by recordIssues / archiveArtifacts
|
2020-02-04 17:36:09 +02:00
|
|
|
def result = sh returnStatus: true, script: "hadolint ${configuration.dockerFile} ${options.join(' ')}"
|
2019-06-04 08:01:43 +02:00
|
|
|
|
|
|
|
archiveArtifacts configuration.reportFile
|
|
|
|
recordIssues(
|
2020-02-04 17:36:09 +02:00
|
|
|
tools: [checkStyle(
|
|
|
|
name: configuration.reportName,
|
|
|
|
pattern: configuration.reportFile,
|
|
|
|
id: configuration.reportName
|
|
|
|
)],
|
2019-06-04 08:01:43 +02:00
|
|
|
qualityGates: configuration.qualityGates,
|
|
|
|
enabledForFailure: true,
|
|
|
|
blameDisabled: true
|
|
|
|
)
|
2020-02-04 17:36:09 +02:00
|
|
|
|
|
|
|
def resultFileSize = 0
|
|
|
|
if (fileExists(configuration.reportFile)) {
|
|
|
|
resultFileSize = readFile(configuration.reportFile).length()
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result != 0 && resultFileSize == 0) {
|
|
|
|
error "HaDoLint scan on file ${configuration.dockerFile} failed due to technical issues, please check the log."
|
|
|
|
}
|
2019-06-04 08:01:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|