From c520f6bf27c0106c055b270977359b3a00ad4475 Mon Sep 17 00:00:00 2001 From: Kirill Gevorkyan <26010098+kgevorkyan@users.noreply.github.com> Date: Thu, 16 Feb 2023 11:46:17 +0300 Subject: [PATCH] Execute tool in sarif mode before report checking (#498) ### What's done: * Execute tool before checking the sarif report, this logic allows: 1) Support checking `.sarif` from stdout 2) Rewrite existing sarif report, if `actual[Warnings|Fix]SarifFileName` was provided and something were changed in report * Minor refactoring * Add possible config for one of the test (however it's not required, since `save-cli` will look into stdout by default) --- .../sarif-actual/save-warnings-actual.sarif | 155 ++++++++++++++++++ examples/kotlin-diktat/sarif-actual/save.toml | 3 + .../saveourtool/save/plugins/fix/FixPlugin.kt | 29 ++-- .../save/plugin/warn/WarnPlugin.kt | 86 ++++++---- .../save/plugin/warn/WarnPluginConfig.kt | 6 +- 5 files changed, 229 insertions(+), 50 deletions(-) create mode 100644 examples/kotlin-diktat/sarif-actual/save-warnings-actual.sarif diff --git a/examples/kotlin-diktat/sarif-actual/save-warnings-actual.sarif b/examples/kotlin-diktat/sarif-actual/save-warnings-actual.sarif new file mode 100644 index 000000000..3bb25bbbd --- /dev/null +++ b/examples/kotlin-diktat/sarif-actual/save-warnings-actual.sarif @@ -0,0 +1,155 @@ +{ + "$schema": "https://raw.githubusercontent.com/oasis-tcs/sarif-spec/master/Schemata/sarif-schema-2.1.0.json", + "version": "2.1.0", + "runs": [ + { + "originalUriBaseIds": { + "%SRCROOT%": { + "uri": "file://D:/projects/" + } + }, + "results": [ + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 35, + "startLine": 6 + } + } + } + ], + "message": { + "text": "[WRONG_DECLARATIONS_ORDER] declarations of constants and enum members should be sorted alphabetically: enum entries order is incorrect (diktat-ruleset:abp-sort-rule)" + }, + "ruleId": "diktat-ruleset:abp-sort-rule" + }, + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 5, + "startLine": 10 + } + } + } + ], + "message": { + "text": "[ENUMS_SEPARATED] enum is incorrectly formatted: enums must end with semicolon (diktat-ruleset:abq-enum-separated)" + }, + "ruleId": "diktat-ruleset:abq-enum-separated" + }, + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 5, + "startLine": 8 + } + } + } + ], + "message": { + "text": "[ENUM_VALUE] enum values should be in selected UPPER_CASE snake/PascalCase format: paSC_SAl_l (diktat-ruleset:aai-identifier-naming)" + }, + "ruleId": "diktat-ruleset:aai-identifier-naming" + }, + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 5, + "startLine": 11 + } + } + } + ], + "message": { + "text": "[ENUM_VALUE] enum values should be in selected UPPER_CASE snake/PascalCase format: PascAsl_f (diktat-ruleset:aai-identifier-naming)" + }, + "ruleId": "diktat-ruleset:aai-identifier-naming" + }, + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 5, + "startLine": 10 + } + } + } + ], + "message": { + "text": "[ENUMS_SEPARATED] enum is incorrectly formatted: last enum entry must end with a comma (diktat-ruleset:abq-enum-separated)" + }, + "ruleId": "diktat-ruleset:abq-enum-separated" + }, + { + "level": "error", + "locations": [ + { + "physicalLocation": { + "artifactLocation": { + "uri": "src\\kotlin\\EnumValueSnakeCaseTest.kt", + "uriBaseId": "%SRCROOT%" + }, + "region": { + "startColumn": 9, + "startLine": 1 + } + } + } + ], + "message": { + "text": "[PACKAGE_NAME_INCORRECT_PREFIX] package name should start from company's domain: com.saveourtool.save (diktat-ruleset:aah-package-naming)" + }, + "ruleId": "diktat-ruleset:aah-package-naming" + } + ], + "tool": { + "driver": { + "downloadUri": "https://github.com/pinterest/ktlint/releases/tag/0.42.0", + "fullName": "ktlint", + "informationUri": "https://github.com/pinterest/ktlint/", + "language": "en", + "name": "ktlint", + "organization": "pinterest", + "rules": [ + ], + "semanticVersion": "0.42.0", + "version": "0.42.0" + } + } + } + ] +} diff --git a/examples/kotlin-diktat/sarif-actual/save.toml b/examples/kotlin-diktat/sarif-actual/save.toml index 6b8e1ffd2..34eaef4d7 100644 --- a/examples/kotlin-diktat/sarif-actual/save.toml +++ b/examples/kotlin-diktat/sarif-actual/save.toml @@ -10,6 +10,9 @@ # regular expression to detect tests testNameRegex = ".*Test.kt" actualWarningsFormat = "SARIF" + # by default, in SARIF mode, it's supposed, that tool will print sarif report into the stdout + # however, it also could be provided via file + # actualWarningsFileName = "save-warnings-actual.sarif" fileNameCaptureGroupOut = 1 lineCaptureGroupOut = 2 columnCaptureGroupOut = 3 diff --git a/save-plugins/fix-plugin/src/commonMain/kotlin/com/saveourtool/save/plugins/fix/FixPlugin.kt b/save-plugins/fix-plugin/src/commonMain/kotlin/com/saveourtool/save/plugins/fix/FixPlugin.kt index 388d5ce9b..ecda1e4c0 100644 --- a/save-plugins/fix-plugin/src/commonMain/kotlin/com/saveourtool/save/plugins/fix/FixPlugin.kt +++ b/save-plugins/fix-plugin/src/commonMain/kotlin/com/saveourtool/save/plugins/fix/FixPlugin.kt @@ -19,7 +19,6 @@ import com.saveourtool.save.core.result.DebugInfo import com.saveourtool.save.core.result.Fail import com.saveourtool.save.core.result.Pass import com.saveourtool.save.core.result.TestResult -import com.saveourtool.save.core.utils.ExecutionResult import com.saveourtool.save.core.utils.PathSerializer import com.saveourtool.save.core.utils.ProcessExecutionException import com.saveourtool.save.core.utils.ProcessTimeoutException @@ -104,21 +103,22 @@ class FixPlugin( logDebug("Executing fix plugin in ${fixPluginConfig.actualFixFormat?.name} mode") - val (executionResult, adjustedTestCopyToExpectedFilesMap) = if (fixPluginConfig.actualFixFormat == ActualFixFormat.IN_PLACE) { - try { - // hold testCopyToExpectedFilesMap as is - pb.exec(execCmd, testConfig.getRootConfig().directory.toString(), redirectTo, time) to testCopyToExpectedFilesMap - } catch (ex: ProcessTimeoutException) { - logWarn("The following tests took too long to run and were stopped: ${chunk.map { it.test }}, timeout for single test: ${ex.timeoutMillis}") - return@map failTestResult(chunk, ex, execCmd) - } catch (ex: ProcessExecutionException) { - return@map failTestResult(chunk, ex, execCmd) - } + val executionResult = try { + pb.exec(execCmd, testConfig.getRootConfig().directory.toString(), redirectTo, time) + } catch (ex: ProcessTimeoutException) { + logWarn("The following tests took too long to run and were stopped: ${chunk.map { it.test }}, timeout for single test: ${ex.timeoutMillis}") + return@map failTestResult(chunk, ex, execCmd) + } catch (ex: ProcessExecutionException) { + return@map failTestResult(chunk, ex, execCmd) + } + + val adjustedTestCopyToExpectedFilesMap = if (fixPluginConfig.actualFixFormat == ActualFixFormat.IN_PLACE) { + // hold testCopyToExpectedFilesMap as is + testCopyToExpectedFilesMap } else { - // replace test files by modificated copies, obtained from sarif lib + // replace test files with modified copies, obtained from sarif lib val fixedTestCopyToExpectedFilesMap = applyFixesFromSarif(fixPluginConfig, testsPaths, testCopyToExpectedFilesMap) - val dbgMsg = "Fixes were obtained from SARIF file, no debug info is available" - ExecutionResult(0, listOf(dbgMsg), listOf(dbgMsg)) to fixedTestCopyToExpectedFilesMap + fixedTestCopyToExpectedFilesMap } val stdout = executionResult.stdout @@ -188,6 +188,7 @@ class FixPlugin( * @param testCopyToExpectedFilesMap list of paths to the copy of tests files, which will be replaced by modificated files * @return updated list of test files copies */ + // TODO: support case, when sarif report is provided via stdout private fun applyFixesFromSarif( fixPluginConfig: FixPluginConfig, testsPaths: List, diff --git a/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPlugin.kt b/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPlugin.kt index bf557c3d9..7d2bfb744 100644 --- a/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPlugin.kt +++ b/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPlugin.kt @@ -143,29 +143,24 @@ class WarnPlugin( val execCmd = cmdExecutor.constructExecCmd(tmpDirName) val expectedWarningsMap = try { - collectExpectedWarnings(generalConfig, warnPluginConfig, originalPaths, copyPaths, workingDirectory) + processExpectedWarnings(generalConfig, warnPluginConfig, originalPaths, copyPaths, workingDirectory) } catch (ex: SarifParsingException) { return failTestResult(originalPaths, ex, execCmd) } - if (expectedWarningsMap.isEmpty()) { - warnMissingExpectedWarnings(warnPluginConfig, generalConfig, originalPaths) + val executionResult = try { + cmdExecutor.execCmdAndGetExecutionResults(redirectTo) + } catch (ex: ProcessTimeoutException) { + logWarn("The following tests took too long to run and were stopped: $originalPaths, timeout for single test: ${ex.timeoutMillis}") + return failTestResult(originalPaths, ex, execCmd) + } catch (ex: ProcessExecutionException) { + return failTestResult(originalPaths, ex, execCmd) } - val (actualWarningsMap, result) = try { - actualWarningsIfExistActualWarningsFile(warnPluginConfig, originalPaths, workingDirectory) + val actualWarningsMap = try { + processActualWarnings(executionResult, warnPluginConfig, originalPaths, workingDirectory) } catch (ex: SarifParsingException) { return failTestResult(originalPaths, ex, execCmd) - } ?: run { - val result = try { - cmdExecutor.execCmdAndGetExecutionResults(redirectTo) - } catch (ex: ProcessTimeoutException) { - logWarn("The following tests took too long to run and were stopped: $originalPaths, timeout for single test: ${ex.timeoutMillis}") - return failTestResult(originalPaths, ex, execCmd) - } catch (ex: ProcessExecutionException) { - return failTestResult(originalPaths, ex, execCmd) - } - collectActualWarningsWithLineNumbers(result, warnPluginConfig, workingDirectory) to result } val resultsChecker = ResultsChecker( @@ -181,8 +176,8 @@ class WarnPlugin( resultsStatus.first, DebugInfo( execCmd, - result.stdout.filter { it.contains(path.name) }.joinToString("\n"), - result.stderr.filter { it.contains(path.name) }.joinToString("\n"), + executionResult.stdout.filter { it.contains(path.name) }.joinToString("\n"), + executionResult.stderr.filter { it.contains(path.name) }.joinToString("\n"), null, resultsStatus.second, ), @@ -190,23 +185,6 @@ class WarnPlugin( }.asSequence() } - private fun actualWarningsIfExistActualWarningsFile( - warnPluginConfig: WarnPluginConfig, - originalPaths: List, - workingDirectory: Path, - ) = warnPluginConfig.actualWarningsFileName?.let { - val sarif = calculatePathToSarifFile( - sarifFileName = warnPluginConfig.actualWarningsFileName, - anchorTestFilePath = originalPaths.first() - ) - val execResult = ExecutionResult( - 0, - fs.readLines(sarif), - listOf("Warnings were obtained from SARIF file, no debug info is available") - ) - collectActualWarningsWithLineNumbers(execResult, warnPluginConfig, workingDirectory) to execResult - } - private fun createTestFiles(paths: List, warnPluginConfig: WarnPluginConfig): List { logDebug("Creating temp copy files of resources for WarnPlugin...") logTrace("Trying to create temp files for: $paths") @@ -230,6 +208,44 @@ class WarnPlugin( } } + private fun processExpectedWarnings( + generalConfig: GeneralConfig, + warnPluginConfig: WarnPluginConfig, + originalPaths: List, + copyPaths: List, + workingDirectory: Path, + ): WarningMap { + val expectedWarningsMap = collectExpectedWarnings(generalConfig, warnPluginConfig, originalPaths, copyPaths, workingDirectory) + + if (expectedWarningsMap.isEmpty()) { + warnMissingExpectedWarnings(warnPluginConfig, generalConfig, originalPaths) + } + return expectedWarningsMap + } + + private fun processActualWarnings( + executionResult: ExecutionResult, + warnPluginConfig: WarnPluginConfig, + originalPaths: List, + workingDirectory: Path, + ): WarningMap { + val actualResult = if (warnPluginConfig.actualWarningsFormat == ActualWarningsFormat.SARIF && + warnPluginConfig.actualWarningsFileName != null) { + // in this case, after tool execution, there was created sarif report, extract warnings from it, + // not from stdout + val sarif = calculatePathToSarifFile( + sarifFileName = warnPluginConfig.actualWarningsFileName, + anchorTestFilePath = originalPaths.first() + ) + ExecutionResult(executionResult.code, fs.readLines(sarif), executionResult.stderr) + } else { + // Note: this case is applicable both for PLAIN and SARIF mode, but in the latter case, we suppose, + // that sarif report is provided in stdout of tool + executionResult + } + return collectActualWarningsWithLineNumbers(actualResult, warnPluginConfig, workingDirectory) + } + private fun failTestResult( paths: List, ex: Exception, @@ -317,7 +333,7 @@ class WarnPlugin( .groupBy { it.fileName } .mapValues { (_, warning) -> warning.sortedBy { it.message } } } catch (e: Exception) { - throw SarifParsingException("We failed to parse sarif. Check the your tool generation of sarif report, cause: ${e.message}", e.cause) + throw SarifParsingException("Failed to parse sarif. Check the your tool generation of sarif report, cause: ${e.message}", e.cause) } else -> result.stdout.mapNotNull { diff --git a/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPluginConfig.kt b/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPluginConfig.kt index 49181ce17..b327f45e4 100644 --- a/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPluginConfig.kt +++ b/save-plugins/warn-plugin/src/commonMain/kotlin/com/saveourtool/save/plugin/warn/WarnPluginConfig.kt @@ -161,8 +161,12 @@ data class WarnPluginConfig( requireValidPatternForRegexInWarning() val expectedWarningsFormat = expectedWarningsFormat ?: ExpectedWarningsFormat.IN_PLACE + val expectedWarningsFileName = expectedWarningsFileName ?: "save-warnings-expected.sarif" + val actualWarningsFormat = actualWarningsFormat ?: ActualWarningsFormat.PLAIN - val expectedWarningsFileName = expectedWarningsFileName ?: "save-warnings.sarif" + // it could be null even, if actualWarningsFormat = sarif: in this case, we suppose, that tool will print + // sarif report into stdout + val actualWarningsFileName = actualWarningsFileName val newWarningTextHasLine = warningTextHasLine ?: true val newWarningTextHasColumn = warningTextHasColumn ?: true