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