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

it.unibo.alchemist.boundary.extractors.MoleculeReader.kt Maven / Gradle / Ivy

There is a newer version: 36.0.11
Show newest version
/*
 * Copyright (C) 2010-2023, Danilo Pianini and contributors
 * listed, for each module, in the respective subproject's build.gradle.kts file.
 *
 * This file is part of Alchemist, and is distributed under the terms of the
 * GNU General Public License, with a linking exception,
 * as described in the file LICENSE in the Alchemist distribution's top directory.
 */

package it.unibo.alchemist.boundary.extractors

import it.unibo.alchemist.boundary.ExportFilter
import it.unibo.alchemist.model.Actionable
import it.unibo.alchemist.model.Environment
import it.unibo.alchemist.model.Incarnation
import it.unibo.alchemist.model.Molecule
import it.unibo.alchemist.model.Node
import it.unibo.alchemist.model.Time
import it.unibo.alchemist.util.StatUtil
import org.apache.commons.math3.stat.descriptive.UnivariateStatistic
import kotlin.math.min

/**
 * Reads the value of a molecule and logs it.
 *
 * @param moleculeName
 *            the target molecule
 * @param property
 *            the target property
 * @param incarnation
 *            the target incarnation
 * @param filter
 *            the [FilteringPolicy] to use
 * @param aggregatorNames
 *            the names of the [UnivariateStatistic] to use for
 *            aggregating data. If an empty list is passed, then the values
 *            will be logged indipendently for each node.
 */
class MoleculeReader @JvmOverloads constructor(
    moleculeName: String,
    private val property: String?,
    private val incarnation: Incarnation<*, *>,
    private val filter: ExportFilter,
    aggregatorNames: List,
    precision: Int? = null,
) : AbstractDoubleExporter(precision) {

    private companion object {
        private const val SHORT_NAME_MAX_LENGTH = 5
    }

    private val molecule: Molecule = incarnation.createMolecule(moleculeName)

    private val aggregators: Map = aggregatorNames
        .associateWith { StatUtil.makeUnivariateStatistic(it) }
        .filter { it.value.isPresent }
        .map { it.key to it.value.get() }
        .toMap()

    private val propertyText =
        if (property.isNullOrEmpty()) {
            ""
        } else {
            property.replace("[^\\d\\w]*".toRegex(), "")
        }

    private val shortProp = propertyText.takeIf(String::isEmpty)
        ?: "${propertyText.substring(0.. = aggregators.keys.takeIf { it.isNotEmpty() }
        ?.map { "$singleColumnName[$it]" }
        ?: listOf("$singleColumnName@node-id")

    override fun  extractData(
        environment: Environment,
        reaction: Actionable?,
        time: Time,
        step: Long,
    ): Map {
        fun Node.extractData() = environment.incarnation.getProperty(this, molecule, property)
        return when {
            aggregators.isEmpty() -> environment.nodes.asSequence().map { node ->
                "$singleColumnName@${node.id}" to node.extractData()
            }.toMap()
            else -> {
                val filtered = environment.nodes.flatMap { filter.apply(it.extractData()) }.toDoubleArray()
                aggregators.map { (aggregatorName, aggregator) ->
                    "$singleColumnName[$aggregatorName]" to aggregator.evaluate(filtered)
                }.toMap()
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy