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

com.lsd.core.LsdContext.kt Maven / Gradle / Ivy

There is a newer version: 7.0.56
Show newest version
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