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

com.skillw.pouvoir.internal.manager.ScriptManagerImpl.kt Maven / Gradle / Ivy

package com.skillw.pouvoir.internal.manager

import com.skillw.pouvoir.Pouvoir
import com.skillw.pouvoir.api.manager.sub.script.CompileManager.Companion.compileScript
import com.skillw.pouvoir.api.manager.sub.script.ScriptEngineManager.Companion.searchEngine
import com.skillw.pouvoir.api.manager.sub.script.ScriptManager
import com.skillw.pouvoir.api.plugin.ManagerTime
import com.skillw.pouvoir.api.plugin.map.MultiExecMap
import com.skillw.pouvoir.api.plugin.map.SingleExecMap
import com.skillw.pouvoir.api.script.PouFileCompiledScript
import com.skillw.pouvoir.internal.core.script.javascript.PouJavaScriptEngine
import com.skillw.pouvoir.internal.manager.PouConfig.debug
import com.skillw.pouvoir.util.legacy.mirrorNow
import com.skillw.pouvoir.util.listSubFiles
import com.skillw.pouvoir.util.safe
import taboolib.common.platform.ProxyCommandSender
import taboolib.common.platform.function.console
import taboolib.common.platform.function.getDataFolder
import taboolib.common5.FileWatcher
import taboolib.module.chat.colored
import taboolib.module.lang.asLangText
import taboolib.module.lang.sendLang
import java.io.File
import java.io.FileNotFoundException
import kotlin.time.ExperimentalTime
import kotlin.time.TimeSource

internal object ScriptManagerImpl : ScriptManager() {
    override val key = "ScriptManager"
    override val priority: Int = 8
    override val subPouvoir = Pouvoir


    private val dirs = HashSet()

    private val watcher by lazy {
        dirs.runCatching { }
        FileWatcher()
    }

    override fun onLoad() {
        call(ManagerTime.BEFORE_LOAD)
        dirs += File(getDataFolder(), "scripts")
        call(ManagerTime.LOAD)
    }

    override fun onEnable() {
        reloadScripts()
        call(ManagerTime.BEFORE_ENABLE)
        call(ManagerTime.ENABLE)
    }

    override fun onActive() {
        call(ManagerTime.BEFORE_ACTIVE)
        call(ManagerTime.ACTIVE)
    }

    override fun onReload() {
        call(ManagerTime.BEFORE_RELOAD)
        reloadScripts()
        call(ManagerTime.RELOAD)
    }

    override fun onDisable() {
        call(ManagerTime.BEFORE_DISABLE)
        call(ManagerTime.DISABLE)
    }

    private fun reloadScripts() {
        clear();dirs.forEach { addScript(it) }
    }

    override fun addScript(file: File) {
        if (file.isDirectory) {
            addScriptDir(file)
            return
        }
        safe { file.compileScript()?.apply { register() } }
        if (watcher.hasListener(file)) return
        watcher.addSimpleListener(file) {
            addScript(file)
        }
    }

    override fun addScriptDir(file: File) {
        dirs.add(file)
        file.listSubFiles().filter {
            it.searchEngine() != null
        }.forEach {
            addScript(it)
        }
    }

    override fun clear() {
        Pouvoir.scriptManager.values.forEach {
            it.delete()
        }
        super.clear()
    }

    override fun  invoke(
        pathWithFunction: String, arguments: Map,
        sender: ProxyCommandSender?, vararg parameters: Any?,
    ): T? {
        val splits = pathWithFunction.split("::")
        if (splits.size < 2) {
            console().sendLang("script-invoke-wrong-format", pathWithFunction)
        }
        val path = splits[0]
        val function = splits[1]
        return invoke(path, function, arguments, sender, *parameters)
    }

    override fun search(
        path: String,
        sender: ProxyCommandSender?,
    ): PouFileCompiledScript? {
        try {
            if (containsKey(path)) return this[path]
            this.keys
                .firstOrNull { it.endsWith(path) }
                ?.let { script ->
                    return this[script]
                }
            val scriptFile = File(getDataFolder(), path)
            if (!scriptFile.exists() || scriptFile.isDirectory) throw FileNotFoundException()
            addScript(scriptFile)
            return this[path]
        } catch (e: FileNotFoundException) {
            sender?.sendLang("script-file-not-found", path)
        }
        return null
    }

    override fun  invoke(
        path: String,
        function: String,
        arguments: Map,
        sender: ProxyCommandSender?,
        vararg parameters: Any?,
    ): T? {
        val script = search(path, sender) ?: return null
        return invoke(script, function, arguments, sender, *parameters)
    }


    override fun  evalJs(
        script: String,
        arguments: Map,
        sender: ProxyCommandSender?,
    ): T? {
        val hash = script.hashCode()
        debug {
            sender?.apply {
                sendLang("script-invoking-info", hash, "EvalScript")
                sendLang("script-invoking-arguments")
                sendMessage("&3$arguments".colored())
            }
        }
        val start = System.currentTimeMillis()
        val compiledScript = Pouvoir.compileManager.compile(script)
        val result = PouJavaScriptEngine.bridge.invoke(compiledScript, "main", arguments = arguments)
        val end = System.currentTimeMillis()
        debug {
            sender?.sendLang(
                "command-script-invoke-end",
                hash,
                result.run { if (this is Unit) console().asLangText("kotlin-unit") else toString() },
                (end - start)
            )
        }
        return result as? T?
    }

    @OptIn(ExperimentalTime::class)
    override fun  invoke(
        script: PouFileCompiledScript,
        function: String,
        arguments: Map,
        sender: ProxyCommandSender?,
        vararg parameters: Any?,
    ): T? {
        debug {
            sender?.apply {
                sendLang("script-invoking-info", script.key, function)
                sendLang("script-invoking-arguments")
                sendMessage("&3${arguments}".colored())
                sendLang("script-invoking-parameters")
                sendMessage("&3${parameters.map { it.toString() }}".colored())
            }
        }
        val mark = TimeSource.Monotonic.markNow()
        val result =
            runCatching {
                mirrorNow("${script.key}::$function") {
                    script.invoke(
                        function,
                        arguments,
                        *parameters
                    )
                }
            }.run {
                if (isSuccess) getOrNull()
                else {
                    console().sendLang("script-invoking-error", script.key, function)
                    exceptionOrNull()?.printStackTrace()
                }
            }
        val duration = mark.elapsedNow()
        debug {
            sender?.sendLang(
                "command-script-invoke-end",
                script.key,
                result.run { if (this is Unit) console().asLangText("kotlin-unit") else toString() },
                duration
            )
        }
        return result as? T?
    }


    private val execMap = MultiExecMap()

    override fun addExec(managerTime: ManagerTime, key: String, exec: () -> Unit) {
        execMap.computeIfAbsent(managerTime.key.lowercase()) { SingleExecMap() }[key.lowercase()] = exec
    }

    private fun call(managerTime: ManagerTime) {
        execMap.run(managerTime.key.lowercase())
    }

    override fun removeExec(managerTime: ManagerTime, key: String) {
        execMap[managerTime.key.lowercase()]?.remove(key.lowercase())
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy