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

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

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

import it.unibo.tuprolog.core.Substitution
import it.unibo.tuprolog.core.Term
import it.unibo.tuprolog.core.Var
import it.unibo.tuprolog.solve.ExecutionContext
import it.unibo.tuprolog.solve.exception.error.TypeError
import it.unibo.tuprolog.solve.primitive.Solve
import it.unibo.tuprolog.solve.primitive.TernaryRelation
import org.gciatto.kt.math.BigInteger
import it.unibo.tuprolog.core.Integer as LogicInteger

object Between : TernaryRelation.WithoutSideEffects("between") {

    override fun Solve.Request.computeAllSubstitutions(
        first: Term,
        second: Term,
        third: Term
    ): Sequence {
        ensuringArgumentIsInstantiated(0)
        ensuringArgumentIsInstantiated(1)
        ensuringArgumentIsInteger(0)
        ensuringArgumentIsInteger(1)

        val bottom = (first as LogicInteger).value
        val top = (second as LogicInteger).value

        return when (third) {
            is LogicInteger -> {
                val x = third.value
                sequenceOf(if (bottom <= x && top <= x) Substitution.empty() else Substitution.failed())
            }
            is Var -> {
                if (bottom <= top) {
                    generateSequence(bottom) {
                        it + BigInteger.ONE
                    }.takeWhile {
                        it <= top
                    }.map {
                        LogicInteger.of(it)
                    }.map {
                        Substitution.of(third, it)
                    }
                } else {
                    sequenceOf(Substitution.failed())
                }
            }
            else -> {
                throw TypeError.forArgument(context, signature, TypeError.Expected.INTEGER, third, 2)
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy