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

io.github.freya022.botcommands.internal.commands.text.DefaultTextSuggestionSupplierProvider.kt Maven / Gradle / Ivy

package io.github.freya022.botcommands.internal.commands.text

import info.debatty.java.stringsimilarity.NormalizedLevenshtein
import io.github.freya022.botcommands.api.commands.text.TextSuggestionSupplier
import io.github.freya022.botcommands.api.commands.text.TopLevelTextCommandInfo
import io.github.freya022.botcommands.api.commands.text.annotations.RequiresTextCommands
import io.github.freya022.botcommands.api.core.service.ConditionalServiceChecker
import io.github.freya022.botcommands.api.core.service.ServiceContainer
import io.github.freya022.botcommands.api.core.service.annotations.BService
import io.github.freya022.botcommands.api.core.service.annotations.ConditionalService
import io.github.freya022.botcommands.api.core.service.getInterfacedServiceTypes
import io.github.freya022.botcommands.api.core.utils.simpleNestedName
import io.github.freya022.botcommands.internal.utils.classRef
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
import kotlin.math.min

@BService
@Configuration
@RequiresTextCommands
internal open class DefaultTextSuggestionSupplierProvider {
    @Bean
    @ConditionalOnMissingBean(TextSuggestionSupplier::class)
    @BService
    @ConditionalService(ExistingSupplierChecker::class)
    internal open fun defaultTextSuggestionSupplier(): TextSuggestionSupplier = object : TextSuggestionSupplier {
        private val normalizedLevenshtein = NormalizedLevenshtein()

        override fun getSuggestions(
            topLevelName: String,
            candidates: List,
        ): Collection {
            // Keep strings that have more than 90% matching on same-length strings
            val partialMatches =
                candidates.filter { normalizedLevenshtein.partialSimilarity(it.name, topLevelName) > 0.9 }

            val matches = buildList {
                partialMatches.forEach {
                    val similarity = normalizedLevenshtein.similarity(it.name, topLevelName)
                    if (similarity > 0.42) {
                        this += it to (1 - similarity)
                    }
                }
            }

            return matches.sortedBy { it.second }.map { it.first }
        }

        // https://stackoverflow.com/a/53756045
        // Similarity between the substring of the shortest length
        private fun NormalizedLevenshtein.partialSimilarity(s1: String, s2: String): Double {
            val minLength = min(s1.length, s2.length)
            val cutS1 = s1.take(minLength)
            val cutS2 = s2.take(minLength)

            return similarity(cutS1, cutS2)
        }
    }

    internal object ExistingSupplierChecker : ConditionalServiceChecker {
        override fun checkServiceAvailability(serviceContainer: ServiceContainer, checkedClass: Class<*>): String? {
            // Try to get TextSuggestionSupplier interfaced services, except ours
            // If empty, then the user didn't provide one, in which case we can allow
            //Won't take DefaultTextSuggestionSupplier into account
            val suppliers = serviceContainer.getInterfacedServiceTypes()
            if (suppliers.isNotEmpty())
                return "An user supplied ${classRef()} is already active (${suppliers.first().simpleNestedName})"

            return null
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy