import { plugin as pluginApi, task, server, channel, constants, proxy } from '@ccms/api' import { Translate } from '@ccms/i18n' import { DefaultContainer as container, JSClass, Autowired } from '@ccms/container' import { interfaces, JSPlugin, Cmd, Tab, enable, Listener, disable } from '@ccms/plugin' import * as fs from '@ccms/common/dist/fs' import * as reflect from '@ccms/common/dist/reflect' import http from '@ccms/common/dist/http' let help = [ '§6========= §6[§aMiaoScriptPackageManager§6] 帮助 §aBy §bMiaoWoo §6=========', '§6/mspm §ainstall §e<插件名称> §6- §3安装仓库插件', '§6/mspm §aload §e<插件名称> §6- §3安装本地插件', '§6/mspm §cunload §e<插件名称> §6- §3卸载已安装插件', '§6/mspm §areload §e<插件名称> §6- §3重载已安装插件(无名称则重载自身)', '§6/mspm §cdelete §e<插件名称> §6- §3删除已安装插件', '§6/mspm §alist [i] §6- §3列出仓库插件[已安装的插件]', '§6/mspm §aupdate §e[插件名称] §6- §3更新插件(无名称则更新源)', '§6/mspm §aupgrade §e[插件名称|system] §6- §3升级插件/§4框架(§csystem§3)', '§6/mspm §arun §e[插件名称] §3 §6- §3运行JS代码', '§6/mspm §aprun §e §6- §3运行JS代码', '§6/mspm §adeploy §e<插件名称> §6- §3发布插件', '§6/mspm §crestart §6- §4重启MiaoScript脚本引擎' ] let langMap = { 'main.command.not.exists': '§4未知的子命令: §c{command}', 'main.command.help.tip': '§6请执行 §b/{command} §ahelp §6查看帮助!', 'main.command.no.permission': '§c你没有此命令的权限!', 'list.install.header': '§6当前 §bMiaoScript §6已安装下列插件:', 'list.install.body': `§6┌插件名称: §b{name} §6├版本: §a{version} §6├作者: §3{author} §6└来源: §c{from}`, 'list.header': '§6当前 §bMiaoScriptPackageCenter §6中存在下列插件:', 'list.body': `§6┌插件名称: §b{name} §6├版本: §a{version} §6├作者: §3{author} §6└更新时间: §9{updated_at}`, 'plugin.not.exists': '§6插件 §b{name} §c不存在!', 'plugin.unload.start': '§c开始卸载 §6插件 §b{name} §6版本 §3{version}!', 'plugin.unload.finish': '§6插件 §b{name} §6版本 §3{version} §a已卸载!', 'plugin.reload.start': '§a开始重载 §6插件 §b{name} §6版本 §3{version}!', 'plugin.reload.finish': '§6插件 §b{name} §6版本 §3{version} §a重载完成!', 'plugin.name.empty': '§c请输入插件名称!', 'cloud.update.finish': '§6成功从 §aMiaoScriptPackageCenter §6获取到 §a{length} §6个插件!', 'cloud.not.exists': '§6当前 §aMiaoScriptPackageCenter §c不存在 §a{name} §c插件!', 'cloud.update.exists': '§6插件 §b{name} §6版本 §3{old_version} §a发现更新 §3{new_version} §r{changelog}§6!', 'download.start': '§6开始下载插件: §b{name} §6版本 §3{version}', 'download.url': '§6插件下载地址: §b{url}', 'download.finish': '§6插件 §b{name} §6版本 §3{version} §a下载完毕 开始加载 ...', 'install.already': '§6插件 §b{name} §6版本 §3{version} §c已安装在服务器 §3更新请用 update 命令!', 'install.finish': '§6插件 §b{name} §6版本 §3{version} §a安装成功!', 'update.finish': '§6插件 §b{name} §6版本 §3{version} §a更新成功!', 'upgrade.confirm': '§6您正在尝试更新 §bMiaoScript §c核心 §6请执行 §b/mpm §aupgrade §cconfirm §6确认执行!', 'upgrade.start': '§6开始§a更新 §bMiaoScript §6核心 §c正在清理 node_modules 请稍候...', 'upgrade.failed': '§6尝试热更新 §bMiaoScript §c核心 §4失败! §6请重启服务器完成更新...', 'deploy.token.not.exists': '§4请先配置发布Token!', 'deploy.success': '§6插件 §b{name} §6版本 §3{version} §a发布成功! §6服务器返回: §a{msg}', 'deploy.fail': '§6插件 §b{name} §6版本 §3{version} §c发布失败! §6服务器返回: §c{msg}', 'prun.script': '§6指定插件:§b {name}', 'run.script': '§b运行脚本:§r {script}', 'run.result': '§a返回结果:§r {result}', 'run.noresult': '§4没有返回结果!', } let fallbackMap = langMap class FakeSender { private _proxy private name: string private plugin: MiaoScriptPackageManager constructor(name: string, plugin: MiaoScriptPackageManager, superclass: any) { this.name = name this.plugin = plugin let FakeSenderAdapter = Java.extend(superclass, { getName: () => this.name, isOp: () => true, hasPermission: () => true, sendMessage: (message) => this.sendMessage(message) }) this._proxy = new FakeSenderAdapter() } sendMessage(message) { return this.plugin.sendBungeeCordMessage(this.name, `§6[§3BPM§6][§a${this.plugin.serverName}§6] ${message}`) } getHandler() { return this._proxy } } class BukkitFakeSender extends FakeSender { constructor(name: string, plugin: MiaoScriptPackageManager) { super(name, plugin, Java.type('org.bukkit.command.CommandSender')) } } class SpongeFakeSender extends FakeSender { constructor(name: string, plugin: MiaoScriptPackageManager) { super(name, plugin, Java.type('org.spongepowered.api.command.CommandSource')) } sendMessage(message) { return super.sendMessage(message.content) } } @JSPlugin({ prefix: 'PM', version: '1.4.0', author: 'MiaoWoo', source: __filename }) export class MiaoScriptPackageManager extends interfaces.Plugin { @Autowired() private pluginManager: pluginApi.PluginManager @Autowired() private taskManager: task.TaskManager @Autowired(server.ServerType) private serverType: string @Autowired() private server: server.Server @Autowired(pluginApi.PluginFolder) private pluginFolder: string @Autowired() private channel: channel.Channel @Autowired() private bungee: proxy.BungeeCord @JSClass('com.google.common.io.ByteStreams') private ByteStreams: any private packageCache: any[] = []; private packageNameCache: string[] = []; private isBungeeCord = false public serverName: string private translate: Translate private channelOff: { off: () => void } private subCommandCache = [] load() { this.translate = new Translate({ langMap, fallbackMap }) this.updateRepo(this.server.getConsoleSender()) this.subCommandCache = Object.keys(this).filter(c => c.startsWith('cmd') && typeof this[c] == "function") } @enable({ servers: [constants.ServerType.Bukkit, constants.ServerType.Sponge] }) serverEnbale() { this.channelOff = this.channel?.listen(this, 'BungeeCord', (data) => { let input = this.ByteStreams.newDataInput(data) let subChannel = input.readUTF() switch (subChannel) { case "GetServer": this.isBungeeCord = true this.serverName = input.readUTF() break case "MiaoScriptPackageManager": this.readForward(input) break } }) let players = this.server.getOnlinePlayers() if (players.length) this.bungeeCordDetect(players[0]) } @disable({ servers: [constants.ServerType.Bukkit, constants.ServerType.Sponge] }) serverDisable() { this.channelOff?.off() } private bungeeCordDetect(player) { if (player) { this.bungee.for(player).getServer().send() } } private bungeeCordForward(player, command) { if (player) { this.bungee.for(player).forward("ALL", "MiaoScriptPackageManager", command).send() } } private readForward(input) { let message = JSON.parse(input.readUTF()) let fakeSender = this.getProxySender(message.sender) this.taskManager.create(() => this.main(fakeSender, message.command, message.args)).async().submit() this.logger.sender(fakeSender, `§6命令 §b/mspm ${message.args?.join?.(' ')} §a执行成功!`) } private getProxySender(name: string) { switch (this.serverType) { case constants.ServerType.Bukkit: return new BukkitFakeSender(name, this).getHandler() case constants.ServerType.Sponge: return new SpongeFakeSender(name, this).getHandler() default: return this.server.getConsoleSender() } } @Listener({ servers: [constants.ServerType.Bukkit] }) PlayerJoinEvent(event: org.bukkit.event.player.PlayerJoinEvent) { this.bungeeCordDetect(event.getPlayer()) if (event.getPlayer().isOp()) { this.updateRepo(event.getPlayer()) } } @Listener({ servers: [constants.ServerType.Sponge] }) ClientConnectionEvent$Join(event: org.spongepowered.api.event.network.ClientConnectionEvent.Join) { this.bungeeCordDetect(event.getTargetEntity()) if (event.getTargetEntity().hasPermission('ms.mpm.admin')) { this.updateRepo(event.getTargetEntity()) } } sendBungeeCordMessage(sender, message) { let players = this.server.getOnlinePlayers() if (players.length) { this.bungee.for(players[0]).message(sender, message).send() } } disable() { this.channelOff?.off() } @Cmd({ servers: [`!${constants.ServerType.Bungee}`] }) gmspm(sender: any, command: string, args: string[]) { if (!sender.hasPermission('mspm.admin')) { return this.i18n(sender, 'main.command.no.permission') } if (!this.isBungeeCord) return this.logger.sender(sender, '§c当前服务器尚未检测到 BungeeCord 链接...') this.taskManager.create(() => this.main(sender, command, args)).async().submit() this.bungeeCordForward(sender, { sender: sender.getName(), command, args }) this.logger.sender(sender, `§6[§3BPM§6][§a${this.serverName}§6] §6命令 §b/mspm ${args.join?.(' ')} §a发布成功!`) } @Cmd({ servers: [constants.ServerType.Bungee] }) bungeemspm(sender: any, command: string, args: string[]) { if (!sender.hasPermission('mspm.admin')) { return this.i18n(sender, 'main.command.no.permission') } this.taskManager.create(() => this.main(sender, command, args)).async().submit() } @Cmd({ servers: [`!${constants.ServerType.Bungee}`] }) mspm(sender: any, command: string, args: string[]) { if (!sender.hasPermission('mspm.admin')) { return this.i18n(sender, 'main.command.no.permission') } this.taskManager.create(() => this.main(sender, command, args)).async().submit() } i18n(sender: any, name: string, params?: any) { this.logger.sender(sender, this.translate.translate(name, params).split('\n')) } main(sender: any, command: string, args: string[]) { let cmdKey = 'cmd' + (args[0] || 'help') if (!this[cmdKey]) { this.i18n(sender, 'main.command.not.exists', { command: args[0] }) this.i18n(sender, 'main.command.help.tip', { command }) return } args.shift() this[cmdKey](sender, ...args) } cmdhelp(sender: any) { this.logger.sender(sender, help) } cmdinfo(sender: any) { this.logger.sender(sender, [ ]) } cmdsudo(sender: any, name: string, ...args: string[]) { this.taskManager.create(() => { let player = this.server.getPlayer(name) if (!player) { return this.logger.sender(sender, `§4玩家 ${player.getName()} 不在线或不存在!`) } let command = args.join(' ') this.server.dispatchCommand(player, command) this.logger.sender(sender, `§6玩家 §a${player.getName()} §6命令 §b${command} §a执行完成!`) }).submit() } cmdload(sender: any, name: string) { let pluginFile = fs.concat(__dirname + '', name) if (!fs.exists(pluginFile)) { this.i18n(sender, 'plugin.not.exists', { name: `${name}(${pluginFile})` }) return } this.pluginManager.loadFromFile(fs.file(pluginFile)) } cmdlist(sender: any, type: string = 'cloud') { if (type == "i" || type == "install") { this.i18n(sender, 'list.install.header') this.pluginManager.getPlugins().forEach((plugin) => { this.i18n(sender, 'list.install.body', { ...plugin.description, from: plugin.description.loadMetadata.type }) }) } else { this.i18n(sender, 'list.header') for (var pkgName in this.packageCache) { this.i18n(sender, 'list.body', this.packageCache[pkgName]) } } } cmdinstall(sender: any, name: string) { if (!name) { return this.i18n(sender, 'plugin.name.empty') } if (this.pluginManager.getPlugins().has(name)) { this.i18n(sender, 'install.already', { name, version: this.pluginManager.getPlugins().get(name).description.version }) return } this.download(sender, name, false, () => { let plugin = this.pluginManager.loadFromFile(fs.concat(root, this.pluginFolder, name + '.js')) this.i18n(sender, 'install.finish', { name, version: plugin.description.version }) }) } cmdupdate(sender: any, name: string) { if (name) { this.update(sender, name) } else { this.updateRepo(sender) } } cmdupgrade(sender: any, name: string, confirm: boolean) { if (name == "system") { if (!confirm) { return this.i18n(sender, 'upgrade.confirm') } let enginePath = fs.path(fs.file(root, 'node_modules')) if (enginePath.startsWith(root)) { try { this.i18n(sender, 'upgrade.start') base.delete(enginePath) // @ts-ignore require.setUpgradeMode?.(true) this.cmdrestart(sender) } catch (ex) { if (global.debug) { console.ex(ex) } if (fs.exists(enginePath)) { this.i18n(sender, 'upgrade.failed') fs.create(fs.file(root, 'upgrade')) } } } return } if (this.checkPlugin(sender, name)) { this.update(sender, name, () => this.reload(sender, name)) } } cmdunload(sender: any, name: string) { if (this.checkPlugin(sender, name)) { let plugin = this.pluginManager.getPlugins().get(name) this.i18n(sender, 'plugin.unload.start', { name, version: plugin.description.version }) this.pluginManager.disable(plugin) this.i18n(sender, 'plugin.unload.finish', { name, version: plugin.description.version }) } } cmddelete(sender: any, name: string) { if (this.checkPlugin(sender, name)) { let plugin = this.pluginManager.getPlugins().get(name) this.i18n(sender, 'plugin.delete.start', { name, version: plugin.description.version }) base.delete(plugin.description.source) this.i18n(sender, 'plugin.delete.finish', { name, version: plugin.description.version }) } } cmdreload(sender: any, name: string) { name = name || this.description.name this.reload(sender, name) } private reload(sender: any, name: string) { if (this.checkPlugin(sender, name)) { let plugin = this.pluginManager.getPlugins().get(name) this.i18n(sender, 'plugin.reload.start', { name, version: plugin.description.version }) this.pluginManager.reload(plugin) this.i18n(sender, 'plugin.reload.finish', { name, version: this.pluginManager.getPlugins().get(name).description.version }) } } checkPlugin(sender: any, name: string) { if (name && this.pluginManager.getPlugins().has(name)) { return true } this.i18n(sender, 'plugin.not.exists', { name }) return false } checkCloudPlugin(sender: any, name: string) { if (name && this.packageNameCache.includes(name)) { return true } this.i18n(sender, 'cloud.not.exists', { name }) return false } cmdrestart(sender: any) { this.taskManager.callSyncMethod(() => { if (this.serverType === "sponge") { this.server.dispatchConsoleCommand('sponge plugins reload') return } try { this.logger.sender(sender, '§6Reloading §3MiaoScript Engine...') ScriptEngineContextHolder.disableEngine() Packages.java.lang.System.gc() if (ScriptEngineContextHolder.loadEngine) { ScriptEngineContextHolder.loadEngine() } ScriptEngineContextHolder.enableEngine() this.logger.sender(sender, '§3MiaoScript Engine §6Reload §aSuccessful...') } catch (ex) { this.logger.sender(sender, "§3MiaoScript Engine §6Reload §cError! ERR: " + ex) this.logger.sender(sender, this.logger.stack(ex)) } }) } cmdrun(sender: any, ...args: any[]) { try { let script = args.join(' ') this.i18n(sender, 'run.script', { script }) let result = this.runCode(script, sender, this) this.i18n(sender, 'run.result', { result: result == undefined ? this.translate.translate('run.noresult') : result + '' }) } catch (ex) { this.logger.sender(sender, this.logger.stack(ex)) } } cmdprun(sender: any, name: string, ...args: any[]) { try { if (!this.pluginManager.getPlugins().has(name)) { return this.logger.sender(sender, `§4插件 §c${name} §4不存在!`) } let script = args.join(' ') this.i18n(sender, 'prun.script', { name }) this.i18n(sender, 'run.script', { script }) let result = this.runCode(script, sender, this.pluginManager.getPlugins().get(name)) this.i18n(sender, 'run.result', { result: result == undefined ? this.translate.translate('run.noresult') : typeof result == "string" ? result : JSON.stringify(result) }) } catch (ex) { this.logger.sender(sender, this.logger.stack(ex)) } } private runCode(code: string, sender: any, _this: any) { let paramNames = [ 'sender', 'reflect', 'container', 'pluginManager' ] let params = [ sender, reflect, container, this.pluginManager ] let tfunc = new Function( ...paramNames, `var api = require('@ccms/api'); if (this.serverType == "spring") { var dbm = container.get(api.database.DataBaseManager) var db = dbm.getMainDatabase() var df = base.getInstance().getAutowireCapableBeanFactory() } return eval(${JSON.stringify(code)});`) return tfunc.apply(_this, params) } cmddeploy(sender: any, name: any) { if (!process.env.AccessToken) { return this.i18n(sender, 'deploy.token.not.exists') } this.taskManager.create(() => { if (this.checkPlugin(sender, name)) { let plugin: pluginApi.Plugin = this.pluginManager.getPlugins().get(name) let result = http.post("http://ms.yumc.pw/api/plugin/deploy?access_token=" + process.env.AccessToken, { name, author: plugin.description.author, version: plugin.description.version, source: base.read((plugin.description.source || plugin.description.loadMetadata.file).toString()) }) this.i18n(sender, result.code == 200 ? 'deploy.success' : 'deploy.fail', { name, version: plugin.description.version, msg: result.msg }) } }).async().submit() } private update(sender: any, name: string, callback?: () => void) { if (this.checkCloudPlugin(sender, name)) { this.download(sender, name, true, () => { this.i18n(sender, 'update.finish', { name, version: this.packageCache[name].version }) callback?.() }) } } @Tab({ alias: ['gmspm', 'bungeemspm'] }) tabmspm(_sender: any, _command: any, args: string | any[]) { if (args.length === 1) { return ['list', 'install', 'update', 'load', 'unload', 'upgrade', 'reload', 'restart', 'run', 'prun', 'sudo', 'help', 'create', 'deploy'] } if (args.length > 1) { switch (args[0]) { case "list": return ["install", "cloud"] case "install": return this.packageNameCache case "upgrade": if (args.length == 2) return ["system", ...this.pluginManager.getPlugins().keys()] if (args.length == 3 && args[1] == "system") return ["confirm"] return [] case "prun": if (args.length == 2) return [...this.pluginManager.getPlugins().keys()] return [] case "sudo": if (args.length == 2) return Java.from(this.server.getOnlinePlayers()).map(p => p.getName()) return [] case "update": case "load": case "unload": case "reload": case "deploy": return [...this.pluginManager.getPlugins().keys()] } } } updateRepo(sender: any) { this.taskManager.create(() => { let result = http.get('http://ms.yumc.pw/api/plugin/list') for (const pl of result.data) { this.packageCache[pl.name] = pl } this.packageNameCache = Object.keys(this.packageCache) this.i18n(sender, 'cloud.update.finish', { length: this.packageNameCache.length }) this.pluginManager.getPlugins().forEach(p => { let cloudPlugin = this.packageCache[p.description.name] //§6插件名称: §b{name}\n§6版本: §a{version}\n§6作者: §3{author}\§6更新时间: §9{updated_at} if (cloudPlugin && cloudPlugin.version != p.description.version) { this.i18n(sender, 'cloud.update.exists', { name: p.description.name, new_version: cloudPlugin.version, old_version: p.description.version, changelog: cloudPlugin.changelog || '' }) } }) }).async().submit() } download(sender: any, name: string, update: boolean, callback?: () => void) { this.taskManager.create(() => { let pluginPkg = this.packageCache[name] this.i18n(sender, 'download.start', { name, version: pluginPkg.version }) this.i18n(sender, 'download.url', { url: pluginPkg.url }) let pluginFile = update ? fs.concat(root, this.pluginFolder, 'update', name + '.js') : fs.concat(root, this.pluginFolder, name + '.js') http.download(pluginPkg.url, pluginFile) this.i18n(sender, 'download.finish', { name, version: pluginPkg.version }) callback?.() }).async().submit() } }