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