application.BundleCommand.kt Maven / Gradle / Ivy
@file:JvmName("BundleCommand_Jvm")
package application
import io.specmatic.core.APPLICATION_NAME_LOWER_CASE
import io.specmatic.core.YAML
import io.specmatic.core.log.logger
import io.specmatic.core.utilities.ContractPathData
import io.specmatic.stub.customImplicitStubBase
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.stereotype.Component
import picocli.CommandLine
import java.io.File
import java.io.FileOutputStream
import java.util.concurrent.Callable
import java.util.zip.ZipEntry
import java.util.zip.ZipOutputStream
interface Bundle {
fun contractPathData(): List
fun ancillaryEntries(pathData: ContractPathData): List
fun configEntry(): List
val bundlePath: String
}
class StubBundle(bundlePath: String?, private val config: SpecmaticConfig, private val fileOperations: FileOperations) : Bundle {
override val bundlePath = bundlePath ?: ".${File.separator}bundle.zip"
override fun contractPathData(): List {
return config.contractStubPathData()
}
override fun ancillaryEntries(pathData: ContractPathData): List {
val customImplicitStubBaseEntries: List = customImplicitStubBase()?.let { zipperEntriesFromCustomImplicitBase(pathData, it) } ?: emptyList()
val defaultBaseEntries = zipperEntriesFromDefaultBase(pathData)
return deDup(defaultBaseEntries.plus(customImplicitStubBaseEntries))
}
private fun deDup(entries: List): List {
val hashMap = entries.fold(HashMap()) { hashMap, entry ->
hashMap[entry.path] = entry
hashMap
}
return hashMap.entries.map {
it.value
}
}
private fun zipperEntriesFromDefaultBase(pathData: ContractPathData): List {
val base = File(pathData.baseDir)
val stubDataDir = stubDataDirRelative(File(pathData.path))
val stubFiles = stubFilesIn(stubDataDir, fileOperations)
return stubFiles.map {
val relativeEntryPath = File(it).relativeTo(base)
ZipperEntry("${base.name}${File.separator}${relativeEntryPath.path}", fileOperations.readBytes(it))
}
}
private fun zipperEntriesFromCustomImplicitBase(
pathData: ContractPathData,
customImplicitStubBase: String
): List {
val base = File(pathData.baseDir)
val contractRelativePath = File(pathData.path).relativeTo(base)
val stubRelativePath = contractRelativePath.parent?.let {
"${contractRelativePath.parent}${File.separator}${contractRelativePath.nameWithoutExtension}_data"
} ?: "${contractRelativePath.nameWithoutExtension}_data"
val stubFiles: List> =
stubFilesIn(base, File(customImplicitStubBase), File(stubRelativePath))
return stubFiles.map { (virtualPath, actualPath) ->
val relativeEntryPath = File(virtualPath).relativeTo(base)
ZipperEntry("${base.name}${File.separator}${relativeEntryPath.path}", fileOperations.readBytes(actualPath))
}
}
override fun configEntry(): List = emptyList()
}
class TestBundle(bundlePath: String?, private val config: SpecmaticConfig, private val fileOperations: FileOperations) : Bundle {
override val bundlePath: String = bundlePath ?: ".${File.separator}test-bundle.zip"
override fun contractPathData(): List {
return config.contractTestPathData()
}
override fun ancillaryEntries(pathData: ContractPathData): List {
val base = File(pathData.baseDir)
return if(yamlExists(pathData.path)) {
val yamlFilePath = File(yamlFileName(pathData.path))
val yamlRelativePath = yamlFilePath.relativeTo(base).path
val yamlEntryName = "${base.name}${File.separator}$yamlRelativePath"
val yamlEntry = ZipperEntry(yamlEntryName, fileOperations.readBytes(yamlFilePath.path))
listOf(yamlEntry)
} else {
emptyList()
}
}
override fun configEntry(): List {
val configEntryName = File(config.configFilePath).name
val configContent = fileOperations.readBytes(config.configFilePath)
return listOf(ZipperEntry(configEntryName, configContent))
}
}
@CommandLine.Command(name = "bundle",
mixinStandardHelpOptions = true,
description = ["Generate a zip file of all stub contracts in $APPLICATION_NAME_LOWER_CASE.json"])
class BundleCommand : Callable {
@CommandLine.Option(names = ["--bundlePath"], description = ["Path in which to create the bundle"], required = false)
var bundlePath: String? = null
@CommandLine.Option(names = ["--test"], description = ["Create a bundle from of the test components"], required = false)
var testBundle: Boolean = false
@Autowired
lateinit var specmaticConfig: SpecmaticConfig
@Autowired
lateinit var zipper: Zipper
@Autowired
lateinit var fileOperations: FileOperations
var bundleOutputPath: String? = null
override fun call() {
createBundle(bundlePath, specmaticConfig, fileOperations, zipper, testBundle, bundleOutputPath)
}
}
private fun createBundle(
bundlePath: String?,
specmaticConfig: SpecmaticConfig,
fileOperations: FileOperations,
zipper: Zipper,
testBundle: Boolean,
bundleOutputPath: String?
) {
val bundle = when {
testBundle -> TestBundle(bundlePath, specmaticConfig, fileOperations)
else -> StubBundle(bundlePath, specmaticConfig, fileOperations)
}
val pathData = bundle.contractPathData()
val zipperEntries = pathData.flatMap { contractPathData ->
pathDataToZipperEntry(bundle, contractPathData, fileOperations)
}.plus(bundle.configEntry())
zipper.compress(bundleOutputPath ?: bundle.bundlePath, zipperEntries)
}
private fun yamlFileName(path: String): String = path.removeSuffix(".spec") + ".${YAML}"
private fun yamlExists(pathData: String): Boolean =
File(yamlFileName(pathData)).exists()
fun pathDataToZipperEntry(bundle: Bundle, pathData: ContractPathData, fileOperations: FileOperations): List {
val base = File(pathData.baseDir)
val contractFile = File(pathData.path)
val relativePath = contractFile.relativeTo(base).path
val zipEntryName = "${base.name}${File.separator}$relativePath"
logger.debug("Reading contract ${pathData.path} (Canonical path: ${File(pathData.path).canonicalPath})")
val contractEntry = ZipperEntry(zipEntryName, fileOperations.readBytes(pathData.path))
val ancillaryEntries = bundle.ancillaryEntries(pathData)
return listOf(contractEntry).plus(ancillaryEntries)
}
fun stubFilesIn(base: File, customImplicitStubBase: File, stubRelativePath: File): List> {
return base.resolve(customImplicitStubBase).resolve(stubRelativePath).listFiles()?.flatMap {
when {
it.extension == "json" -> {
val actualPath = it.path
val virtualPath = base.resolve(it.relativeTo(base.resolve(customImplicitStubBase))).path
listOf(Pair(virtualPath, actualPath))
}
it.isDirectory -> stubFilesIn(base, customImplicitStubBase, it.relativeTo(base.resolve(customImplicitStubBase)))
else -> emptyList()
}
} ?: emptyList()
}
fun stubFilesIn(stubDataDir: String, fileOperations: FileOperations): List =
fileOperations.files(stubDataDir).flatMap {
when {
fileOperations.isJSONFile(it) -> listOf(it.path)
it.isDirectory -> stubFilesIn(File(stubDataDir).resolve(it.name).path, fileOperations)
else -> emptyList()
}
}
fun stubDataDirRelative(path: File): String {
return "${path.parent}${File.separator}${path.nameWithoutExtension}_data"
}
@Component
class Zipper {
fun compress(zipFilePath: String, zipperEntries: List) {
logger.log("Writing contracts to $zipFilePath")
FileOutputStream(zipFilePath).use { zipFile ->
ZipOutputStream(zipFile).use { zipOut ->
for (zipperEntry in zipperEntries) {
val entry = ZipEntry(zipperEntry.path)
zipOut.putNextEntry(entry)
zipOut.write(zipperEntry.bytes)
}
}
}
}
}
data class ZipperEntry(val path: String, val bytes: ByteArray) {
override fun equals(other: Any?): Boolean {
if (this === other) return true
if (javaClass != other?.javaClass) return false
other as ZipperEntry
if (path != other.path) return false
if (!bytes.contentEquals(other.bytes)) return false
return true
}
override fun hashCode(): Int {
var result = path.hashCode()
result = 31 * result + bytes.contentHashCode()
return result
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy