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

it.unibo.alchemist.boundary.graphql.schema.model.surrogates.EnvironmentSurrogate.kt Maven / Gradle / Ivy

There is a newer version: 36.0.12
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.graphql.schema.model.surrogates

import com.expediagroup.graphql.generator.annotations.GraphQLDescription
import com.expediagroup.graphql.generator.annotations.GraphQLIgnore
import it.unibo.alchemist.boundary.graphql.schema.util.NodeToPosMap
import it.unibo.alchemist.boundary.graphql.schema.util.toNodeToPosMap
import it.unibo.alchemist.model.Environment
import it.unibo.alchemist.model.Layer
import it.unibo.alchemist.model.Molecule
import it.unibo.alchemist.model.Position
import it.unibo.alchemist.model.times.DoubleTime
import kotlinx.coroutines.sync.Semaphore
import kotlin.jvm.optionals.getOrNull

/**
 * A surrogate for [Environment].
 * @param T the concentration type
 * @param P the position
 * @param dimensions the number of dimensions of this environment.
 */
@GraphQLDescription("The simulation environment")
data class EnvironmentSurrogate>(
    @GraphQLIgnore override val origin: Environment,
    val dimensions: Int = origin.dimensions,
) : GraphQLSurrogate>(origin) {
    /**
     * The [Layer]s of this environment associated with their corresponding [Molecule].
     * The first initialization is made upon all molecules contained inside this environment's nodes.
     * Subsequent updates should be made whenever the [Environment.addLayer] is called.
     */
    private val moleculeToLayer: Map?> =
        origin.nodes
            .map { it.contents.keys }
            .flatten()
            .distinct()
            .associateWith { origin.getLayer(it).getOrNull() }

    /**
     * The nodes inside this environment.
     * @return the nodes in this environment.
     */
    @GraphQLDescription("The nodes in this environment")
    fun nodes() = origin.nodes.map { NodeSurrogate(it) }

    /**
     * The layers inside this environment.
     * @return the layers in this environment.
     */
    @GraphQLDescription("The layers in this environment")
    fun layers() =
        origin.layers.map {
            it.toGraphQLLayerSurrogate { coordinates ->
                origin.makePosition(*coordinates.toTypedArray())
            }
        }

    /**
     * Returns the node with the given id.
     * @param id the id of the node
     */
    @GraphQLDescription("The node with the given id")
    fun nodeById(id: Int): NodeSurrogate = origin.getNodeByID(id).toGraphQLNodeSurrogate()

    /**
     * Returns a [NodeToPosMap] representing all nodes associated with their position.
     */
    @GraphQLDescription("A list of entries NodeId-Position")
    fun nodeToPos(): NodeToPosMap = origin.nodes.associate { it.id to origin.getPosition(it) }.toNodeToPosMap()

    /**
     * Returns the neighborhood of the node with the given id.
     *
     * @param nodeId the id of the node
     * @return the neighborhood of the node with the given id.
     */
    @GraphQLDescription("The neighborhood of the node with the given id")
    fun getNeighborhood(nodeId: Int): NeighborhoodSurrogate =
        origin.getNeighborhood(origin.getNodeByID(nodeId)).toGraphQLNeighborhoodSurrogate()

    /**
     * Clone the node associated with the given id to the specified position.
     *
     * @param nodeId the id of the node to clone
     * @param position the position where to clone the node
     * @return true if the node has been cloned, false otherwise
     */
    @GraphQLDescription("Clone the node associated with the given id to the specified position")
    suspend fun cloneNode(
        nodeId: Int,
        position: PositionInput,
        time: Double,
    ): NodeSurrogate? {
        val newNode = origin.getNodeByID(nodeId).cloneNode(DoubleTime(time))
        val mutex = Semaphore(1, 1)
        var isAdded = false
        origin.simulation.schedule {
            try {
                isAdded = origin.addNode(newNode, origin.makePosition(*position.coordinates.toTypedArray()))
            } finally {
                mutex.release()
            }
        }
        mutex.acquire()
        return if (isAdded) newNode.toGraphQLNodeSurrogate() else null
    }

    /**
     * Returns the [LayerSurrogate] associated with the molecule represented by the given [MoleculeInput].
     *
     * @param m the [MoleculeInput] object associated to the layer
     * @return the [LayerSurrogate] associated with the molecule represented by the given [MoleculeInput]
     */
    @GraphQLDescription("The layer associated with the molecule represented by the given MoleculeInput")
    fun getLayer(m: MoleculeInput): LayerSurrogate? =
        getLayerFromMoleculeInput(m)?.toGraphQLLayerSurrogate { coordinates ->
            origin.makePosition(*coordinates.toTypedArray())
        }

    /**
     * Returns the [Layer] associated with the molecule represented by the given [MoleculeInput].
     * NB: molecule is resolved with name matching.
     *
     * @param m the [MoleculeInput] object
     * @return the [Layer] associated with the molecule represented by the given [MoleculeInput]
     */
    private fun getLayerFromMoleculeInput(m: MoleculeInput) =
        moleculeToLayer.filterKeys { it.name == m.name }.values.firstOrNull()
}

/**
 * Converts an [Environment] to a [EnvironmentSurrogate].
 * @param T the concentration type
 * @param P the position
 * @return a [EnvironmentSurrogate] representing the given [Environment]
 */
fun > Environment.toGraphQLEnvironmentSurrogate() = EnvironmentSurrogate(this)




© 2015 - 2025 Weber Informatics LLC | Privacy Policy