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

org.jetbrains.kotlinx.jupyter.util.AcceptanceRules.kt Maven / Gradle / Ivy

package org.jetbrains.kotlinx.jupyter.util

import kotlinx.serialization.Serializable
import org.jetbrains.kotlinx.jupyter.api.TypeName
import org.jetbrains.kotlinx.jupyter.api.libraries.VariablesSubstitutionAware

/**
 * Acceptance rule either says it accepts an object or delegates it to some other rule returning null
 */
fun interface AcceptanceRule {
    fun accepts(obj: T): Boolean?
}

/**
 * Acceptance rule that has only two answers: yes/no (depending on the [acceptsFlag]) and "don't know"
 */
interface FlagAcceptanceRule : AcceptanceRule {
    val acceptsFlag: Boolean

    fun appliesTo(obj: T): Boolean

    override fun accepts(obj: T): Boolean? {
        return if (appliesTo(obj)) acceptsFlag else null
    }
}

/**
 * Acceptance rule for type names
 */
class NameAcceptanceRule(
    override val acceptsFlag: Boolean,
    private val appliesPredicate: (TypeName) -> Boolean,
) : FlagAcceptanceRule {
    override fun appliesTo(obj: TypeName): Boolean {
        return appliesPredicate(obj)
    }
}

/**
 * Acceptance rule for type names based on [pattern].
 * Pattern may consist of any characters and of 3 special combinations:
 * 1) `?` - any single character or no character
 * 2) `*` - any character sequence excluding dot (`.`)
 * 3) `**` - any character sequence
 *
 * For example, pattern `org.jetbrains.kotlin?.**.jupyter.*` matches following names:
 * - `org.jetbrains.kotlin.my.package.jupyter.Integration`
 * - `org.jetbrains.kotlinx.some_package.jupyter.SomeClass`
 *
 * It doesn't match name `org.jetbrains.kotlin.my.package.jupyter.integration.MyClass`
 */
@Serializable(PatternNameAcceptanceRuleSerializer::class)
class PatternNameAcceptanceRule(
    override val acceptsFlag: Boolean,
    val pattern: String,
) : FlagAcceptanceRule, VariablesSubstitutionAware {
    private val regex by lazy {
        buildString {
            var i = 0
            while (i < pattern.length) {
                val c = pattern[i]
                val nextC = pattern.getOrNull(i + 1)

                when (c) {
                    '.' -> append("\\.")
                    '*' ->
                        when (nextC) {
                            '*' -> {
                                append(".*")
                                ++i
                            }
                            else -> append("[^.]*")
                        }
                    '?' -> append(".?")
                    '[', ']', '(', ')', '{', '}', '\\', '$', '^', '+', '|' -> {
                        append('\\')
                        append(c)
                    }
                    else -> append(c)
                }
                ++i
            }
        }.toRegex()
    }

    override fun appliesTo(obj: TypeName): Boolean {
        return regex.matches(obj)
    }

    override fun replaceVariables(mapping: Map): PatternNameAcceptanceRule {
        val newPattern = replaceVariables(pattern, mapping)
        if (pattern == newPattern) return this
        return PatternNameAcceptanceRule(acceptsFlag, newPattern)
    }
}

/**
 * List of acceptance rules:
 * 1) accepts [obj] if latest not-null acceptance result is `true`
 * 2) doesn't accept [obj] if latest not-null acceptance result is `false`
 * 3) returns `null` if all acceptance results are `null` or the iterable is empty
 */
fun  Iterable>.accepts(obj: T): Boolean? {
    return unionAcceptance(map { it.accepts(obj) })
}

fun unionAcceptance(results: Iterable): Boolean? {
    return results.filterNotNull().lastOrNull()
}

fun unionAcceptance(vararg result: Boolean?): Boolean? {
    return unionAcceptance(result.toList())
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy