All Downloads are FREE. Search and download functionalities are using the official Maven repository.

application.CompatibleCommand.kt Maven / Gradle / Ivy

There is a newer version: 2.2.0
Show newest version
package application

import io.specmatic.conversions.OpenApiSpecification
import io.specmatic.conversions.wsdlContentToFeature
import io.specmatic.core.*
import io.specmatic.core.git.GitCommand
import io.specmatic.core.git.NonZeroExitError
import io.specmatic.core.git.SystemGit
import io.specmatic.core.log.CompositePrinter
import io.specmatic.core.log.Verbose
import io.specmatic.core.log.logger
import io.specmatic.core.pattern.ContractException
import io.specmatic.core.utilities.exceptionCauseMessage
import io.specmatic.stub.hasOpenApiFileExtension
import io.specmatic.test.ResultAssert
import org.junit.jupiter.api.DynamicTest
import org.junit.jupiter.api.TestFactory
import org.junit.platform.engine.discovery.DiscoverySelectors
import org.junit.platform.launcher.Launcher
import org.junit.platform.launcher.LauncherDiscoveryRequest
import org.junit.platform.launcher.core.LauncherDiscoveryRequestBuilder
import org.junit.platform.reporting.legacy.xml.LegacyXmlReportGeneratingListener
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import org.springframework.stereotype.Component
import picocli.CommandLine
import picocli.CommandLine.*
import java.io.File
import java.io.FileNotFoundException
import java.io.PrintWriter
import java.nio.file.Paths
import java.util.concurrent.Callable

@Configuration
open class SystemObjects {
    @Bean
    open fun getSystemGit(): GitCommand {
        return SystemGit()
    }
}

open class JUnitBackwardCompatibilityTestRunner {
    companion object {
        fun outcome(): Outcome {
            return Outcome(results.fold(Results()) { acc, results -> acc.plus(results) }.distinct())
        }

        var tests: List = emptyList()
        var results: MutableList = mutableListOf()
    }

    @TestFactory
    fun backwardCompatibilityTest(): Collection {
        return tests.map { test ->
            DynamicTest.dynamicTest(test.name) {
                val testResults = Results(test.execute())

                results.add(testResults)
                ResultAssert.assertThat(withoutFluff(testResults).distinct().toResultIfAny()).isSuccess()
            }
        }
    }

    private fun withoutFluff(testResults: Results): Results {
        val resultsFluff1 = testResults.withoutFluff(1)

        return when {
            resultsFluff1.hasResults() -> resultsFluff1
            else -> testResults
        }
    }
}

@Component
@Command(name = "git",
        mixinStandardHelpOptions = true,
        description = ["Checks backward compatibility of a contract in a git repository"])
class GitCompatibleCommand : Callable {
    @Autowired
    lateinit var gitCommand: GitCommand

    @Autowired
    lateinit var fileOperations: FileOperations

    @Autowired
    lateinit var junitLauncher: Launcher

    @Command(name = "file", description = ["Compare file in working tree against HEAD"])
    fun file(@Parameters(paramLabel = "contractPath", defaultValue = ".") inputContractPath: String,
             @Option(names = ["--junitReportDir"], required = false, defaultValue = "") junitReportDirName: String,
             @Option(names = ["--debug"], required = false, defaultValue = "false") verbose: Boolean): Int {
        if(verbose)
            logger = Verbose(CompositePrinter())

        if(!inputContractPath.isContractFile() && !hasOpenApiFileExtension(inputContractPath) && !File(inputContractPath).isDirectory) {
            logger.log(invalidContractExtensionMessage(inputContractPath))
            return 1
        }

        return try {
            backwardCompatibleOnFileOrDirectory(inputContractPath, fileOperations) { contractPath ->
                val testGenerationOutcome = generateFileBackwardCompatibilityTests(contractPath, fileOperations, gitCommand)

                testGenerationOutcome.onSuccess { tests ->
                    JUnitBackwardCompatibilityTestRunner.tests = tests

                    val request: LauncherDiscoveryRequest = LauncherDiscoveryRequestBuilder.request()
                        .selectors(DiscoverySelectors.selectClass(JUnitBackwardCompatibilityTestRunner::class.java))
                        .build()

                    junitLauncher.discover(request)

                    if (junitReportDirName.isNotBlank()) {
                        val reportListener = LegacyXmlReportGeneratingListener(
                            Paths.get(junitReportDirName),
                            PrintWriter(System.out, true)
                        )
                        junitLauncher.registerTestExecutionListeners(reportListener)
                    }

                    junitLauncher.execute(request)

                    JUnitBackwardCompatibilityTestRunner.outcome()
                }
            }
        } catch (e: Throwable) {
            logger.log(e)
            1
        }
    }

    @Command(name = "commits", description = ["Compare file in newer commit against older commit"])
    fun commits(@Parameters(paramLabel = "contractPath", defaultValue = ".") path: String,
                @Parameters(paramLabel = "newerCommit") newerCommit: String,
                @Parameters(paramLabel = "olderCommit") olderCommit: String,
                @Option(names = ["--junitReportDir"], required = false, defaultValue = "") junitReportDirName: String,
                @Option(names = ["--debug"], required = false, defaultValue = "false") verbose: Boolean): Int {
        if(verbose)
            logger = Verbose()

        return try {
            backwardCompatibleOnFileOrDirectory(path, fileOperations) { contractPath ->
                val testGenerationOutcome =
                    generateCommitBackwardCompatibleTests(contractPath, newerCommit, olderCommit, gitCommand)

                testGenerationOutcome.onSuccess { tests ->
                    JUnitBackwardCompatibilityTestRunner.tests = tests

                    val request: LauncherDiscoveryRequest = LauncherDiscoveryRequestBuilder.request()
                        .selectors(DiscoverySelectors.selectClass(JUnitBackwardCompatibilityTestRunner::class.java))
                        .build()

                    junitLauncher.discover(request)

                    if (junitReportDirName.isNotBlank()) {
                        val reportListener = LegacyXmlReportGeneratingListener(
                            Paths.get(junitReportDirName),
                            PrintWriter(System.out, true)
                        )
                        junitLauncher.registerTestExecutionListeners(reportListener)
                    }

                    junitLauncher.execute(request)

                    JUnitBackwardCompatibilityTestRunner.outcome()
                }
            }
        } catch (e: Throwable) {
            logger.log(e)
            1
        }
    }

    override fun call(): Int {
        CommandLine(GitCompatibleCommand()).usage(System.out)
        return 0
    }
}

@Command(name = "compatible",
        mixinStandardHelpOptions = true,
        description = ["Checks if the newer contract is backward compatible with the older one"],
        subcommands = [ GitCompatibleCommand::class ])
internal class CompatibleCommand : Callable {
    override fun call() {
        CommandLine(CompatibleCommand()).usage(System.out)
    }
}

interface BackwardCompatibilityScope {
    fun executeCheck(): Int
}

class FileBackwardCompatibilityScope(val path: String, val fn: (String) -> Outcome): BackwardCompatibilityScope {
    override fun executeCheck(): Int {
        val outcome: Outcome = fn(path)

        val output = checkCompatibility(outcome)

        println(output.message)

        return output.exitCode
    }
}

class DirectoryBackwardCompatibilityScope(val path: String, val fn: (String) -> Outcome): BackwardCompatibilityScope {
    override fun executeCheck(): Int {
        val file = File(path)
        val outputs = file.walkTopDown().filter {
            it.extension in CONTRACT_EXTENSIONS
        }.map {
            val results = fn(it.path)
            Triple(it.path, checkCompatibility(results), results)
        }.toList()

        return if (outputs.isEmpty()) {
            logger.log("No contract files were found")
            0
        } else {
            logger.log(outputs.joinToString("${System.lineSeparator()}${System.lineSeparator()}") { (path, output) ->
                """$path:${System.lineSeparator()}${output.message.prependIndent("  ")}"""
            })

            outputs.map { (_, output) -> output.exitCode }.find { it != 0 } ?: 0
        }
    }

}

private fun backwardCompatibleOnFileOrDirectory(
    path: String,
    fileOperations: FileOperations,
    fn: (String) -> Outcome
): Int {
    val scope = when {
        fileOperations.isFile(path) -> FileBackwardCompatibilityScope(path, fn)
        fileOperations.isDirectory(path) -> DirectoryBackwardCompatibilityScope(path, fn)
        else -> throw ContractException("$path was of an unexpected file type.")
    }

    return scope.executeCheck()
}

internal fun compatibilityReport(results: Results, resultMessage: String): String {
    val countsMessage = "Tests run: ${results.successCount + results.failureCount}, Passed: ${results.successCount}, Failed: ${results.failureCount}\n\n"
    val resultReport = results.report(PATH_NOT_RECOGNIZED_ERROR).trim().let {
        when {
            it.isNotEmpty() -> "$it\n\n"
            else -> it
        }
    }

    return "$countsMessage$resultReport$resultMessage".trim()
}

internal fun generateFileBackwardCompatibilityTests(
    contractPath: String,
    fileOperations: FileOperations,
    git: GitCommand): Outcome> {

    return try {
        logger.debug("Newer version of $contractPath")

        val newerFeature = parseContract(logger.debug(fileOperations.read(contractPath)), contractPath)
        val result = getOlderFeature(contractPath, git)

        result.onSuccess { olderFeature ->
            Outcome(generateBackwardCompatibilityTests(olderFeature, newerFeature))
        }
    } catch(e: NonZeroExitError) {
        Outcome(emptyList(), "Could not find $contractPath at HEAD")
    } catch(e: FileNotFoundException) {
        Outcome(emptyList(), "Could not find $contractPath on the file system")
    }
}

internal fun backwardCompatibleFile(
    contractPath: String,
    fileOperations: FileOperations,
    git: GitCommand
): Outcome {
    return try {
        logger.debug("Newer version of $contractPath")

        val newerFeature = parseContract(logger.debug(fileOperations.read(contractPath)), contractPath)
        val result = getOlderFeature(contractPath, git)

        result.onSuccess {
            Outcome(testBackwardCompatibility(it, newerFeature))
        }
    } catch(e: NonZeroExitError) {
        Outcome(Results(mutableListOf(Result.Success())), "Could not find $contractPath at HEAD")
    } catch(e: FileNotFoundException) {
        Outcome(Results(mutableListOf(Result.Success())), "Could not find $contractPath on the file system")
    }
}

internal fun backwardCompatibleCommit(
    contractPath: String,
    newerCommit: String,
    olderCommit: String,
    git: GitCommand,
): Outcome {
    val (gitRoot, relativeContractPath) = git.relativeGitPath(contractPath)

    val partial = getFileContentAtSpecifiedCommit(gitRoot)(relativeContractPath)(contractPath)

    return partial(newerCommit).onSuccess { newerGherkin ->
        val olderCommitOutcome = partial(olderCommit)

        when(olderCommitOutcome.result) {
            null -> Outcome(Results())
            else -> olderCommitOutcome.onSuccess { olderGherkin ->
                Outcome(testBackwardCompatibility(parseContract(olderGherkin, contractPath), parseContract(newerGherkin, contractPath)))
            }
        }
    }
}

internal fun generateCommitBackwardCompatibleTests(
    contractPath: String,
    newerCommit: String,
    olderCommit: String,
    git: GitCommand,
): Outcome> {
    val (gitRoot, relativeContractPath) = git.relativeGitPath(contractPath)

    val partial = getFileContentAtSpecifiedCommit(gitRoot)(relativeContractPath)(contractPath)

    return partial(newerCommit).onSuccess { newerGherkin ->
        val olderCommitOutcome = partial(olderCommit)

        when(olderCommitOutcome.result) {
            null -> Outcome(emptyList())
            else -> olderCommitOutcome.onSuccess { olderGherkin ->
                Outcome(generateBackwardCompatibilityTests(parseContract(olderGherkin, contractPath), parseContract(newerGherkin, contractPath)))
            }
        }
    }
}

internal fun parseContract(content: String, path: String): Feature {
    return when(val extension = File(path).extension) {
        in OPENAPI_FILE_EXTENSIONS -> OpenApiSpecification.fromYAML(content, path).toFeature()
        WSDL -> wsdlContentToFeature(content, path)
        in CONTRACT_EXTENSIONS -> parseGherkinStringToFeature(content, path)
        else -> throw unsupportedFileExtensionContractException(path, extension)
    }
}

internal fun getOlderFeature(contractPath: String, git: GitCommand): Outcome {
    if(!git.fileIsInGitDir(contractPath))
        return Outcome(null, "Older contract file must be provided, or the file must be in a git directory")

    val(contractGit, relativeContractPath) = git.relativeGitPath(contractPath)
    logger.debug("Older version of $contractPath")
    return Outcome(parseContract(logger.debug(contractGit.show("HEAD", relativeContractPath)), contractPath))
}

internal data class CompatibilityOutput(val exitCode: Int, val message: String)

internal fun compatibilityMessage(results: Outcome): CompatibilityOutput {
    return when {
        results.result == null -> CompatibilityOutput(1, results.errorMessage)
        results.result.hasFailures() -> CompatibilityOutput(1, compatibilityReport(results.result, "The newer contract is NOT backward compatible"))
        else -> CompatibilityOutput(0, results.errorMessage.ifEmpty { "The newer contract is backward compatible" })
    }
}

internal fun checkCompatibility(results: Outcome): CompatibilityOutput =
    try {
        compatibilityMessage(results)
    } catch(e: Throwable) {
        CompatibilityOutput(1, "Could not run backward compatibility check, got exception\n${exceptionCauseMessage(e)}")
    }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy