com.lsd.core.LsdContext.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lsd-core Show documentation
Show all versions of lsd-core Show documentation
A library used to generate living sequence diagrams
package com.lsd.core
import com.lsd.core.builders.ScenarioBuilder
import com.lsd.core.builders.ScenarioModelBuilder.Companion.scenarioModelBuilder
import com.lsd.core.builders.SequenceDiagramGeneratorBuilder.Companion.sequenceDiagramGeneratorBuilder
import com.lsd.core.diagram.ComponentDiagramGenerator
import com.lsd.core.domain.*
import com.lsd.core.properties.LsdProperties
import com.lsd.core.properties.LsdProperties.DETERMINISTIC_IDS
import com.lsd.core.properties.LsdProperties.getBoolean
import com.lsd.core.report.*
import com.lsd.core.report.HtmlIndexWriter.writeToFile
import com.lsd.core.report.model.*
import java.io.File
import java.nio.file.Path
import java.time.Duration
open class LsdContext {
val idGenerator = IdGenerator(getBoolean(DETERMINISTIC_IDS))
val outputDirectory = File(LsdProperties[LsdProperties.OUTPUT_DIR])
private val htmlReportWriter = HtmlReportWriter(HtmlReportRenderer())
private val scenarios: MutableList = ArrayList()
private val reportFiles: MutableList = ArrayList()
private val participants = linkedMapOf()
private val includes = linkedSetOf()
private var currentScenario = ScenarioBuilder()
private val combinedEvents = linkedSetOf()
private val options = ReportOptions()
fun addParticipants(vararg participants: Participant) = addParticipants(participants.toList())
fun addParticipants(additionalParticipants: List) =
additionalParticipants.forEach { participants[it.componentName.normalisedName] = it }
fun includeFiles(additionalIncludes: Set) = includes.addAll(additionalIncludes)
fun addFact(key: String, value: String = "") = currentScenario.addFact(key, value)
/**
* Capture a sequence event for the current scenario
*
* @param events The events to be captured on the sequence diagram for the current scenario.
*/
open fun capture(vararg events: SequenceEvent) {
currentScenario.addAll(events)
combinedEvents.addAll(events)
}
@JvmOverloads
fun completeScenario(title: String, description: String? = "", status: Status = Status.SUCCESS) {
currentScenario.title(title)
currentScenario.description(description)
currentScenario.status(status)
scenarios.add(currentScenario.build())
currentScenario = ScenarioBuilder()
}
@JvmOverloads
fun completeReport(
title: String,
options: ReportOptions = this.options
): Path {
val report = buildReport(title, options)
return htmlReportWriter.writeToFile(
report = report,
outputDir = outputDirectory,
devMode = options.devMode
).also {
reportFiles.add(ReportFile(filename = it.fileName.toString(), title = report.title, status = report.status))
scenarios.clear()
currentScenario = ScenarioBuilder()
}
}
/**
* Generates a html report with a component diagram of all the events that have been seen up until this point.
* Note that the events will also be cleared after invoking this function.
*/
@JvmOverloads
fun completeComponentsReport(title: String, devMode: Boolean = options.devMode): Path {
return ComponentReportWriter.writeToFile(
content = renderComponentReport(title, devMode),
fileName = "components-report.html"
).also {
combinedEvents.clear()
}
}
private fun renderComponentReport(title: String, isDevMode: Boolean): String {
return ComponentDiagramGenerator(
idGenerator = idGenerator,
events = combinedEvents,
participants = participants.values.toSet()
).diagram()?.let {
ComponentReportRenderer().render(
model = Model(title = title, uml = it.uml, svg = it.svg),
devMode = isDevMode
)
} ?: ""
}
@JvmOverloads
fun renderReport(
title: String,
reportOptions: ReportOptions = options
): String =
htmlReportWriter.renderReport(
report = buildReport(
title = title,
options = reportOptions
),
devMode = reportOptions.devMode
)
fun createIndex(devMode: Boolean = options.devMode): Path = writeToFile(reportFiles, devMode)
/**
* This clears down the context - all scenarios, reports, events and ids will be reset.
*/
fun clear() {
idGenerator.reset()
scenarios.clear()
reportFiles.clear()
includes.clear()
currentScenario = ScenarioBuilder()
}
/**
* Clears the captured sequence events for the current scenario only.
*
* This may be useful for removing events captured during test-setup for example.
*/
fun clearScenarioEvents() {
currentScenario.clearEvents()
}
internal fun buildReport(title: String, options: ReportOptions): Report {
return Report(
title = title,
status = determineOverallStatus(scenarios),
showContentsMenu = scenarios.size > 1,
scenarios = scenarios
.map { scenario ->
val (sequenceDuration, sequenceDiagram) = sequenceDiagramWithDuration(
scenario,
options.maxEventsPerDiagram,
options.devMode
)
val (componentDuration, componentDiagram) = componentDiagramWithDuration(scenario)
val metrics = if (options.metricsEnabled) Metrics(
events = scenario.events,
sequenceDuration = sequenceDuration,
componentDuration = componentDuration
) else null
scenarioModelBuilder()
.id(idGenerator.next())
.title(scenario.title)
.status(scenario.status.toCssClass())
.description(scenario.description)
.facts(scenario.facts)
.metrics(metrics)
.dataHolders(
scenario.events
.filterIsInstance()
.map {
DataHolder(
id = it.id,
abbreviatedLabel = it.label.abbreviate(),
data = it.data
)
}
)
.sequenceDiagram(sequenceDiagram)
.componentDiagram(componentDiagram)
.build()
})
}
private fun sequenceDiagramWithDuration(
scenario: Scenario,
maxEventsPerDiagram: Int,
isDevMode: Boolean,
): Pair {
return timedResult {
sequenceDiagramGeneratorBuilder()
.idGenerator(idGenerator)
.events(scenario.events)
.participants(participants.values.toList())
.includes(includes.toList())
.build()
.diagram(maxEventsPerDiagram, isDevMode)
}
}
private fun componentDiagramWithDuration(scenario: Scenario): Pair {
return timedResult {
ComponentDiagramGenerator(
idGenerator = idGenerator,
events = scenario.events.toSet(),
participants = participants.values.toSet()
).diagram()
}
}
private fun determineOverallStatus(scenarios: List): String {
return scenarios.map { it.status }.sortedWith(compareBy {
when (it) {
Status.ERROR -> 0
Status.FAILURE -> 1
Status.SUCCESS -> 2
}
}).firstOrNull()?.toCssClass() ?: ""
}
companion object {
/**
* Convenience method to create a singleton instance.
* If you prefer to use other means to share a single instance or
* want to create multiple instances then use the default constructor instead.
*
* @return A singleton instance of LsdContext
*/
@JvmStatic
val instance = LsdContext()
}
}
fun Status.toCssClass(): String = when (this) {
Status.SUCCESS -> "success"
Status.ERROR -> "error"
Status.FAILURE -> "warn"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy