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

commonMain.it.unibo.tuprolog.solve.stdlib.primitive.AbstractCollectionOf.kt Maven / Gradle / Ivy

package it.unibo.tuprolog.solve.stdlib.primitive

import it.unibo.tuprolog.core.Struct
import it.unibo.tuprolog.core.Substitution
import it.unibo.tuprolog.core.Term
import it.unibo.tuprolog.core.Tuple
import it.unibo.tuprolog.core.Var
import it.unibo.tuprolog.solve.ExecutionContext
import it.unibo.tuprolog.solve.primitive.Solve
import kotlin.collections.Set
import it.unibo.tuprolog.core.List as LogicList

@Suppress("PrivatePropertyName", "ktlint:standard:property-naming")
abstract class AbstractCollectionOf(val name: String) : AbstractCollectingPrimitive(name) {
    private val VARS = Var.of("VARS")
    private val GOAL = Var.of("GOAL")
    private val APEX_TEMPLATE = Struct.of("^", VARS, GOAL)

    override fun Solve.Request.computeAllSubstitutions(
        first: Term,
        second: Term,
        third: Term,
    ): Sequence {
        ensuringArgumentIsInstantiated(1)
        ensuringArgumentIsCallable(1)
        val mgu = mgu(APEX_TEMPLATE, second)
        val uninteresting: Set =
            when (mgu) {
                is Substitution.Unifier ->
                    when (val vars = mgu[VARS]) {
                        is Tuple -> vars.toSequence().filterIsInstance().toSet()
                        is Var -> setOf(vars)
                        else -> emptySet()
                    }
                else -> emptySet()
            }
        val goal = if (mgu is Substitution.Unifier) mgu[GOAL]!! else second
        val solutions = computeIntermediateSolutions(goal.castToStruct())
        val free = goal.variables.toSet() - first.variables.toSet()
        val interesting = free - uninteresting
        val groups = solutions.groupBy { it.substitution.filter(interesting) }
        val nonPresentable = first.variables.toSet()
        return groups.asSequence().map { (sub, sols) ->
            val solValues =
                sols.map { first[it.substitution] }
                    .filterNot { it in nonPresentable }
                    .map { it.freshCopy() }
            sub + mgu(third, LogicList.of(processSolutions(solValues)))
        }
    }

    protected abstract fun processSolutions(list: List): Iterable
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy