application.VirtualServiceCommand.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of specmatic-executable Show documentation
Show all versions of specmatic-executable Show documentation
Command-line standalone executable jar for Specmatic
package application
import io.specmatic.core.Configuration
import io.specmatic.core.Configuration.Companion.DEFAULT_HTTP_STUB_HOST
import io.specmatic.core.Configuration.Companion.DEFAULT_HTTP_STUB_PORT
import io.specmatic.core.DEFAULT_WORKING_DIRECTORY
import io.specmatic.core.Feature
import io.specmatic.core.log.StringLog
import io.specmatic.core.log.consoleLog
import io.specmatic.core.log.logger
import io.specmatic.core.utilities.*
import io.specmatic.mock.ScenarioStub
import io.specmatic.stub.stateful.StatefulHttpStub
import picocli.CommandLine.Command
import picocli.CommandLine.Option
import java.util.concurrent.Callable
import java.util.concurrent.CountDownLatch
@Command(
name = "virtual-service",
mixinStandardHelpOptions = true,
description = ["Start a stateful virtual service with contract"]
)
class VirtualServiceCommand : Callable {
@Option(names = ["--host"], description = ["Host for the virtual service"], defaultValue = DEFAULT_HTTP_STUB_HOST)
lateinit var host: String
@Option(names = ["--port"], description = ["Port for the virtual service"], defaultValue = DEFAULT_HTTP_STUB_PORT)
var port: Int = 0
@Option(names = ["--examples"], description = ["Directories containing JSON examples"], required = false)
var exampleDirs: List = mutableListOf()
private val stubLoaderEngine = StubLoaderEngine()
private var server: StatefulHttpStub? = null
private val latch = CountDownLatch(1)
private val newLine = System.lineSeparator()
override fun call(): Int {
setup()
try {
startServer()
} catch(e: Exception) {
logger.log("An error occurred while starting the virtual service: ${e.message}")
return 1
}
return 0
}
private fun setup() {
exitIfContractPathsDoNotExist()
addShutdownHook()
}
private fun exitIfContractPathsDoNotExist() {
val contractPaths = contractStubPaths(Configuration.configFilePath).map { it.path }
exitIfAnyDoNotExist("The following specifications do not exist", contractPaths)
}
private fun addShutdownHook() {
Runtime.getRuntime().addShutdownHook(object : Thread() {
override fun run() {
try {
latch.countDown()
consoleLog(StringLog("Shutting down the virtual service"))
server?.close()
} catch (e: InterruptedException) {
currentThread().interrupt()
}
}
})
}
private fun stubContractPathData(): List {
return contractFilePathsFrom(Configuration.configFilePath, DEFAULT_WORKING_DIRECTORY) {
source -> source.stubContracts
}
}
private fun startServer() {
val stubData: List>> = stubLoaderEngine.loadStubs(
stubContractPathData(),
exampleDirs,
Configuration.configFilePath,
false
)
server = StatefulHttpStub(
host,
port,
stubData.map { it.first },
Configuration.configFilePath,
stubData.flatMap { it.second }.also { it.logExamplesCachedAsSeedData() }
)
logger.log("Virtual service is running on ${consolePrintableURL(host, port)}. Ctrl + C to stop.")
latch.await()
}
private fun List.logExamplesCachedAsSeedData() {
logger.log("${newLine}Injecting the data read from the following stub files into the stub server's state..".prependIndent(" "))
this.forEach { logger.log(it.filePath.orEmpty().prependIndent(" ")) }
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy