Compare commits
41 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e3378a257e | |||
| 75302195e3 | |||
| 78ab1a73d6 | |||
| 867fc802ec | |||
| ed588e4502 | |||
| cbf00d107e | |||
| b9a9334655 | |||
| 2a58ad46d2 | |||
| b301948583 | |||
| b21aa1051d | |||
| 0f418f39df | |||
| 75cb430230 | |||
| 6aedd8e680 | |||
| cd31f80805 | |||
| 08ba1c1a98 | |||
| d5c2a825fc | |||
| b36b63277f | |||
| 7d02194ac7 | |||
| ce4ad6f046 | |||
| 24691a9ce8 | |||
| 9fa13f49cd | |||
| 64a698089b | |||
| 53843b65d2 | |||
| 5f00431e8b | |||
| c947ff7a14 | |||
| be2988fc58 | |||
| 3beed64319 | |||
| 23c7cb955a | |||
| 3be1f78a14 | |||
| d919fa07fc | |||
| d8fd7b0a7d | |||
| 907f9ed03f | |||
| e28d881976 | |||
| 1bdb324fc4 | |||
| 5c90ac3b1f | |||
| 653d7268cf | |||
| efa5e6d110 | |||
| ceb354f11e | |||
| d97af356fd | |||
| 6fd7174ca5 | |||
| 784ea2d65a |
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"useWorkspaces": true,
|
||||
"npmClient": "yarn",
|
||||
"packages": [
|
||||
|
||||
@@ -12,7 +12,7 @@
|
||||
"build": "lerna run build --scope=\"@ccms/!(plugins)\"",
|
||||
"build:plugins": "lerna run build --scope=\"@ccms/plugins\"",
|
||||
"ug": "yarn upgrade-interactive --latest",
|
||||
"np": "lerna exec \"npm publish --access=public --registry https://registry.npmjs.org\" --scope=\"@ccms/!(client|plugins)\"",
|
||||
"np": "./script/push.sh",
|
||||
"lsp": "npm login --registry=https://registry.npmjs.org --scope=@ccms",
|
||||
"lp": "lerna publish --registry https://registry.npmjs.org"
|
||||
},
|
||||
@@ -20,6 +20,6 @@
|
||||
"packages/*"
|
||||
],
|
||||
"devDependencies": {
|
||||
"lerna": "^3.22.1"
|
||||
"lerna": "^4.0.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/amqp",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript amqp package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,17 +19,17 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"@javatypes/amqp-client": "^0.0.3",
|
||||
"@javatypes/spring-amqp": "^0.0.3",
|
||||
"@javatypes/spring-rabbit": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/api",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,9 +19,9 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0",
|
||||
"@ccms/ployfill": "^0.13.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0",
|
||||
"@ccms/polyfill": "^0.17.0",
|
||||
"base64-js": "^1.5.1",
|
||||
"source-map-builder": "^0.0.7"
|
||||
},
|
||||
@@ -29,6 +29,6 @@
|
||||
"@types/base64-js": "^1.3.0",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,7 +5,7 @@ import * as base64 from 'base64-js'
|
||||
const Arrays = Java.type('java.util.Arrays')
|
||||
const Level = Java.type('java.util.logging.Level')
|
||||
const Paths = Java.type('java.nio.file.Paths')
|
||||
const ignoreLogPrefix = ['java.', 'javax.', 'sun.', 'net.minecraft.', 'org.bukkit.', 'jdk.nashorn.', 'io.netty.', 'org.spongepowered.', 'org.apache', 'org.springframework']
|
||||
const ignoreLogPrefix = ['java.', 'javax.', 'sun.', 'net.minecraft.', 'org.bukkit.', 'jdk.nashorn.', 'org.openjdk.nashorn', 'io.netty.', 'org.spongepowered.', 'org.apache', 'org.springframework']
|
||||
|
||||
enum LogLevel {
|
||||
ALL,
|
||||
@@ -19,7 +19,8 @@ enum LogLevel {
|
||||
}
|
||||
|
||||
export class MiaoScriptConsole implements Console {
|
||||
Console: NodeJS.ConsoleConstructor
|
||||
Console: any
|
||||
memory: any
|
||||
|
||||
private static sourceMaps: { [key: string]: SourceMapBuilder } = {}
|
||||
private static sourceFileMaps: { [key: string]: string } = {}
|
||||
@@ -104,7 +105,7 @@ export class MiaoScriptConsole implements Console {
|
||||
let sourceMappingURL = lastLine.split('sourceMappingURL=', 2)[1]
|
||||
if (sourceMappingURL.startsWith('data:application/json;base64,')) {
|
||||
sourceContent = String.fromCharCode(...Array.from(base64.toByteArray(sourceMappingURL.split(',', 2)[1])))
|
||||
} else if (sourceMappingURL.startsWith('http')) {
|
||||
} else if (sourceMappingURL.startsWith('http://') || sourceMappingURL.startsWith('https://')) {
|
||||
// TODO
|
||||
} else {
|
||||
let file = Paths.get(Paths.get(fileName, '..', sourceMappingURL).toFile().getCanonicalPath()).toFile()
|
||||
@@ -155,7 +156,7 @@ export class MiaoScriptConsole implements Console {
|
||||
let className = trace.className
|
||||
var fileName = trace.fileName as string
|
||||
var lineNumber = trace.lineNumber
|
||||
if (className.startsWith('jdk.nashorn.internal.scripts')) {
|
||||
if (className.startsWith('jdk.nashorn.internal.scripts') || className.startsWith('org.openjdk.nashorn.internal.scripts')) {
|
||||
className = className.substr(className.lastIndexOf('$') + 1)
|
||||
var { fileName, lineNumber } = this.readSourceMap(fileName, lineNumber)
|
||||
if (fileName.startsWith(root)) { fileName = fileName.split(root)[1] }
|
||||
@@ -189,12 +190,15 @@ export class MiaoScriptConsole implements Console {
|
||||
countReset(label?: string): void {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
dir(obj: any, options?: NodeJS.InspectOptions): void {
|
||||
dir(obj: any, options?: any): void {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
dirxml(...data: any[]): void {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
exception(message?: string, ...optionalParams: any[]): void {
|
||||
throw new Error('Method not implemented.')
|
||||
}
|
||||
group(...label: any[]): void {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
|
||||
@@ -16,15 +16,28 @@ export namespace server {
|
||||
* Runtime Server Instance
|
||||
*/
|
||||
export const ServerInstance = Symbol("ServerInstance")
|
||||
export interface NativePlugin {
|
||||
name: string
|
||||
version: string
|
||||
authors?: string | string[]
|
||||
enable: boolean
|
||||
depends?: string[]
|
||||
softDepends?: string[]
|
||||
/**
|
||||
* 插件本体
|
||||
*/
|
||||
origin: any
|
||||
[key: string]: any
|
||||
}
|
||||
@injectable()
|
||||
export abstract class NativePluginManager {
|
||||
list(): any[] {
|
||||
list(): NativePlugin[] {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
has(name: string): boolean {
|
||||
return true
|
||||
}
|
||||
get(name: string): any {
|
||||
get(name: string): NativePlugin {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
load(name: string): boolean {
|
||||
@@ -60,6 +73,12 @@ export namespace server {
|
||||
getService(service: string): any {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
broadcast(message: string, permission: string) {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
broadcastMessage(message: string) {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
dispatchCommand(sender: string | any, command: string): boolean {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
|
||||
@@ -17,7 +17,7 @@ export namespace task {
|
||||
this.cacheTasks.delete(taskId)
|
||||
let ownerName = task.getOwner()?.description.name
|
||||
if (ownerName && this.pluginCacheTasks.has(ownerName)) {
|
||||
this.pluginCacheTasks.get(ownerName).delete(taskId)
|
||||
this.pluginCacheTasks.get(ownerName)?.delete(taskId)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/bukkit",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript bukkit package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,11 +22,11 @@
|
||||
"@javatypes/spigot-api": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,6 +15,7 @@ let downgrade = false
|
||||
* 获取NMS版本
|
||||
*/
|
||||
let nmsVersion = undefined
|
||||
let nmsSubVersion = undefined
|
||||
/**
|
||||
* 获取NMS类
|
||||
*/
|
||||
@@ -49,15 +50,23 @@ function remapFieldName(clazz: any, origin: string, test: string) {
|
||||
function init() {
|
||||
//@ts-ignore
|
||||
nmsVersion = org.bukkit.Bukkit.server.class.name.split('.')[3]
|
||||
nmsSubVersion = nmsVersion.split("_")[1]
|
||||
try {
|
||||
RemapUtils = Java.type('catserver.server.remapper.RemapUtils')
|
||||
} catch (ex) {
|
||||
}
|
||||
let nmsChatSerializerClass = nmsCls(nmsVersion.split("_")[1] > 7 ? "IChatBaseComponent$ChatSerializer" : "ChatSerializer")
|
||||
let nmsChatSerializerClass = undefined
|
||||
if (nmsSubVersion < 8) {
|
||||
nmsChatSerializerClass = nmsCls("ChatSerializer")
|
||||
} else if (nmsSubVersion < 17) {
|
||||
nmsChatSerializerClass = nmsCls("IChatBaseComponent$ChatSerializer")
|
||||
} else {
|
||||
nmsChatSerializerClass = base.getClass('net.minecraft.network.chat.IChatBaseComponent$ChatSerializer')
|
||||
}
|
||||
let nmsChatSerializerMethod = remapMethod(nmsChatSerializerClass, 'a', 'func_150699_a', base.getClass('java.lang.String'))
|
||||
nmsChatSerializerMethodName = nmsChatSerializerMethod.getName()
|
||||
ChatSerializer = Java.type(nmsChatSerializerClass.getName())
|
||||
let packetTypeClass = nmsCls("PacketPlayOutChat")
|
||||
let packetTypeClass = nmsSubVersion < 17 ? nmsCls("PacketPlayOutChat") : base.getClass('net.minecraft.network.protocol.game.PacketPlayOutChat')
|
||||
PacketPlayOutChat = Java.type(packetTypeClass.getName())
|
||||
let packetTypeConstructor: { parameterTypes: any[] }
|
||||
let constructors = packetTypeClass.constructors
|
||||
@@ -75,9 +84,14 @@ function init() {
|
||||
if (nmsChatMessageTypeClass.isEnum()) {
|
||||
chatMessageTypes = nmsChatMessageTypeClass.getEnumConstants()
|
||||
}
|
||||
let playerConnectionField = remapFieldName(nmsCls('EntityPlayer'), 'playerConnection', 'field_71135_a')
|
||||
let playerConnectionField = undefined
|
||||
if (nmsSubVersion < 17) {
|
||||
playerConnectionField = remapFieldName(nmsCls('EntityPlayer'), 'playerConnection', 'field_71135_a')
|
||||
} else {
|
||||
playerConnectionField = base.getClass('net.minecraft.server.level.EntityPlayer').getField('b')
|
||||
}
|
||||
playerConnectionFieldName = playerConnectionField.getName()
|
||||
sendPacketMethodName = remapMethod(playerConnectionField.getType(), 'sendPacket', 'func_179290_a', nmsCls('Packet')).getName()
|
||||
sendPacketMethodName = remapMethod(playerConnectionField.getType(), 'sendPacket', 'func_179290_a', nmsSubVersion < 17 ? nmsCls('Packet') : base.getClass('net.minecraft.network.protocol.Packet')).getName()
|
||||
}
|
||||
|
||||
function json(sender: { name: string }, json: string) {
|
||||
|
||||
@@ -1,12 +1,36 @@
|
||||
import { server } from '@ccms/api'
|
||||
|
||||
const Bukkit = org.bukkit.Bukkit
|
||||
const Bukkit: typeof org.bukkit.Bukkit = Java.type('org.bukkit.Bukkit')
|
||||
|
||||
export class BukkitNativePluginManager extends server.NativePluginManager {
|
||||
has(name: string) {
|
||||
return !!this.get(name)
|
||||
private bukkitPluginManager: org.bukkit.plugin.PluginManager
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
this.bukkitPluginManager = Bukkit.getPluginManager()
|
||||
}
|
||||
get(name: string) {
|
||||
return Bukkit.getPluginManager().getPlugin(name)
|
||||
|
||||
list(): server.NativePlugin[] {
|
||||
return Java.from(this.bukkitPluginManager.getPlugins()).map(plugin => this.convert(plugin))
|
||||
}
|
||||
has(name: string): boolean {
|
||||
return !!this.bukkitPluginManager.getPlugin(name)
|
||||
}
|
||||
get(name: string): server.NativePlugin {
|
||||
return this.convert(this.bukkitPluginManager.getPlugin(name))
|
||||
}
|
||||
|
||||
private convert(plugin: org.bukkit.plugin.Plugin): server.NativePlugin {
|
||||
if (!plugin) return plugin as any
|
||||
let desc = plugin.getDescription()
|
||||
return {
|
||||
name: plugin.getName(),
|
||||
version: desc.getVersion(),
|
||||
authors: Java.from(desc.getAuthors()),
|
||||
depends: Java.from(desc.getDepend()),
|
||||
softDepends: Java.from(desc.getSoftDepend()),
|
||||
enable: plugin.isEnabled(),
|
||||
origin: plugin
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -4,7 +4,7 @@ import { provideSingleton } from '@ccms/container'
|
||||
import * as reflect from '@ccms/common/dist/reflect'
|
||||
import chat from './enhance/chat'
|
||||
|
||||
let Bukkit = org.bukkit.Bukkit
|
||||
let Bukkit: typeof org.bukkit.Bukkit = org.bukkit.Bukkit
|
||||
|
||||
@provideSingleton(server.Server)
|
||||
export class BukkitServer extends server.ReflectServer {
|
||||
@@ -30,6 +30,12 @@ export class BukkitServer extends server.ReflectServer {
|
||||
getService(service: string) {
|
||||
return Bukkit.getServicesManager().getRegistration(base.getClass(service))?.getProvider()
|
||||
}
|
||||
broadcast(message: string, permission: string) {
|
||||
return Bukkit.broadcast(message, permission)
|
||||
}
|
||||
broadcastMessage(message: string) {
|
||||
return Bukkit.broadcastMessage(message)
|
||||
}
|
||||
dispatchCommand(sender: string | any, command: string): boolean {
|
||||
if (typeof sender === 'string') {
|
||||
sender = this.getPlayer(sender)
|
||||
|
||||
@@ -21,11 +21,11 @@ export class BukkitTaskManager extends task.TaskManager {
|
||||
export class BukkitTask extends task.Task {
|
||||
submit0(...args: any[]): task.Cancelable {
|
||||
let run = new BukkitRunnable({ run: () => this.run(...args) })
|
||||
let funcName = `runTask${this.interval ? 'Timer' : 'Later'}${this.isAsync ? 'Asynchronously' : ''}`
|
||||
let suffix = this.isAsync ? 'Asynchronously' : ''
|
||||
if (this.interval) {
|
||||
return run[funcName](base.getInstance(), this.laterTime, this.interval)
|
||||
return run[`runTaskTimer${suffix}`](base.getInstance(), this.laterTime, this.interval)
|
||||
} else {
|
||||
return run[funcName](base.getInstance(), this.laterTime)
|
||||
return run[`runTaskLater${suffix}`](base.getInstance(), this.laterTime)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/bungee",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript bungee package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,11 +22,11 @@
|
||||
"@javatypes/bungee-api": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -3,10 +3,34 @@ import { server } from '@ccms/api'
|
||||
let Bungee: net.md_5.bungee.api.ProxyServer = base.getInstance().getProxy()
|
||||
|
||||
export class BungeeNativePluginManager extends server.NativePluginManager {
|
||||
has(name: string) {
|
||||
return !!this.get(name)
|
||||
private bungeePluginManager: net.md_5.bungee.api.plugin.PluginManager
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
this.bungeePluginManager = Bungee.getPluginManager()
|
||||
}
|
||||
get(name: string) {
|
||||
return Bungee.getPluginManager().getPlugin(name)
|
||||
|
||||
list(): server.NativePlugin[] {
|
||||
return Java.from(this.bungeePluginManager.getPlugins()).map(plugin => this.convert(plugin))
|
||||
}
|
||||
has(name: string): boolean {
|
||||
return !!this.bungeePluginManager.getPlugin(name)
|
||||
}
|
||||
get(name: string): server.NativePlugin {
|
||||
return this.convert(this.bungeePluginManager.getPlugin(name))
|
||||
}
|
||||
|
||||
private convert(plugin: net.md_5.bungee.api.plugin.Plugin): server.NativePlugin {
|
||||
if (!plugin) return plugin as any
|
||||
let desc = plugin.getDescription()
|
||||
return {
|
||||
name: desc.getName(),
|
||||
version: desc.getVersion(),
|
||||
authors: [desc.getAuthor()],
|
||||
depends: Java.from(desc.getDepends()) as any,
|
||||
softDepends: Java.from(desc.getSoftDepends()) as any,
|
||||
enable: true,
|
||||
origin: plugin
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -58,6 +58,12 @@ export class BungeeServer implements server.Server {
|
||||
getService(service: string) {
|
||||
throw new Error("Method not implemented.")
|
||||
}
|
||||
broadcast(message: string, permission: string) {
|
||||
return Bungee.broadcast(message)
|
||||
}
|
||||
broadcastMessage(message: string) {
|
||||
return Bungee.broadcast(message)
|
||||
}
|
||||
dispatchCommand(sender: string | any, command: string): boolean {
|
||||
if (typeof sender === 'string') {
|
||||
sender = this.getPlayer(sender)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"private": true,
|
||||
"name": "@ccms/client",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript client package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,10 +22,11 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"minecraft-protocol": "^1.19.0"
|
||||
"minecraft-protocol": "^1.25.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/node": "^16.4.12",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -4,10 +4,12 @@ import { createClient } from 'minecraft-protocol'
|
||||
import { attachForge } from './forge'
|
||||
import { attachEvents } from './event'
|
||||
|
||||
let username = process.argv[2] || 'Mr_jtb'
|
||||
let password = process.argv[3] || ''
|
||||
let version = process.argv[4] || '1.12.2'
|
||||
let readAddress = process.argv[5] || '192.168.2.5:25577'
|
||||
let readUserInfo = process.argv[2] || 'Mr_jtb'
|
||||
let realUserInfo = readUserInfo.split(":")
|
||||
let username = realUserInfo[0]
|
||||
let password = realUserInfo[1] || ''
|
||||
let version = process.argv[3] || '1.12.2'
|
||||
let readAddress = process.argv[4] || '192.168.2.25:25565'
|
||||
let realAddress = readAddress.split(":")
|
||||
let address = realAddress[0]
|
||||
let port = parseInt(realAddress[1] || "25565")
|
||||
@@ -18,17 +20,23 @@ function commandLineCreateClient() {
|
||||
}
|
||||
|
||||
function createConnection(host: string, port: number, username: string, password: string) {
|
||||
let client = createClient({
|
||||
let clientOptions: any = {
|
||||
version,
|
||||
host,
|
||||
port,
|
||||
username,
|
||||
password,
|
||||
clientToken: 'd02c7f39-2376-45da-a5a5-50e24fa8b185',
|
||||
// clientToken: 'd02c7f39-2376-45da-a5a5-50e24fa8b185',
|
||||
//@ts-ignore
|
||||
authServer: 'https://mcsso.yumc.pw/api/yggdrasil/authserver',
|
||||
sessionServer: 'https://mcsso.yumc.pw/api/yggdrasil/sessionserver'
|
||||
})
|
||||
// authServer: 'https://skin.yumc.pw/api/yggdrasil/authserver',
|
||||
// sessionServer: 'https://skin.yumc.pw/api/yggdrasil/sessionserver'
|
||||
}
|
||||
if (clientOptions.password) {
|
||||
clientOptions.clientToken = 'd02c7f39-2376-45da-a5a5-50e24fa8b185'
|
||||
clientOptions.authServer = 'https://skin.yumc.pw/api/yggdrasil/authserver'
|
||||
clientOptions.sessionServer = 'https://skin.yumc.pw/api/yggdrasil/sessionserver'
|
||||
}
|
||||
let client = createClient(clientOptions)
|
||||
|
||||
attachCommon(client)
|
||||
attachForge(client)
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/common",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,11 +19,11 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"@javatypes/jdk": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"gitHead": "562e2d00175c9d3a99c8b672aa07e6d92706a027"
|
||||
}
|
||||
|
||||
@@ -23,7 +23,7 @@ interface RequestConfig {
|
||||
}
|
||||
|
||||
function request(config: RequestConfig) {
|
||||
// @ts-ignore
|
||||
// @ts-ignore XMLHttpRequest class only exist nashorn polyfill
|
||||
let xhr = new XMLHttpRequest()
|
||||
xhr.open(config.method, config.url, false)
|
||||
for (const header in config.headers) {
|
||||
@@ -44,6 +44,7 @@ function request(config: RequestConfig) {
|
||||
if (xhr.getResponseHeader("Content-Type").indexOf('application/json') != -1) {
|
||||
xhr.responseType = "json"
|
||||
}
|
||||
// @ts-ignore get only exist nashorn polyfill
|
||||
return xhr.get()
|
||||
}
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/compile",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript compile package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -21,6 +21,6 @@
|
||||
"devDependencies": {
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/container",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript container package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,13 +19,13 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"inversify": "^5.0.1",
|
||||
"inversify": "^5.1.1",
|
||||
"inversify-binding-decorators": "^4.0.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/core",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -21,11 +21,11 @@
|
||||
"devDependencies": {
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
},
|
||||
"gitHead": "781524f83e52cad26d7c480513e3c525df867121"
|
||||
}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
let containerStartTime = Date.now()
|
||||
console.i18n("ms.core.ioc.initialize", { scope: global.scope })
|
||||
import { plugin, server, task, constants } from '@ccms/api'
|
||||
import { DefaultContainer as container, inject, provideSingleton, ContainerInstance, buildProviderModule, Autowired } from '@ccms/container'
|
||||
import { DefaultContainer as container, provideSingleton, ContainerInstance, buildProviderModule, Autowired } from '@ccms/container'
|
||||
console.i18n("ms.core.ioc.completed", { scope: global.scope, time: (Date.now() - containerStartTime) / 1000 })
|
||||
import http from '@ccms/common/dist/http'
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/database",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript database package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,10 +22,10 @@
|
||||
"@javatypes/spring-jdbc": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
ms.i18n.completed: "Internationalization component initialization completed. Current Language: English"
|
||||
ms.ployfill.initialize: "Initialization Java Nashorn ployfill. Please wait..."
|
||||
ms.ployfill.completed: "Java Nashorn ployfill loading completed... Cost ({time}s)!"
|
||||
ms.polyfill.initialize: "Initialization Java Nashorn polyfill. Please wait..."
|
||||
ms.polyfill.completed: "Java Nashorn polyfill loading completed... Cost ({time}s)!"
|
||||
|
||||
ms.core.ioc.initialize: "Initialization MiaoScript IOC Container {scope}/container. Please wait..."
|
||||
ms.core.ioc.completed: "MiaoScript IOC Container {scope}/container loading completed({time}s)!"
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
ms.i18n.completed: "国际化组件 初始化完成 当前语言: 简体中文"
|
||||
ms.ployfill.initialize: "加载 Java Nashorn 补丁. 请稍候..."
|
||||
ms.ployfill.completed: "Java Nashorn 补丁 加载完成... 耗时 ({time}s)!"
|
||||
ms.polyfill.initialize: "加载 Java Nashorn 补丁. 请稍候..."
|
||||
ms.polyfill.completed: "Java Nashorn 补丁 加载完成... 耗时 ({time}s)!"
|
||||
|
||||
ms.core.ioc.initialize: "初始化 MiaoScript IOC 容器 {scope}/container. 请稍候..."
|
||||
ms.core.ioc.completed: "MiaoScript IOC 容器 {scope}/container 加载完成 耗时({time}s)"
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/i18n",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript i18n package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,14 +19,14 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@types/js-yaml": "^3.12.5",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"@types/js-yaml": "^4.0.2",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"js-yaml": "^3.14.0"
|
||||
"js-yaml": "^4.1.0"
|
||||
},
|
||||
"gitHead": "781524f83e52cad26d7c480513e3c525df867121"
|
||||
}
|
||||
|
||||
@@ -50,7 +50,7 @@ export class Translate {
|
||||
|
||||
readYamlFile(dir: string, name: string) {
|
||||
let langFile = this.concat(dir, 'languages', name + '.yml')
|
||||
return this.exists(langFile) && yaml.safeLoad(base.read(langFile))
|
||||
return this.exists(langFile) && yaml.load(base.read(langFile))
|
||||
}
|
||||
|
||||
concat(...args: string[]) {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/keyvalue",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript keyvalue package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,18 +19,18 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"@javatypes/amqp-client": "^0.0.3",
|
||||
"@javatypes/spring-amqp": "^0.0.3",
|
||||
"@javatypes/spring-rabbit": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"gitHead": "2589633069d24f646ac09261b1b2304c21d4ea75"
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/nashorn",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,6 +22,6 @@
|
||||
"devDependencies": {
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/nodejs",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript nodejs package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,10 +19,11 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"tslib": "^2.3.0",
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"gitHead": "781524f83e52cad26d7c480513e3c525df867121"
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/nukkit",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript nukkit package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,11 +22,11 @@
|
||||
"@javatypes/nukkit-api": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -3,10 +3,34 @@ import { server } from '@ccms/api'
|
||||
let Nukkit: cn.nukkit.Server = base.getInstance().getServer()
|
||||
|
||||
export class NukkitNativePluginManager extends server.NativePluginManager {
|
||||
private nukkitPluginManager: cn.nukkit.plugin.PluginManager
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
this.nukkitPluginManager = Nukkit.getPluginManager()
|
||||
}
|
||||
|
||||
list(): server.NativePlugin[] {
|
||||
return Java.from(this.nukkitPluginManager.getPlugins().values()).map(plugin => this.convert(plugin))
|
||||
}
|
||||
has(name: string) {
|
||||
return !!this.get(name)
|
||||
return !!this.nukkitPluginManager.getPlugin(name)
|
||||
}
|
||||
get(name: string) {
|
||||
return Nukkit.getPluginManager().getPlugin(name)
|
||||
return this.convert(this.nukkitPluginManager.getPlugin(name))
|
||||
}
|
||||
|
||||
private convert(plugin: cn.nukkit.plugin.Plugin): server.NativePlugin {
|
||||
if (!plugin) return plugin as any
|
||||
let desc = plugin.getDescription()
|
||||
return {
|
||||
name: plugin.getName(),
|
||||
version: desc.getVersion(),
|
||||
authors: Java.from(desc.getAuthors() as string[]),
|
||||
depends: Java.from(desc.getDepend() as string[]),
|
||||
softDepends: Java.from(desc.getSoftDepend() as string[]),
|
||||
enable: plugin.isEnabled(),
|
||||
origin: plugin
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -27,6 +27,12 @@ export class NukkitServer implements server.Server {
|
||||
getService(service: string) {
|
||||
return Nukkit.getServiceManager().getProvider(base.getClass(service))
|
||||
}
|
||||
broadcast(message: string, permission: string) {
|
||||
return Nukkit.broadcast(message, permission)
|
||||
}
|
||||
broadcastMessage(message: string) {
|
||||
return Nukkit.broadcastMessage(message)
|
||||
}
|
||||
dispatchCommand(sender: string | any, command: string): boolean {
|
||||
if (typeof sender === 'string') {
|
||||
sender = this.getPlayer(sender)
|
||||
|
||||
@@ -1,141 +0,0 @@
|
||||
(function nashornEventLoopMain(context) {
|
||||
'use strict';
|
||||
|
||||
var Thread = Java.type('java.lang.Thread');
|
||||
var Phaser = Java.type('java.util.concurrent.Phaser');
|
||||
var ArrayDeque = Java.type('java.util.ArrayDeque');
|
||||
var HashMap = Java.type('java.util.HashMap');
|
||||
var TimeUnit = Java.type("java.util.concurrent.TimeUnit");
|
||||
var Runnable = Java.type('java.lang.Runnable');
|
||||
|
||||
var globalTimerId;
|
||||
var timerMap;
|
||||
var eventLoop;
|
||||
var phaser = new Phaser();
|
||||
|
||||
// __NASHORN_POLYFILL_TIMER__ type is ScheduledExecutorService
|
||||
var scheduler = context.__NASHORN_POLYFILL_TIMER__;
|
||||
|
||||
resetEventLoop();
|
||||
|
||||
// console.log('main javasript thread ' + Thread.currentThread().getName());
|
||||
|
||||
function resetEventLoop() {
|
||||
globalTimerId = 1;
|
||||
if (timerMap) {
|
||||
timerMap.forEach(function(key, value) {
|
||||
value.cancel(true);
|
||||
})
|
||||
}
|
||||
timerMap = new HashMap();
|
||||
eventLoop = new ArrayDeque();
|
||||
}
|
||||
|
||||
function waitForMessages() {
|
||||
phaser.register();
|
||||
var wait = !(eventLoop.size() === 0);
|
||||
phaser.arriveAndDeregister();
|
||||
return wait;
|
||||
}
|
||||
|
||||
function processNextMessages() {
|
||||
var remaining = 1;
|
||||
while (remaining) {
|
||||
phaser.register();
|
||||
var message = eventLoop.removeFirst();
|
||||
remaining = eventLoop.size();
|
||||
phaser.arriveAndDeregister();
|
||||
|
||||
var fn = message.fn;
|
||||
var args = message.args;
|
||||
|
||||
try {
|
||||
fn.apply(context, args);
|
||||
} catch (e) {
|
||||
console.trace(e);
|
||||
console.trace(fn);
|
||||
console.trace(args);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
context.nashornEventLoop = {
|
||||
process: function() {
|
||||
while (waitForMessages()) {
|
||||
processNextMessages()
|
||||
}
|
||||
},
|
||||
reset: resetEventLoop
|
||||
};
|
||||
|
||||
|
||||
function createRunnable(fn, timerId, args, repeated) {
|
||||
return new Runnable({
|
||||
run: function() {
|
||||
try {
|
||||
var phase = phaser.register();
|
||||
eventLoop.addLast({
|
||||
fn: fn,
|
||||
args: args
|
||||
});
|
||||
} catch (e) {
|
||||
console.trace(e);
|
||||
} finally {
|
||||
if (!repeated) timerMap.remove(timerId);
|
||||
phaser.arriveAndDeregister();
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
var setTimeout = function(fn, millis /* [, args...] */) {
|
||||
var args = [].slice.call(arguments, 2, arguments.length);
|
||||
|
||||
var timerId = globalTimerId++;
|
||||
var runnable = createRunnable(fn, timerId, args, false);
|
||||
|
||||
var task = scheduler.schedule(runnable, millis, TimeUnit.MILLISECONDS);
|
||||
timerMap.put(timerId, task);
|
||||
|
||||
return timerId;
|
||||
};
|
||||
|
||||
var setImmediate = function(fn /* [, args...] */) {
|
||||
var args = [].slice.call(arguments, 1, arguments.length);
|
||||
// @ts-ignore
|
||||
return setTimeout(fn, 0, args);
|
||||
}
|
||||
|
||||
var clearImmediate = function(timerId) {
|
||||
clearTimeout(timerId);
|
||||
}
|
||||
|
||||
var clearTimeout = function(timerId) {
|
||||
var task = timerMap.get(timerId);
|
||||
if (task) {
|
||||
task.cancel(true);
|
||||
timerMap.remove(timerId);
|
||||
}
|
||||
};
|
||||
|
||||
var setInterval = function(fn, delay /* [, args...] */) {
|
||||
var args = [].slice.call(arguments, 2, arguments.length);
|
||||
var timerId = globalTimerId++;
|
||||
var runnable = createRunnable(fn, timerId, args, true);
|
||||
var task = scheduler.scheduleWithFixedDelay(runnable, delay, delay, TimeUnit.MILLISECONDS);
|
||||
timerMap.put(timerId, task);
|
||||
return timerId;
|
||||
};
|
||||
|
||||
var clearInterval = function(timerId) {
|
||||
clearTimeout(timerId);
|
||||
};
|
||||
|
||||
context.setTimeout = setTimeout;
|
||||
context.clearTimeout = clearTimeout;
|
||||
context.setImmediate = setImmediate;
|
||||
context.clearImmediate = clearImmediate;
|
||||
context.setInterval = setInterval;
|
||||
context.clearInterval = clearInterval;
|
||||
// @ts-ignore
|
||||
})(typeof global !== "undefined" && global || typeof self !== "undefined" && self || this);
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/plugin",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -19,16 +19,17 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/js-yaml": "^3.12.5",
|
||||
"@types/js-yaml": "^4.0.2",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0",
|
||||
"@ccms/i18n": "^0.13.0",
|
||||
"js-yaml": "^3.14.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0",
|
||||
"@ccms/i18n": "^0.17.0",
|
||||
"js-yaml": "^4.1.0",
|
||||
"yaml": "^1.10.2"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -47,8 +47,8 @@ export class PluginCommandManager {
|
||||
let subcommand = args[0] || 'help'
|
||||
let cmdKey = 'cmd' + subcommand
|
||||
if (!pluginInstance[cmdKey]) {
|
||||
console.sender(sender, '§4未知的子命令: §c' + subcommand)
|
||||
pluginInstance['cmdhelp'] && console.sender(sender, `§6请执行 §b/${command} §ahelp §6查看帮助!`)
|
||||
pluginInstance.logger.sender(sender, '§4未知的子命令: §c' + subcommand)
|
||||
pluginInstance['cmdhelp'] && pluginInstance.logger.sender(sender, `§6请执行 §b/${command} §ahelp §6查看帮助!`)
|
||||
return
|
||||
}
|
||||
args.shift()
|
||||
|
||||
@@ -13,10 +13,10 @@ export interface PluginConfigLoader {
|
||||
|
||||
export class YamlPluginConfig implements PluginConfigLoader {
|
||||
load(content: string) {
|
||||
return yaml.safeLoad(content)
|
||||
return yaml.load(content)
|
||||
}
|
||||
dump(variable: any): string {
|
||||
return yaml.safeDump(variable, { skipInvalid: true, lineWidth: 120 })
|
||||
return yaml.dump(variable, { skipInvalid: true, lineWidth: 120 })
|
||||
}
|
||||
}
|
||||
|
||||
@@ -85,37 +85,46 @@ export class PluginConfigManager {
|
||||
try {
|
||||
metadata.file = fs.concat(fs.file(plugin.description.loadMetadata.file).parent, plugin.description.name, metadata.filename)
|
||||
let configLoader = this.getConfigLoader(metadata.format)
|
||||
let value = plugin[metadata.variable]
|
||||
let defaultValue = metadata.default ?? plugin[metadata.variable]
|
||||
let configValue = defaultValue
|
||||
if (!fs.exists(metadata.file)) {
|
||||
base.save(metadata.file, configLoader.dump(value))
|
||||
base.save(metadata.file, configLoader.dump(defaultValue))
|
||||
console.i18n("ms.plugin.manager.config.save.default", { plugin: plugin.description.name, name: metadata.name, format: metadata.format })
|
||||
} else {
|
||||
value = configLoader.load(base.read(metadata.file))
|
||||
console.debug(`[${plugin.description.name}] Load Config ${metadata.variable} from file ${metadata.file} =>\n${JSON.stringify(value, undefined, 4)}`)
|
||||
if (metadata.default) {
|
||||
let needSave = false
|
||||
for (const key of Object.keys(metadata.default)) {
|
||||
if (!value[key]) {
|
||||
value[key] = metadata.default[key]
|
||||
needSave = true
|
||||
}
|
||||
}
|
||||
needSave && base.save(metadata.file, configLoader.dump(value))
|
||||
configValue = configLoader.load(base.read(metadata.file)) || {}
|
||||
if (defaultValue && this.setDefaultValue(configValue, defaultValue)) {
|
||||
base.save(metadata.file, configLoader.dump(configValue))
|
||||
}
|
||||
console.debug(`[${plugin.description.name}] Load Config ${metadata.variable} from file ${metadata.file} =>\n${JSON.stringify(configValue, undefined, 4).substr(0, 500)}`)
|
||||
}
|
||||
this.defienConfigProp(plugin, metadata, value)
|
||||
this.defienConfigProp(plugin, metadata, configValue)
|
||||
} catch (error) {
|
||||
console.i18n("ms.plugin.manager.config.load.error", { plugin: plugin.description.name, name: metadata.name, format: metadata.format, error })
|
||||
console.ex(error)
|
||||
}
|
||||
}
|
||||
|
||||
private setDefaultValue(configValue, defaultValue) {
|
||||
let needSave = false
|
||||
for (const key of Object.keys(defaultValue)) {
|
||||
// 当配置文件不存在当前属性时才进行赋值
|
||||
if (!Object.prototype.hasOwnProperty.call(configValue, key)) {
|
||||
configValue[key] = defaultValue[key]
|
||||
needSave = true
|
||||
} else if (Object.prototype.toString.call(configValue[key]) == "[object Object]") {
|
||||
// 对象需要递归检测
|
||||
needSave ||= this.setDefaultValue(configValue[key], defaultValue[key])
|
||||
}
|
||||
}
|
||||
return needSave
|
||||
}
|
||||
|
||||
private saveConfig0(plugin: plugin.Plugin, metadata: interfaces.ConfigMetadata) {
|
||||
try {
|
||||
metadata.file = fs.concat(fs.file(plugin.description.loadMetadata.file).parent, plugin.description.name, metadata.filename)
|
||||
let result = this.getConfigLoader(metadata.format).dump(plugin[metadata.variable])
|
||||
base.save(metadata.file, result)
|
||||
console.debug(`[${plugin.description.name}] Save Config ${metadata.variable} to file ${metadata.file} =>\n${result}`)
|
||||
console.debug(`[${plugin.description.name}] Save Config ${metadata.variable} to file ${metadata.file} =>\n${result.substr(0, 500)}`)
|
||||
return true
|
||||
} catch (error) {
|
||||
console.i18n("ms.plugin.manager.config.save.error", { plugin: plugin.description.name, name: metadata.name, format: metadata.format, error })
|
||||
|
||||
@@ -8,9 +8,8 @@ import { getPluginMetadatas, getPluginCommandMetadata, getPluginListenerMetadata
|
||||
* MiaoScript plugin
|
||||
* @param metadata PluginMetadata
|
||||
*/
|
||||
export function plugin(metadata: pluginApi.PluginMetadata | any) {
|
||||
export function plugin(metadata: pluginApi.PluginMetadata) {
|
||||
return function (target: any) {
|
||||
if (!metadata.source) metadata = { souece: metadata }
|
||||
metadata = { name: target.name, version: '1.0.0', author: 'Unknow', target, type: 'ioc', ...metadata }
|
||||
decorate(injectable(), target)
|
||||
Reflect.defineMetadata(METADATA_KEY.plugin, metadata, target)
|
||||
|
||||
@@ -41,6 +41,10 @@ export class PluginManagerImpl implements plugin.PluginManager {
|
||||
|
||||
private instanceMap: Map<string, plugin.Plugin>
|
||||
private metadataMap: Map<string, plugin.PluginMetadata>
|
||||
/**
|
||||
* 延时加载插件
|
||||
*/
|
||||
private lazyMetadataMap: Map<string, plugin.PluginMetadata>
|
||||
|
||||
constructor() {
|
||||
this.sacnnerMap = new Map()
|
||||
@@ -48,6 +52,7 @@ export class PluginManagerImpl implements plugin.PluginManager {
|
||||
|
||||
this.instanceMap = new Map()
|
||||
this.metadataMap = new Map()
|
||||
this.lazyMetadataMap = new Map()
|
||||
|
||||
// ignore unused
|
||||
this.taskManager
|
||||
@@ -239,14 +244,14 @@ export class PluginManagerImpl implements plugin.PluginManager {
|
||||
}
|
||||
|
||||
private buildPlugins() {
|
||||
this.metadataMap.forEach((metadata) => {
|
||||
try {
|
||||
this.metadataMap.forEach((metadata, key) => {
|
||||
if (metadata?.depends?.length) {
|
||||
this.lazyMetadataMap.set(key, metadata)
|
||||
} else {
|
||||
this.buildPlugin(metadata)
|
||||
} catch (error) {
|
||||
console.console(`§4无法加载插件 §b${metadata.name} §4构建插件失败!`)
|
||||
console.ex(error)
|
||||
}
|
||||
})
|
||||
this.lazyMetadataMap.forEach((metadata, key) => this.buildPlugin(metadata))
|
||||
}
|
||||
|
||||
private checkDepends(depends: string | string[]) {
|
||||
@@ -262,16 +267,21 @@ export class PluginManagerImpl implements plugin.PluginManager {
|
||||
return loseDepends
|
||||
}
|
||||
private buildPlugin(metadata: plugin.PluginMetadata) {
|
||||
if (this.instanceMap.has(metadata.name)) { throw new Error(`Plugin ${metadata.name} is already load from ${metadata.source}...`) }
|
||||
if (!this.loaderMap.has(metadata.type)) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查 §c${metadata.type} §4加载器是否正常启用!`) }
|
||||
if (!this.serverChecker.check(metadata.servers)) { throw new Error(`§6插件 §b${metadata.name} §c服务器类型不兼容(${metadata.servers.join(',')}) §6忽略加载...`) }
|
||||
let loseDepends = this.checkDepends(metadata.depends) || []
|
||||
if (loseDepends.length) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查依赖 §3[${loseDepends.join(',')}] §4是否安装完整!`) }
|
||||
let loseNativeDepends = this.checkNativeDepends(metadata.nativeDepends) || []
|
||||
if (loseNativeDepends.length) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查插件依赖 §3[${loseNativeDepends.join(',')}] §4是否安装完整!`) }
|
||||
let pluginInstance = this.loaderMap.get(metadata.type).build(metadata)
|
||||
if (!pluginInstance) { throw new Error(`§4加载器 §c${metadata.type} §4加载插件 §c${metadata.name} §4失败!`) }
|
||||
this.instanceMap.set(metadata.name, pluginInstance)
|
||||
return pluginInstance
|
||||
try {
|
||||
if (this.instanceMap.has(metadata.name)) { throw new Error(`Plugin ${metadata.name} is already load from ${metadata.source}...`) }
|
||||
if (!this.loaderMap.has(metadata.type)) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查 §c${metadata.type} §4加载器是否正常启用!`) }
|
||||
if (!this.serverChecker.check(metadata.servers)) { throw new Error(`§6插件 §b${metadata.name} §c服务器类型不兼容(${metadata.servers.join(',')}) §6忽略加载...`) }
|
||||
let loseDepends = this.checkDepends(metadata.depends) || []
|
||||
if (loseDepends.length) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查脚本依赖 §3[${loseDepends.join(',')}] §4是否安装完整!`) }
|
||||
let loseNativeDepends = this.checkNativeDepends(metadata.nativeDepends) || []
|
||||
if (loseNativeDepends.length) { throw new Error(`§4无法加载插件 §b${metadata.name} §4请检查插件依赖 §3[${loseNativeDepends.join(',')}] §4是否安装完整!`) }
|
||||
let pluginInstance = this.loaderMap.get(metadata.type).build(metadata)
|
||||
if (!pluginInstance) { throw new Error(`§4加载器 §c${metadata.type} §4加载插件 §c${metadata.name} §4失败!`) }
|
||||
this.instanceMap.set(metadata.name, pluginInstance)
|
||||
return pluginInstance
|
||||
} catch (error) {
|
||||
console.console(`§4无法加载插件 §b${metadata.name} §4构建插件失败!`)
|
||||
console.ex(error)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -19,7 +19,7 @@ export class JSFileScanner implements plugin.PluginScanner {
|
||||
load(metadata: plugin.PluginLoadMetadata): plugin.PluginLoadMetadata {
|
||||
if (metadata.type !== this.type) { return }
|
||||
this.updatePlugin(metadata.file)
|
||||
//@ts-ignore
|
||||
//@ts-ignore load plugin not use cache
|
||||
metadata.instance = require(metadata.file.toString(), { cache: false })
|
||||
return metadata
|
||||
}
|
||||
|
||||
35
packages/plugins/docs/MiaoLink.md
Normal file
35
packages/plugins/docs/MiaoLink.md
Normal file
@@ -0,0 +1,35 @@
|
||||
[综合|前置]MiaoLink —— 喵式映射 用于无公网环境的自动化端口映射[1.7.10+全版本]
|
||||
# MiaoLink
|
||||
|
||||
## 插件介绍
|
||||
|
||||
> 自动化端口公网映射
|
||||
|
||||
## 图片展示
|
||||
|
||||
- 
|
||||
|
||||
## 使用方式
|
||||
|
||||
- 本插件依赖于 `MiaoScript` 请前往 [站内帖子](https://www.mcbbs.net/thread-774401-1-1.html) 完成安装
|
||||
- 执行 `/mspm install MiaoLink` 安装 MiaoLink 脚本插件
|
||||
- 访问 [圈云映射](https://nps.yumc.pw) 申请一键映射指令
|
||||
- 
|
||||
- 执行网页上提供的指令 等待客户端上线
|
||||
- 
|
||||
- 使用访问地址即可链接服务器
|
||||
- 
|
||||
|
||||
### Roadmap
|
||||
|
||||
- 支持Bukkit端自动化映射(已完成)
|
||||
- 支持Bungee端自动化映射(开发中)
|
||||
- 支持Sponge端自动化映射(开发中)
|
||||
|
||||
### 感谢
|
||||
|
||||
- [NPS](https://github.com/ehang-io/nps) 开源项目
|
||||
- [蓝科数据](https://www.lankodata.com/aff.php?aff=32) 提供的映射节点
|
||||
- [AkkoCloud](https://www.akkocloud.com/aff.php?aff=698) 提供的映射节点
|
||||
|
||||
#### 本插件所用所有代码均为原创,不存在借用/抄袭等行为
|
||||
76
packages/plugins/docs/MiaoLobby.md
Normal file
76
packages/plugins/docs/MiaoLobby.md
Normal file
@@ -0,0 +1,76 @@
|
||||
# MiaoLobby
|
||||
|
||||
## 插件简介
|
||||
|
||||
- 用户进入服务器/用户登录后 自动选择大厅进行传送 防止堆积在登录服
|
||||
- 支持配置 是否登录后自动分配 或 玩家手动执行随机分配
|
||||
- 支持通过 ActionBar 展示传送状态
|
||||
|
||||
### 插件截图
|
||||
|
||||
- 
|
||||
- 
|
||||
|
||||
### 插件配置
|
||||
|
||||
```yaml
|
||||
#配置文件版本 请勿修改
|
||||
Version: 1.5
|
||||
|
||||
#服务器列表
|
||||
Servers:
|
||||
- lobby1
|
||||
- lobby2
|
||||
#传送超时时间(单位: Tick)
|
||||
WaitTime: 35
|
||||
#自带传送(如果开启 则Login自动传送失效)
|
||||
AutoTP: false
|
||||
#登录自动传送(暂时支持AuthMe)
|
||||
LoginAutoTP: true
|
||||
#传送延时(单位: 秒)
|
||||
AutoTPDelay: 10
|
||||
#尝试完毕后是否继续重试
|
||||
ReTry: true
|
||||
#传送提示
|
||||
Message: '&a请稍候 正在传送至服务器 %s ...'
|
||||
TimeOut: '&c传送超时 正在切换到服务器 %s ...'
|
||||
TPDelay: '&a登陆成功 正在为您匹配服务器 剩余 %s 秒...'
|
||||
Unavailable: '&4已尝试所有可用服务器 传送失败!'
|
||||
```
|
||||
|
||||
### 插件命令
|
||||
|
||||
```
|
||||
插件注册命令:
|
||||
- MiaoLobby
|
||||
别名: ml
|
||||
描述: MiaoLobby - Minecraft 服务器插件父项目
|
||||
权限: MiaoLobby.reload
|
||||
用法: 使用/MiaoLobby help 查看帮助!
|
||||
```
|
||||
|
||||
### 插件权限
|
||||
|
||||
```
|
||||
插件注册权限:
|
||||
- MiaoLobby.default - MiaoLobby 默认权限!
|
||||
- MiaoLobby.admin - MiaoLobby 管理员权限!
|
||||
- MiaoLobby.reload - 重新载入插件!
|
||||
```
|
||||
|
||||
### 插件下载
|
||||
|
||||
[attach]1802025[/attach]
|
||||
|
||||
### Miao系列插件
|
||||
|
||||
- [[经济]MiaoReward —— 喵式奖励 让玩家看广告为服务器提供收入吧[1.7.10+全版本]](https://www.mcbbs.net/thread-1121423-1-1.html)
|
||||
- [[编程]MiaoBlockly —— 喵式积木 用简单的积木来写插件吧[1.12.2+全版本]](https://www.mcbbs.net/thread-1129411-1-1.html)
|
||||
- [[编程]MiaoConsole —— 喵式终端 通过MC端口直接控制服务器 调试插件[1.12.2+全版本]](https://www.mcbbs.net/thread-1129227-1-1.html)
|
||||
- [[管理]MiaoBind —— 喵式绑定 兼容SoulBound的绑定插件 支持自定义关键词[1.7+全版本]](https://www.mcbbs.net/thread-922072-1-1.html)
|
||||
- [[信息]MiaoBoard —— 喵式记分板 自定义动态记分板[1.7+全版本]](https://www.mcbbs.net/thread-631482-1-1.html)
|
||||
- [[聊天]MiaoChat —— 喵式聊天 多功能自定义聊天格式 新增支持跨服[1.7.10+全版本]](https://www.mcbbs.net/thread-631240-1-1.html)
|
||||
- [[菜单]MiaoMenu —— 喵式菜单 强大的自定义菜单 支持多种自定义操作[1.7+全版本]](https://www.mcbbs.net/thread-860047-1-1.html)
|
||||
- [[管理]YUM —— 全能的服务器插件管理工具 全自动安装插件 更新插件[1.7.2+全版本]](https://www.mcbbs.net/thread-701333-1-1.html)
|
||||
|
||||
#### 本插件所用所有代码均为原创,不存在借用/抄袭等行为
|
||||
33
packages/plugins/docs/MiaoMail.md
Normal file
33
packages/plugins/docs/MiaoMail.md
Normal file
@@ -0,0 +1,33 @@
|
||||
private static void injectInventoryAddItemMethod() {
|
||||
System.out.println("Injectoring Mail...");
|
||||
try {
|
||||
ClassPool pool = ClassPool.getDefault();
|
||||
String inventoryClassName = Bukkit.getServer().getClass().getPackage().getName() + ".inventory.CraftInventory";
|
||||
File classFile = new File(URLDecoder.decode(InventoryDragEvent.class.getProtectionDomain().getCodeSource().getLocation().getPath().split("!")[0], "UTF-8"));
|
||||
pool.appendClassPath(classFile.getPath());
|
||||
classFile = new File(URLDecoder.decode(MailPlugin.class.getProtectionDomain().getCodeSource().getLocation().getPath().split("!")[0], "UTF-8"));
|
||||
pool.appendClassPath(classFile.getPath());
|
||||
CtClass inventoryClass = pool.get(inventoryClassName);
|
||||
CtMethod addItemMethod = inventoryClass.getDeclaredMethod("addItem");
|
||||
|
||||
addItemMethod.insertAfter(
|
||||
"if(this instanceof org.bukkit.inventory.PlayerInventory){"
|
||||
+ " final org.bukkit.inventory.PlayerInventory playerInventory = (org.bukkit.inventory.PlayerInventory)this;"
|
||||
+ " final org.bukkit.entity.Player player = (org.bukkit.entity.Player)playerInventory.getHolder();"
|
||||
+ " if(!$_.values().isEmpty()){"
|
||||
+ " org.bukkit.event.inventory.InventoryDragEvent event = new org.bukkit.event.inventory.InventoryDragEvent(player.getOpenInventory(), new org.bukkit.inventory.ItemStack(org.bukkit.Material.AIR), new org.bukkit.inventory.ItemStack(org.bukkit.Material.AIR), true, $_);"
|
||||
+ " event.setCancelled(true);"
|
||||
+ " event.setResult(org.bukkit.event.Event.Result.DENY);"
|
||||
+ " org.bukkit.Bukkit.getPluginManager().callEvent(event);"
|
||||
+ " }"
|
||||
+ "}");
|
||||
try {
|
||||
inventoryClass.toClass();
|
||||
System.out.println("Injected");
|
||||
} catch (Exception e) {
|
||||
System.out.println("Inject fail, Please restart server");
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
22
packages/plugins/docs/MiaoNashorn.md
Normal file
22
packages/plugins/docs/MiaoNashorn.md
Normal file
@@ -0,0 +1,22 @@
|
||||
[综合|前置]MiaoNashorn —— 喵式犀牛引擎 用于 Java14+ 自动安装脚本引擎[全版本]
|
||||
# MiaoNashorn
|
||||
|
||||
## 插件介绍
|
||||
|
||||
> 在Java14+环境下缺少Nashorn 本插件用于自动下载并且加载Nashorn依赖
|
||||
> 自动从云端下载依赖 插件仅 8kb 大小
|
||||
|
||||
## 可用于下列插件 在 Java14+ 环境运行
|
||||
|
||||
- PlaceholderAPI 的 Javascript 扩展
|
||||
- MiaoMenu
|
||||
- MiaoScript
|
||||
- TrMenu
|
||||
- AttributePlus
|
||||
- 等其他任何需要 Nashorn 引擎的插件
|
||||
|
||||
## 下载地址
|
||||
|
||||
[attach]1834431[/attach]
|
||||
|
||||
#### 本插件所用所有代码均为原创,不存在借用/抄袭等行为
|
||||
57
packages/plugins/docs/MiaoPay.md
Normal file
57
packages/plugins/docs/MiaoPay.md
Normal file
@@ -0,0 +1,57 @@
|
||||
# MiaoPay
|
||||
|
||||
## 安装方式
|
||||
|
||||
先按照帖子
|
||||
|
||||
### 网关
|
||||
|
||||
- https://pay.yumc.pw/api
|
||||
|
||||
### 请求规范
|
||||
|
||||
- 除业务参数外 每个请求必须包含下列系统参数
|
||||
- 系统级参数
|
||||
- 应用 ID `appid`
|
||||
- 时间戳 `timestamp` 单位: 秒
|
||||
- 随机字符串 `nonce` 32 位以内的随机字符串
|
||||
|
||||
#### 签名生成
|
||||
|
||||
- 对参数按照字典升序排列
|
||||
- 拼接成查询字符串后追加 key=secret
|
||||
- 获得字符串的 MD5 值 并且转换成大写
|
||||
|
||||
```php
|
||||
ksort($data);
|
||||
$signStr = urldecode(http_build_query($data)).'&key='.\getAppSecret();
|
||||
return strtoupper(md5($signStr));
|
||||
```
|
||||
|
||||
### 相关接口
|
||||
|
||||
#### 创建订单
|
||||
|
||||
- METHOD: /create
|
||||
- PARAM:
|
||||
- 订单标题 `subject` 必填
|
||||
- 订单金额 `amount` 必填 单位: 元
|
||||
- 用户名 `username` 选填
|
||||
- 用户唯一 ID `unionId` 选填
|
||||
- 外部订单 ID `outOrderId` 选填 用于三方系统
|
||||
- 通知地址 `notifyUrl` 选填 用于三方系统回调
|
||||
- RETURN:
|
||||
- 订单ID `order_id`
|
||||
- 订单金额 `amount`
|
||||
- 订单支付地址 `url`
|
||||
|
||||
#### 查询订单
|
||||
|
||||
- METHOD: /query
|
||||
- PARAM:
|
||||
- 订单ID `subject` 必填
|
||||
- 订单金额 `amount` 必填 单位: 元
|
||||
- 用户名 `username` 选填
|
||||
- 用户唯一 ID `uuid` 选填
|
||||
- RETURN:
|
||||
- 订单数据
|
||||
@@ -1,5 +1,6 @@
|
||||
### Miao系列插件
|
||||
|
||||
- [[综合|前置]MiaoLink —— 喵式映射 用于无公网环境的自动化端口映射[全版本]](https://www.mcbbs.net/thread-1121423-1-1.html)
|
||||
- [[经济]MiaoReward —— 喵式奖励 让玩家看广告为服务器提供收入吧[1.7.10+全版本]](https://www.mcbbs.net/thread-1121423-1-1.html)
|
||||
- [[编程]MiaoBlockly —— 喵式积木 用简单的积木来写插件吧[1.12.2+全版本]](https://www.mcbbs.net/thread-1129411-1-1.html)
|
||||
- [[编程]MiaoConsole —— 喵式终端 通过MC端口直接控制服务器 调试插件[1.12.2+全版本]](https://www.mcbbs.net/thread-1129227-1-1.html)
|
||||
|
||||
37
packages/plugins/docs/MiaoRGBSupport.md
Normal file
37
packages/plugins/docs/MiaoRGBSupport.md
Normal file
@@ -0,0 +1,37 @@
|
||||
[综合|信息]MiaoRGBSupport —— 喵式RGB 支持任意插件的RGB展示[1.16+]
|
||||
|
||||
# MiaoRGBSupport
|
||||
|
||||
## 插件简介
|
||||
- 支持1.16+彩色字体展示
|
||||
- 兼容原版聊天以及任意聊天插件
|
||||
- 兼容任意记分板插件
|
||||
- 仅需安装插件配置权限即可生效
|
||||
|
||||
## 插件权限
|
||||
- `MiaoRGBSupport.color` 默认玩家没权限 需要手动添加
|
||||
|
||||
## 安装方式
|
||||
- 本插件基于 MiaoScript 开发
|
||||
- 请先安装 MiaoScript [坛内地址](https://www.mcbbs.net/thread-774401-1-1.html)
|
||||
- 然后安装 ProtocolLib 自己解决
|
||||
- 执行 `/mspm install MiaoChatRGBSupport`
|
||||
|
||||
## 使用方式
|
||||
- 颜色格式 `#FFFFFF` 标准 HTML 的色彩格式
|
||||
|
||||
### 聊天插件配置
|
||||
- 例如 配置 `MiaoChat` 的 `format.yml`
|
||||
- 
|
||||
- 配置完成后 重载插件
|
||||
|
||||
### 聊天时使用
|
||||
- 聊天时 直接输入颜色代码即可
|
||||
- 
|
||||
|
||||
## 星球特供版本
|
||||
- 
|
||||
- 此版本为知识星球特供
|
||||
- 支持 聊天和记分板
|
||||
- 支持 彩虹字
|
||||
- 详情加群 650545561
|
||||
114
packages/plugins/docs/MiaoRebate.md
Normal file
114
packages/plugins/docs/MiaoRebate.md
Normal file
@@ -0,0 +1,114 @@
|
||||
# MiaoRebate
|
||||
|
||||
## 插件简介
|
||||
|
||||
- 还在为服务器收入不足而倒闭烦恼嘛
|
||||
- 还在为肝帝不氪金而烦恼嘛
|
||||
- 快来接入 喵式返利
|
||||
- 饿了么美团战略合作 玩家点外卖 腐竹拿返利 增加服务器收入
|
||||
|
||||
### 先来一张 1 块钱吃一餐的图
|
||||
|
||||

|
||||
|
||||
### 再来一张红包兑换的图
|
||||
|
||||

|
||||
|
||||
### 限时活动
|
||||
|
||||
> 即日起 至 7 月 15 日 额外奖励活动
|
||||
> 每满 50 人关注公众号 额外奖励 50 元 上不封顶 (当月取消关注不算)
|
||||
> 请绑定后加 QQ 群 1055983539 参加活动
|
||||
|
||||
## 插件展示
|
||||
|
||||
> 多图预警 折叠了
|
||||
|
||||
`[spoiler]`
|
||||
|
||||
- 命令帮助
|
||||
- 扫码绑定
|
||||
- 个人信息
|
||||
- 兑换列表
|
||||
|
||||
`[/spoiler]`
|
||||
|
||||
## 插件命令
|
||||
|
||||
```
|
||||
>mre help
|
||||
[外卖系统]====== [外卖系统] 帮助菜单 ======
|
||||
[外卖系统]/mre bind 绑定账号
|
||||
[外卖系统]/mre draw <兑换数量> 兑换点券
|
||||
[外卖系统]由于您是管理员 以为您展示额外命令
|
||||
[外卖系统]/mrd bind server 绑定服务器
|
||||
```
|
||||
|
||||
## 接下来就是赚钱的操作
|
||||
|
||||
### 服务器准备工作
|
||||
|
||||
- 本插件依赖于 `MiaoReward` 请前往 [站内帖子](https://www.mcbbs.net/thread-1121423-1-1.html) 完成安装
|
||||
- 执行 `/mspm install MiaoRebate` 安装 MiaoRebate 脚本插件
|
||||
- 完成安装
|
||||
|
||||
### 绑定服务器
|
||||
|
||||
- 执行 `/mre bind server`
|
||||
- 使用绿色儿的那个 APP 扫码 完成绑定
|
||||
|
||||
### 玩家绑定账号
|
||||
|
||||
- 执行 `/mre bind`
|
||||
- 使用绿色儿的那个 APP 扫码 完成绑定
|
||||
|
||||
## 使用说明
|
||||
|
||||
- 玩家可以通过下列方式获取圈币
|
||||
- 进入公众号 领取红包
|
||||
- 小程序直接点餐或到饿了么/美团 APP 点餐
|
||||
- 点餐后 发送订单号 兑换奖励
|
||||
- 返利额度约为实付金额的 `1%-3%` 左右
|
||||
- 获得的圈币 在服务器使用 `/mre draw 兑换金额`
|
||||
- 腐竹获得圈币后 在公众号兑换成红包即可
|
||||
|
||||
## PAPI 兼容
|
||||
|
||||
- 目前暂不支持 PAPI 变量 后续会支持
|
||||
|
||||
## 配置文件
|
||||
|
||||
```yml
|
||||
# 提示前缀
|
||||
prefix: §6[§b外卖系统§6]§r
|
||||
# 用于检查货币的变量
|
||||
check: "%playerpoints_points%"
|
||||
# 用于充值货币的命令
|
||||
command: points give %player_name% %amount%
|
||||
# 兑换比例 圈币 对应多少 货币
|
||||
ratio: 1
|
||||
# 货币名称
|
||||
coinName: 点券
|
||||
# 进服提示
|
||||
joinTip: true
|
||||
# 绑定数据(请勿手动修改 绑定后会自动填写数据)
|
||||
owner:
|
||||
userid:
|
||||
ccid:
|
||||
openid:
|
||||
```
|
||||
|
||||
## 插件源码
|
||||
|
||||
- [MiaoScript 包管理中心](https://git.yumc.pw/circlecloud/ms/src/branch/master/packages/plugins/src/MiaoRebate.ts)
|
||||
|
||||
## 更新日志
|
||||
|
||||
- 暂无
|
||||
|
||||
## Roadmap
|
||||
|
||||
- 绑定服务器(已完成)
|
||||
- 绑定玩家(已完成)
|
||||
- 兑换圈币(已完成)
|
||||
112
packages/plugins/docs/MiaoReport.md
Normal file
112
packages/plugins/docs/MiaoReport.md
Normal file
@@ -0,0 +1,112 @@
|
||||
# MiaoReport(喵式报告)
|
||||
|
||||
## 插件简介
|
||||
|
||||
- 本插件只有一个功能 就是上报当前服务器信息和运行日志到 Ubuntu Paste
|
||||
- 适用于哪些 有问题 但是不知道怎么提供报错的用户
|
||||
|
||||
## 插件命令
|
||||
|
||||
- 只有一个 `/mbrp`
|
||||
- 并且只有 OP 权限用户才可以执行
|
||||
|
||||
## 功能预览
|
||||
|
||||
- 报告地址 https://paste.ubuntu.com/p/mVjJtb6Xcg/
|
||||
- 报告详情(已删除无用信息 并不是完整报告)
|
||||
|
||||
```
|
||||
当前报告由 MiaoBugReport(0.0.1) 生成 作者 MiaoWoo 官网 https://w.yumc.pw
|
||||
生成时间: 2021-02-13 21:52:54
|
||||
============================== 以下为本次报告内容 ==============================
|
||||
服务器版本: git-Spigot-dcd1643-e60fc34 (MC: 1.12.2)
|
||||
Bukkit版本: 1.12.2-R0.1-SNAPSHOT
|
||||
============================== 运行数据 ==============================
|
||||
运行时间: 1 小时 25 分 48 秒
|
||||
CPU 核心: 16
|
||||
最大内存: 7282 MB
|
||||
分配内存: 626 MB
|
||||
空闲内存: 307 MB
|
||||
============================== 数据统计 ==============================
|
||||
世界列表:
|
||||
- world(NORMAL)
|
||||
- 已加载区块: 272
|
||||
- 实体数量: 146
|
||||
- Tile数量: 10
|
||||
- 玩家数量: 0
|
||||
- world_nether(NETHER)
|
||||
- 已加载区块: 0
|
||||
- 实体数量: 0
|
||||
- Tile数量: 0
|
||||
- 玩家数量: 0
|
||||
- world_the_end(THE_END)
|
||||
- 已加载区块: 0
|
||||
- 实体数量: 0
|
||||
- Tile数量: 0
|
||||
- 玩家数量: 0
|
||||
在线玩家: Mr_jtb
|
||||
插件列表: MiaoBugReport(0.0.1)
|
||||
============================== 服务器线程堆栈 ==============================
|
||||
"Spigot Metrics Thread" daemon prio=5 id=32
|
||||
java.lang.Thread.State: TIMED_WAITING
|
||||
at java.lang.Object.wait(Native Method)
|
||||
at java.util.TimerThread.mainLoop(Unknown Source)
|
||||
at java.util.TimerThread.run(Unknown Source)
|
||||
|
||||
"Spigot Watchdog Thread" prio=5 id=31
|
||||
java.lang.Thread.State: TIMED_WAITING
|
||||
at java.lang.Thread.sleep(Native Method)
|
||||
at org.spigotmc.WatchdogThread.run(WatchdogThread.java:92)
|
||||
|
||||
"Server thread" prio=5 id=26
|
||||
java.lang.Thread.State: TIMED_WAITING
|
||||
at java.lang.Thread.sleep(Native Method)
|
||||
at net.minecraft.server.v1_12_R1.MinecraftServer.run(MinecraftServer.java:560)
|
||||
at java.lang.Thread.run(Unknown Source)
|
||||
|
||||
"Server console handler" daemon prio=5 id=29
|
||||
java.lang.Thread.State: RUNNABLE
|
||||
at org.fusesource.jansi.internal.Kernel32.ReadConsoleInputW(Native Method)
|
||||
at org.fusesource.jansi.internal.Kernel32.readConsoleInputHelper(Kernel32.java:761)
|
||||
at org.fusesource.jansi.internal.Kernel32.readConsoleKeyInput(Kernel32.java:794)
|
||||
at org.fusesource.jansi.internal.WindowsSupport.readConsoleInput(WindowsSupport.java:97)
|
||||
at org.bukkit.craftbukkit.libs.jline.WindowsTerminal.readConsoleInput(WindowsTerminal.java:215)
|
||||
at org.bukkit.craftbukkit.libs.jline.WindowsTerminal.access$000(WindowsTerminal.java:55)
|
||||
at org.bukkit.craftbukkit.libs.jline.WindowsTerminal$1.read(WindowsTerminal.java:157)
|
||||
at org.bukkit.craftbukkit.libs.jline.internal.NonBlockingInputStream.read(NonBlockingInputStream.java:169)
|
||||
at org.bukkit.craftbukkit.libs.jline.internal.NonBlockingInputStream.read(NonBlockingInputStream.java:137)
|
||||
at org.bukkit.craftbukkit.libs.jline.internal.NonBlockingInputStream.read(NonBlockingInputStream.java:246)
|
||||
at org.bukkit.craftbukkit.libs.jline.internal.InputStreamReader.read(InputStreamReader.java:261)
|
||||
at org.bukkit.craftbukkit.libs.jline.internal.InputStreamReader.read(InputStreamReader.java:198)
|
||||
at org.bukkit.craftbukkit.libs.jline.console.ConsoleReader.readCharacter(ConsoleReader.java:2145)
|
||||
at org.bukkit.craftbukkit.libs.jline.console.ConsoleReader.readLine(ConsoleReader.java:2349)
|
||||
at net.minecraft.server.v1_12_R1.DedicatedServer$2.run(DedicatedServer.java:85)
|
||||
|
||||
"Netty Server IO #0" daemon prio=5 id=34
|
||||
java.lang.Thread.State: RUNNABLE
|
||||
at sun.nio.ch.WindowsSelectorImpl$SubSelector.poll0(Native Method)
|
||||
at sun.nio.ch.WindowsSelectorImpl$SubSelector.poll(Unknown Source)
|
||||
at sun.nio.ch.WindowsSelectorImpl$SubSelector.access$400(Unknown Source)
|
||||
at sun.nio.ch.WindowsSelectorImpl.doSelect(Unknown Source)
|
||||
at sun.nio.ch.SelectorImpl.lockAndDoSelect(Unknown Source)
|
||||
at sun.nio.ch.SelectorImpl.select(Unknown Source)
|
||||
at io.netty.channel.nio.SelectedSelectionKeySetSelector.select(SelectedSelectionKeySetSelector.java:62)
|
||||
at io.netty.channel.nio.NioEventLoop.select(NioEventLoop.java:753)
|
||||
at io.netty.channel.nio.NioEventLoop.run(NioEventLoop.java:409)
|
||||
at io.netty.util.concurrent.SingleThreadEventExecutor$5.run(SingleThreadEventExecutor.java:884)
|
||||
at java.lang.Thread.run(Unknown Source)
|
||||
...忽略其他线程信息
|
||||
============================== 本次运行日志 ==============================
|
||||
[20:27:18] [Server thread/INFO]: Starting minecraft server version 1.12.2
|
||||
...忽略启动日志
|
||||
[20:27:27] [Server thread/INFO]: [MiaoBugReport] Enabling MiaoBugReport v0.0.1
|
||||
[20:27:28] [Server thread/INFO]: Server permissions file permissions.yml is empty, ignoring it
|
||||
[20:27:28] [Server thread/INFO]: Done (8.205s)! For help, type "help" or "?"
|
||||
[20:27:53] [Craft Scheduler Thread - 0/INFO]: [0;33;22m[[0;36;1mMiaoBugReport[0;33;22m][m [0;32;1m正在生成日志数据...[m
|
||||
[20:27:55] [Craft Scheduler Thread - 0/INFO]: [0;33;22m[[0;36;1mMiaoBugReport[0;33;22m][m [0;32;1m正在上报数据至UbuntuPaste...[m
|
||||
[20:27:55] [Craft Scheduler Thread - 0/INFO]: [0;33;22m[[0;36;1mMiaoBugReport[0;33;22m][m [0;32;1m数据上报完成 您可以分享地址给他人[m
|
||||
============================== 报告结束 ==============================
|
||||
```
|
||||
|
||||
## 下载地址
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"private": true,
|
||||
"name": "@ccms/plugins",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript plugins package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -26,16 +26,16 @@
|
||||
"@javatypes/spring-data-redis": "^0.0.3",
|
||||
"@javatypes/spring-web": "^0.0.3",
|
||||
"@javatypes/tomcat": "^0.0.3",
|
||||
"@types/crypto-js": "^4.0.1",
|
||||
"@types/crypto-js": "^4.0.2",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@babel/standalone": "^7.12.9",
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0",
|
||||
"@ccms/plugin": "^0.13.0",
|
||||
"crypto-js": "^4.0.0"
|
||||
"@babel/standalone": "^7.15.0",
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0",
|
||||
"@ccms/plugin": "^0.17.0",
|
||||
"crypto-js": "^4.1.1"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -50,7 +50,7 @@ class MiaoMessage {
|
||||
}
|
||||
}
|
||||
|
||||
@plugin({ version: '1.0.1', author: 'MiaoWoo', source: __filename })
|
||||
@plugin({ version: '1.1.0', author: 'MiaoWoo', nativeDepends: ['PlaceholderAPI'], source: __filename })
|
||||
export class MiaoChat extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private Server: server.Server
|
||||
|
||||
109
packages/plugins/src/MiaoChatRGBSupport.ts
Normal file
109
packages/plugins/src/MiaoChatRGBSupport.ts
Normal file
@@ -0,0 +1,109 @@
|
||||
import { task, server, constants } from "@ccms/api"
|
||||
import { Autowired, JSClass } from "@ccms/container"
|
||||
import { interfaces, JSPlugin } from "@ccms/plugin"
|
||||
|
||||
let createPacketAdapterFunction = eval(`
|
||||
function(cls, plugin, type, onPacketSending){
|
||||
return new cls(plugin, type) {
|
||||
onPacketSending: onPacketSending
|
||||
}
|
||||
}
|
||||
`)
|
||||
const ChatColor = Java.type('net.md_5.bungee.api.ChatColor')
|
||||
const Pattern = Java.type('java.util.regex.Pattern')
|
||||
|
||||
@JSPlugin({ prefix: 'MCRS', version: '1.0.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], nativeDepends: ['ProtocolLib'], source: __filename })
|
||||
export class MiaoChatRGBSupport extends interfaces.Plugin {
|
||||
private supportRGB: boolean = false
|
||||
// 用于匹配 '#FFFFFF' 颜色格式
|
||||
private RGBCOLOR_PATTERN = Pattern.compile("(#[a-fA-F0-9]{6}?)([^#?]*)");
|
||||
|
||||
@JSClass('com.comphenix.protocol.events.PacketAdapter')
|
||||
private PacketAdapter: any
|
||||
@JSClass('com.comphenix.protocol.PacketType')
|
||||
private PacketType: any
|
||||
@JSClass('com.comphenix.protocol.ProtocolLibrary')
|
||||
private ProtocolLibrary: any
|
||||
|
||||
private adapter: any
|
||||
|
||||
load() {
|
||||
try {
|
||||
ChatColor.of('#FFFFFF').toString()
|
||||
this.supportRGB = true
|
||||
this.logger.console('§a检测到兼容RGB的服务端 已启动相关支持...')
|
||||
} catch (error) {
|
||||
this.logger.console('§c当前服务端不支持RGB色彩 Error: ' + error)
|
||||
}
|
||||
}
|
||||
|
||||
enable() {
|
||||
if (this.supportRGB) {
|
||||
this.initPacketAdapter()
|
||||
}
|
||||
}
|
||||
|
||||
disable() {
|
||||
if (this.supportRGB) {
|
||||
this.ProtocolLibrary.getProtocolManager().removePacketListener(this.adapter)
|
||||
}
|
||||
}
|
||||
|
||||
createPacketAdapter(onPacketSending: (event) => void) {
|
||||
return createPacketAdapterFunction(this.PacketAdapter, base.getInstance(), [this.PacketType.Play.Server.CHAT], onPacketSending)
|
||||
}
|
||||
|
||||
colorJson(jsonObj) {
|
||||
if (jsonObj.extra && jsonObj.extra.length) {
|
||||
for (const extra of jsonObj.extra) {
|
||||
this.colorJson(extra)
|
||||
}
|
||||
}
|
||||
let text: string = jsonObj.text
|
||||
var matcher = this.RGBCOLOR_PATTERN.matcher(text)
|
||||
let colors = []
|
||||
let texts = []
|
||||
let lastStart = 0
|
||||
while (matcher.find()) {
|
||||
if (lastStart == 0) {
|
||||
texts.push(text.substr(lastStart, matcher.start()))
|
||||
lastStart = matcher.end()
|
||||
}
|
||||
colors.push(matcher.group(1))
|
||||
texts.push(matcher.group(2))
|
||||
}
|
||||
if (colors.length) {
|
||||
jsonObj.text = ''
|
||||
let extras = []
|
||||
let firstText = texts.shift()
|
||||
if (firstText) { extras.push({ text: firstText }) }
|
||||
texts.forEach((value, index) => {
|
||||
extras.push({
|
||||
text: value,
|
||||
color: colors[index]
|
||||
})
|
||||
})
|
||||
if (jsonObj.extra) {
|
||||
jsonObj.extra = extras.concat(jsonObj.extra)
|
||||
} else {
|
||||
jsonObj.extra = extras
|
||||
}
|
||||
}
|
||||
return jsonObj
|
||||
}
|
||||
|
||||
initPacketAdapter() {
|
||||
this.adapter = this.createPacketAdapter((event) => {
|
||||
try {
|
||||
if (!event.getPlayer().hasPermission('MiaoChatRGBSupport.color')) { return }
|
||||
let wcc = event.getPacket().getChatComponents().read(0)
|
||||
if (wcc == null) { return }
|
||||
wcc.setJson(JSON.stringify(this.colorJson(JSON.parse(wcc.getJson()))))
|
||||
event.getPacket().getChatComponents().writeSafely(0, wcc)
|
||||
} catch (error) {
|
||||
console.ex(error)
|
||||
}
|
||||
})
|
||||
this.ProtocolLibrary.getProtocolManager().addPacketListener(this.adapter)
|
||||
}
|
||||
}
|
||||
@@ -3,10 +3,12 @@
|
||||
import { plugin as pluginApi, server, task, constants, command } from '@ccms/api'
|
||||
import { plugin, interfaces, cmd, tab, enable, config, disable, PluginConfig } from '@ccms/plugin'
|
||||
import { ContainerInstance, Container, Autowired } from '@ccms/container'
|
||||
import io, { Server as SocketIOServer, Socket as SocketIOSocket } from '@ccms/websocket'
|
||||
import io from '@ccms/websocket'
|
||||
import * as fs from '@ccms/common/dist/fs'
|
||||
import * as reflect from '@ccms/common/dist/reflect'
|
||||
|
||||
import type { Namespace, Server as SocketIOServer, Socket as SocketIOSocket } from '@ccms/websocket'
|
||||
|
||||
const suffixMap = {
|
||||
ts: 'typescript',
|
||||
js: 'javascript',
|
||||
@@ -19,7 +21,7 @@ let help = [
|
||||
'§6/mconsole §areload §6- §3重载插件',
|
||||
]
|
||||
|
||||
@plugin({ prefix: 'Console', version: '1.1.0', author: 'MiaoWoo', servers: ['!nukkit'], source: __filename })
|
||||
@plugin({ prefix: 'Console', version: '1.1.1', author: 'MiaoWoo', servers: ['!nukkit'], source: __filename })
|
||||
export class MiaoConsole extends interfaces.Plugin {
|
||||
@Autowired(ContainerInstance)
|
||||
private container: Container
|
||||
@@ -43,6 +45,7 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
private appender: any
|
||||
private handler: any
|
||||
private babel: any
|
||||
private downgrade = true
|
||||
|
||||
private logCache: string[] = []
|
||||
|
||||
@@ -57,19 +60,25 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
this.token = Java.type('java.util.UUID').randomUUID().toString()
|
||||
this.logger.console(`§6已生成随机Token: §3${this.token} §c重启后或重新生成后失效!`)
|
||||
}
|
||||
process.on('message', (msg) => {
|
||||
process.on('message', (msg: string) => {
|
||||
this.logCache.push(msg)
|
||||
if (this.logCache.length > 30) {
|
||||
this.logCache = this.logCache.slice(this.logCache.length - 30, this.logCache.length)
|
||||
if (this.logCache.length > 100) {
|
||||
this.logCache = this.logCache.slice(this.logCache.length - 100, this.logCache.length)
|
||||
}
|
||||
})
|
||||
this.task.create(() => {
|
||||
if (!this.babel) {
|
||||
this.logger.console('§3脚本 Babel 引擎初始化中 请稍候...')
|
||||
let startTime = Date.now()
|
||||
this.babel = require('@babel/standalone')
|
||||
this.compileCode(`() => console.log("Babel ready!")`)
|
||||
this.logger.console(`§3脚本 Babel 引擎初始化完毕 耗时 §a${Date.now() - startTime}ms...`)
|
||||
if (!this.babel && this.serverType != constants.ServerType.Bungee) {
|
||||
try {
|
||||
this.logger.console('§3脚本 Babel 引擎初始化中 请稍候...')
|
||||
let startTime = Date.now()
|
||||
this.babel = require('@babel/standalone')
|
||||
this.compileCode(`() => console.log("Babel ready!")`)
|
||||
this.logger.console(`§3脚本 Babel 引擎初始化完毕 耗时 §a${Date.now() - startTime}ms...`)
|
||||
this.downgrade = false
|
||||
} catch (error) {
|
||||
this.logger.console(`§c脚本 Babel 引擎初始化失败 将无法使用ES5以上语法!`)
|
||||
console.ex(error)
|
||||
}
|
||||
}
|
||||
}).async().submit()
|
||||
}
|
||||
@@ -155,7 +164,11 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
if (this.rootLogger) {
|
||||
let AbstractHandler = Java.type('java.util.logging.Handler')
|
||||
let ProxyHandler = Java.extend(AbstractHandler, {
|
||||
publish: (record) => process.emit('message', record.getMessage()),
|
||||
publish: (record) => {
|
||||
if (record.getLevel().intValue() > 500) {
|
||||
process.emit('message', record.getMessage())
|
||||
}
|
||||
},
|
||||
flush: () => { },
|
||||
close: () => { }
|
||||
})
|
||||
@@ -184,6 +197,7 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
disable() {
|
||||
if (this.socketIOServer) {
|
||||
this.socketIOServer.close()
|
||||
process.removeAllListeners('websocket.create')
|
||||
process.removeAllListeners('message')
|
||||
}
|
||||
if (this.container.isBound(io.Instance)) {
|
||||
@@ -223,30 +237,46 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
this.socketIOServer = io(this.instance, {
|
||||
path: '/ws',
|
||||
root: fs.concat(root, 'wwwroot')
|
||||
})
|
||||
} as any)
|
||||
this.container.bind(io.Instance).toConstantValue(this.socketIOServer)
|
||||
process.emit('websocket.create', this.socketIOServer)
|
||||
}
|
||||
|
||||
registryWebSocketNamespace(namespace: string, initialization: (namespace: Namespace) => void) {
|
||||
if (this.socketIOServer) {
|
||||
initialization(this.socketIOServer.of(namespace))
|
||||
} else {
|
||||
process.once('websocket.create', (server) => initialization(server.of(namespace)))
|
||||
}
|
||||
}
|
||||
|
||||
checkWebSocketClient(client: SocketIOSocket) {
|
||||
if (!this.token) {
|
||||
return this.notifyDisconnect(client, `§6客户端 §b${client.id} §a请求连接 §4服务器尚未设置 Token 无法连接!`)
|
||||
}
|
||||
if (this.token != client.handshake.query.token) {
|
||||
return this.notifyDisconnect(client, `§6客户端 §b${client.id} §c无效请求 §4请提供正确Token后再次连接!`)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
private notifyDisconnect(client: SocketIOSocket, reason: string) {
|
||||
this.logger.console(reason)
|
||||
client.emit('unauthorized', () => client.disconnect(true))
|
||||
setTimeout(() => { if (client.connected) { client.disconnect(true) } }, 5)
|
||||
return false
|
||||
}
|
||||
|
||||
startSocketIOServer() {
|
||||
let namespace = this.socketIOServer.of('/MiaoConsole')
|
||||
process.on('message', (msg) => namespace.emit('log', msg))
|
||||
namespace.on('connection', (client: SocketIOSocket) => {
|
||||
if (!this.token) {
|
||||
this.logger.console(`§6客户端 §b${client.id} §a请求连接 §4服务器尚未设置 Token 无法连接!`)
|
||||
client.emit('unauthorized', () => client.disconnect(true))
|
||||
return
|
||||
if (this.checkWebSocketClient(client)) {
|
||||
this.initWebSocketClient(client)
|
||||
this.logCache.forEach(msg => client.emit('log', msg))
|
||||
this.logger.console(`§6客户端 §b${client.id} §a新建连接 ${this.rootLogger ? '启动日志转发' : '§4转发日志启动失败'}...`)
|
||||
}
|
||||
if (this.token != client.handshake.query.token) {
|
||||
this.logger.console(`§6客户端 §b${client.id} §c无效请求 §4请提供正确Token后再次连接!`)
|
||||
client.emit('unauthorized', () => client.disconnect(true))
|
||||
return
|
||||
}
|
||||
this.initWebSocketClient(client)
|
||||
this.logCache.forEach(msg => client.emit('log', msg))
|
||||
this.logger.console(`§6客户端 §b${client.id} §a新建连接 ${this.rootLogger ? '启动日志转发' : '§4转发日志启动失败'}...`)
|
||||
})
|
||||
process.emit('websocket.start', this.socketIOServer)
|
||||
}
|
||||
|
||||
private initWebSocketClient(client: SocketIOSocket) {
|
||||
@@ -332,6 +362,17 @@ export class MiaoConsole extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
private compileCode(code: string) {
|
||||
if (!this.downgrade && this.babel?.transform) {
|
||||
code = this.babel.transform(code, {
|
||||
filename: 'miaoconsole-temp.ts',
|
||||
presets: ['typescript', 'es2015'],
|
||||
plugins: [
|
||||
['proposal-decorators', { legacy: true }],
|
||||
'transform-runtime'
|
||||
],
|
||||
sourceMaps: "inline"
|
||||
}).code
|
||||
}
|
||||
return `var api = require('@ccms/api');
|
||||
if (this.serverType == "spring") {
|
||||
var dbm = container.get(api.database.DataBaseManager)
|
||||
@@ -339,15 +380,7 @@ if (this.serverType == "spring") {
|
||||
var bf = base.getInstance().getAutowireCapableBeanFactory()
|
||||
}
|
||||
var startTime = Date.now()
|
||||
var result = eval(${JSON.stringify(this.babel.transform(code, {
|
||||
filename: 'miaoconsole-temp.ts',
|
||||
presets: ['typescript', 'es2015'],
|
||||
plugins: [
|
||||
['proposal-decorators', { legacy: true }],
|
||||
'transform-runtime'
|
||||
],
|
||||
sourceMaps: "inline"
|
||||
}).code)});
|
||||
var result = eval(${JSON.stringify(code)});
|
||||
return '§3代码执行完成 耗时 §e' + (Date.now() - startTime) + 'ms §a返回结果: §r'+ result`
|
||||
}
|
||||
}
|
||||
|
||||
344
packages/plugins/src/MiaoDashboard.ts
Normal file
344
packages/plugins/src/MiaoDashboard.ts
Normal file
@@ -0,0 +1,344 @@
|
||||
/// <reference types="@javatypes/bungee-api" />
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
|
||||
import { constants, plugin, server, task } from '@ccms/api'
|
||||
import { Autowired } from '@ccms/container'
|
||||
import { Config, disable, enable, interfaces, JSPlugin, PluginConfig } from '@ccms/plugin'
|
||||
import { client } from '@ccms/websocket'
|
||||
|
||||
import http from '@ccms/common/dist/http'
|
||||
|
||||
import type { Socket as SocketIOSocket, Namespace } from '@ccms/websocket'
|
||||
import type { MiaoConsole } from './MiaoConsole'
|
||||
|
||||
const defaultConfig = {
|
||||
statistics: {
|
||||
max: 300
|
||||
}
|
||||
}
|
||||
|
||||
const defaultDataConfig = {
|
||||
server_online: "%server_online%",
|
||||
server_tps: "%server_tps_1%",
|
||||
server_ram_used: "%server_ram_used%",
|
||||
server_total_chunks: "%server_total_chunks%",
|
||||
server_total_living_entities: "%server_total_living_entities%",
|
||||
server_total_entities: "%server_total_entities%",
|
||||
}
|
||||
|
||||
@JSPlugin({ prefix: 'Dashboard', version: '1.0.0', author: 'MiaoWoo', depends: ['MiaoConsole'], source: __filename })
|
||||
export class MiaoDashboard extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private server: server.Server
|
||||
@Autowired()
|
||||
private nativePluginManager: server.NativePluginManager
|
||||
@Autowired()
|
||||
private pluginManager: plugin.PluginManager
|
||||
@Autowired()
|
||||
private taskManager: task.TaskManager
|
||||
|
||||
private namespace: Namespace
|
||||
|
||||
@Config()
|
||||
private config: PluginConfig & typeof defaultConfig = defaultConfig
|
||||
@Config()
|
||||
private dataConfig: PluginConfig & typeof defaultDataConfig = defaultDataConfig
|
||||
@Config({ autosave: true })
|
||||
private dataCache: PluginConfig & { [key: string]: { time: string, value: Number }[] } = {}
|
||||
|
||||
private statisticTimer: task.Task
|
||||
|
||||
private PlaceholderAPI: { setPlaceholders: (player: any, str: string) => string }
|
||||
|
||||
load() {
|
||||
for (const key of Object.keys(this.dataConfig)) {
|
||||
this.dataCache[key] = this.dataCache[key] ?? []
|
||||
}
|
||||
}
|
||||
|
||||
enable() {
|
||||
let consolePlugin: MiaoConsole = this.pluginManager.getPlugin('MiaoConsole') as MiaoConsole
|
||||
consolePlugin.registryWebSocketNamespace('/MiaoDashboard', (namespace: Namespace) => {
|
||||
this.namespace = namespace
|
||||
this.namespace.on('connection', (client: SocketIOSocket) => {
|
||||
if (consolePlugin.checkWebSocketClient(client)) {
|
||||
this.initWebSocketClient(client)
|
||||
this.logger.console(`§6客户端 §b${client.id} §a新建连接...`)
|
||||
}
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
@enable({ servers: [constants.ServerType.Bukkit] })
|
||||
enableBukkit() {
|
||||
this.PlaceholderAPI = base.getClass("me.clip.placeholderapi.PlaceholderAPI").static
|
||||
this.statisticTimer = this.taskManager.create(() => {
|
||||
for (const key of Object.keys(this.dataConfig)) {
|
||||
let dataArray = this.dataCache[key]
|
||||
dataArray.push({
|
||||
time: this.dateFormat('HH:MM:SS'),
|
||||
value: parseFloat(this.taskManager.callSyncMethod(() => this.PlaceholderAPI['setPlaceholders(Player,String)'](null, this.dataConfig[key]))) ?? 0
|
||||
})
|
||||
if (dataArray.length > this.config.statistics.max) {
|
||||
this.dataCache[key] = dataArray.slice(dataArray.length - this.config.statistics.max, dataArray.length)
|
||||
}
|
||||
}
|
||||
}, this).async().timer(20 * 10).submit()
|
||||
this.proxys = client.io('ws://192.168.2.25:25577/MiaoConsole?access_token=325325', {
|
||||
path: "/ws"
|
||||
})
|
||||
this.proxys.on('connect', () => {
|
||||
this.logger.info('connect')
|
||||
this.proxys.emit('type', (type) => {
|
||||
console.log('server type is ' + type)
|
||||
})
|
||||
})
|
||||
this.proxys.on('log', (msg) => {
|
||||
console.log(msg)
|
||||
})
|
||||
}
|
||||
|
||||
private proxys
|
||||
|
||||
@enable({ servers: [constants.ServerType.Bungee] })
|
||||
enbaleBungee() {
|
||||
this.proxys = client.io('ws://192.168.2.25:25565/MiaoConsole?access_token=325325', {
|
||||
path: "/ws"
|
||||
})
|
||||
this.proxys.on('connect', () => {
|
||||
this.logger.info('connect')
|
||||
this.proxys.emit('type', (type) => {
|
||||
console.log('server type is ' + type)
|
||||
})
|
||||
})
|
||||
this.proxys.on('log', (msg) => {
|
||||
console.log(msg)
|
||||
})
|
||||
}
|
||||
|
||||
@disable({ servers: [constants.ServerType.Bungee] })
|
||||
disableBungee() {
|
||||
|
||||
}
|
||||
|
||||
private dateFormat(fmt: string, date = new Date()) {
|
||||
let ret: RegExpExecArray
|
||||
const opt = {
|
||||
"Y+": date.getFullYear().toString(), // 年
|
||||
"m+": (date.getMonth() + 1).toString(), // 月
|
||||
"d+": date.getDate().toString(), // 日
|
||||
"H+": date.getHours().toString(), // 时
|
||||
"M+": date.getMinutes().toString(), // 分
|
||||
"S+": date.getSeconds().toString() // 秒
|
||||
// 有其他格式化字符需求可以继续添加,必须转化成字符串
|
||||
}
|
||||
for (let k in opt) {
|
||||
ret = new RegExp("(" + k + ")").exec(fmt)
|
||||
if (ret) {
|
||||
fmt = fmt.replace(ret[1], (ret[1].length == 1) ? (opt[k]) : (opt[k].padStart(ret[1].length, "0")))
|
||||
};
|
||||
};
|
||||
return fmt
|
||||
}
|
||||
|
||||
disable() {
|
||||
this.namespace?.close()
|
||||
this.statisticTimer?.cancel()
|
||||
}
|
||||
|
||||
private wrapper(fn, data) {
|
||||
fn({ status: 0, data })
|
||||
}
|
||||
|
||||
private initWebSocketClient(client: SocketIOSocket) {
|
||||
client.on('message', (...args) => {
|
||||
console.log(args)
|
||||
})
|
||||
this.initStatistics(client)
|
||||
this.initPlayers(client)
|
||||
this.initPlugins(client)
|
||||
this.initServices(client)
|
||||
client.on('error', (error) => {
|
||||
this.logger.console(`§6客户端 §b${client.id} §c触发异常: ${error}`)
|
||||
this.logger.error(error)
|
||||
})
|
||||
client.on('disconnect', () => {
|
||||
this.logger.console(`§6客户端 §b${client.id} §c断开连接...`)
|
||||
})
|
||||
}
|
||||
|
||||
private initStatistics(client: SocketIOSocket) {
|
||||
client.on('statistics', (data, callback) => {
|
||||
let result = {}
|
||||
data.keys.map((key: string) => result[key] = this.dataCache[key])
|
||||
this.wrapper(callback, { time: Date.now(), data: result })
|
||||
})
|
||||
}
|
||||
private initPlayers(client: SocketIOSocket) {
|
||||
client.on('players', (callback) => {
|
||||
let players = Java.from(this.server.getOnlinePlayers()).map((player: org.bukkit.entity.Player) => {
|
||||
let loc = player.getLocation()
|
||||
return {
|
||||
name: player.getName(),
|
||||
location: {
|
||||
world: loc.getWorld().getName(),
|
||||
x: loc.getX().toFixed(0),
|
||||
y: loc.getY().toFixed(0),
|
||||
z: loc.getZ().toFixed(0)
|
||||
},
|
||||
health: player.getHealth().toFixed(0),
|
||||
foodlevel: player.getFoodLevel().toFixed(0),
|
||||
gamemode: player.getGameMode().name(),
|
||||
ip: player.getAddress().getAddress().getHostAddress()
|
||||
}
|
||||
})
|
||||
this.wrapper(callback, {
|
||||
items: players
|
||||
})
|
||||
})
|
||||
this.initPlayerModify(client)
|
||||
client.on('players.heal', ({ name }, callback) => {
|
||||
let player: org.bukkit.entity.Player = this.server.getPlayer(name)
|
||||
if (!player) {
|
||||
return callback({
|
||||
status: 1,
|
||||
msg: '玩家 ' + name + ' 不在线!'
|
||||
})
|
||||
}
|
||||
player.setHealth(player.getMaxHealth())
|
||||
player.setFoodLevel(20)
|
||||
player.sendMessage('§a您已被治疗 §7- 来自MiaoDashboard')
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: '操作成功!'
|
||||
})
|
||||
})
|
||||
client.on('players.kick', ({ name, reason }, callback) => {
|
||||
let player: org.bukkit.entity.Player = this.server.getPlayer(name)
|
||||
if (!player) {
|
||||
return callback({
|
||||
status: 1,
|
||||
msg: '玩家 ' + name + ' 不在线!'
|
||||
})
|
||||
}
|
||||
player.kickPlayer(reason)
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: '操作成功!'
|
||||
})
|
||||
})
|
||||
}
|
||||
private initPlayerModify(client: SocketIOSocket) {
|
||||
client.on('players.modify.gamemode', ({ name, gamemode }, callback) => {
|
||||
let player: org.bukkit.entity.Player = this.server.getPlayer(name)
|
||||
if (!player) {
|
||||
return callback({
|
||||
status: 1,
|
||||
msg: '玩家 ' + name + ' 不在线!'
|
||||
})
|
||||
}
|
||||
let GameMode = Java.type('org.bukkit.GameMode')
|
||||
let mode = GameMode.valueOf(gamemode)
|
||||
this.taskManager.callSyncMethod(() => player.setGameMode(mode))
|
||||
player.sendMessage('§a您的游戏模式已被修改为 ' + gamemode + ' §7- 来自MiaoDashboard')
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: '操作成功!'
|
||||
})
|
||||
})
|
||||
}
|
||||
private initPlugins(client: SocketIOSocket) {
|
||||
this.initNativePlugins(client)
|
||||
this.initScriptPlugins(client)
|
||||
}
|
||||
private initServices(client: SocketIOSocket) {
|
||||
client.on('services.plugins', (callback) => {
|
||||
this.wrapper(callback, http.get('http://w.yumc.pw/api/free_plugin/find').data.map(plugin => {
|
||||
let installed = this.nativePluginManager.get(plugin.name)
|
||||
if (installed) {
|
||||
plugin.installed = true
|
||||
plugin.installedVersion = installed.version
|
||||
}
|
||||
plugin.installed = this.nativePluginManager.has(plugin.name)
|
||||
return plugin
|
||||
}))
|
||||
})
|
||||
client.on('services.plugins.install', ({ name }, callback) => {
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 安装成功!`
|
||||
})
|
||||
})
|
||||
client.on('services.plugins.update', ({ name }, callback) => {
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 安装成功!`
|
||||
})
|
||||
})
|
||||
}
|
||||
private initNativePlugins(client: SocketIOSocket) {
|
||||
client.on('plugins.natives', callback => {
|
||||
this.wrapper(callback, this.nativePluginManager.list().map(plugin => {
|
||||
plugin.status = plugin.enable ? 1 : 0
|
||||
return plugin
|
||||
}))
|
||||
})
|
||||
client.on('plugins.natives.reload', ({ name }, callback) => {
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 不存在`
|
||||
})
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 安装完成`
|
||||
})
|
||||
})
|
||||
client.on('plugins.natives.install', ({ name }, callback) => {
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 不存在`
|
||||
})
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 安装完成`
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
private initScriptPlugins(client: SocketIOSocket) {
|
||||
client.on('plugins.scripts', (callback) => {
|
||||
let plugins = []
|
||||
this.pluginManager.getPlugins().forEach((plugin) => {
|
||||
plugins.push({
|
||||
name: plugin.description.name,
|
||||
version: plugin.description.version,
|
||||
author: plugin.description.author,
|
||||
source: plugin.description.source.toString().replace(root, ''),
|
||||
type: plugin.description.type,
|
||||
scanner: plugin.description.loadMetadata.scanner.type,
|
||||
loader: plugin.description.loadMetadata.loader.type,
|
||||
status: 1
|
||||
})
|
||||
})
|
||||
this.wrapper(callback, {
|
||||
items: plugins,
|
||||
total: plugins.length
|
||||
})
|
||||
})
|
||||
client.on('plugins.scripts.reload', ({ name }, callback) => {
|
||||
let plugin: any = this.pluginManager.getPlugin(name)
|
||||
if (!plugin) {
|
||||
return callback({
|
||||
status: 404,
|
||||
msg: `插件 ${name} 不存在`
|
||||
})
|
||||
}
|
||||
this.pluginManager.reload(plugin)
|
||||
return callback({
|
||||
status: 0,
|
||||
msg: `插件 ${name} 重载完成`
|
||||
})
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -9,6 +9,8 @@ import { Server as SocketIOServer, Socket as SocketIOSocket, Namespace } from '@
|
||||
|
||||
import * as fs from '@ccms/common/dist/fs'
|
||||
|
||||
import type { MiaoConsole } from './MiaoConsole'
|
||||
|
||||
const FileFilter = Java.type('java.io.FileFilter')
|
||||
const ByteArrayInputStream = java.io.ByteArrayInputStream
|
||||
const ByteArrayOutputStream = java.io.ByteArrayOutputStream
|
||||
@@ -16,14 +18,13 @@ const StandardCharsets = Java.type("java.nio.charset.StandardCharsets")
|
||||
const GZIPInputStream = Java.type('java.util.zip.GZIPInputStream')
|
||||
const ByteArray = Java.type("byte[]")
|
||||
|
||||
@JSPlugin({ prefix: 'Explorer', version: '1.0.0', author: 'MiaoWoo', source: __filename })
|
||||
@JSPlugin({ prefix: 'Explorer', version: '1.0.0', author: 'MiaoWoo', source: __filename, depends: ['MiaoConsole'] })
|
||||
export class MiaoExplorer extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private Server: server.Server
|
||||
@Autowired()
|
||||
private pluginManager: plugin.PluginManager
|
||||
|
||||
private token: string
|
||||
private namespace: any
|
||||
private chunkCacheMap: Map<string, Array<string>>
|
||||
|
||||
@@ -32,44 +33,25 @@ export class MiaoExplorer extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
enable() {
|
||||
let consolePlugin: any = this.pluginManager.getPlugin('MiaoConsole')
|
||||
if (consolePlugin.socketIOServer) {
|
||||
this.startWebSocketServer(consolePlugin.socketIOServer)
|
||||
} else {
|
||||
process.on('websocket.create', (server: SocketIOServer) => {
|
||||
this.startWebSocketServer(server)
|
||||
let consolePlugin: MiaoConsole = this.pluginManager.getPlugin('MiaoConsole') as MiaoConsole
|
||||
consolePlugin.registryWebSocketNamespace('/MiaoExplorer', (namespace: Namespace) => {
|
||||
this.namespace = namespace
|
||||
this.namespace.on('connection', (client: SocketIOSocket) => {
|
||||
if (consolePlugin.checkWebSocketClient(client)) {
|
||||
this.initWebSocketClient(client)
|
||||
this.logger.console(`§6客户端 §b${client.id} §a新建连接...`)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
private startWebSocketServer(server: SocketIOServer) {
|
||||
let consolePlugin: any = this.pluginManager.getPlugin('MiaoConsole')
|
||||
this.token = consolePlugin.token
|
||||
this.namespace = server.of('/MiaoExplorer')
|
||||
this.namespace.on('connection', (client: SocketIOSocket) => {
|
||||
if (!this.token) {
|
||||
this.logger.console(`§6客户端 §b${client.id} §a请求连接 §4服务器尚未设置 Token 无法连接!`)
|
||||
client.emit('unauthorized', () => client.disconnect(true))
|
||||
return
|
||||
}
|
||||
if (this.token != client.handshake.query.token) {
|
||||
this.logger.console(`§6客户端 §b${client.id} §c无效请求 §4请提供正确Token后再次连接!`)
|
||||
client.emit('unauthorized', () => client.disconnect(true))
|
||||
return
|
||||
}
|
||||
this.initWebSocketClient(client)
|
||||
this.logger.console(`§6客户端 §b${client.id} §a新建连接...`)
|
||||
})
|
||||
}
|
||||
|
||||
disable() {
|
||||
this.namespace.removeAllListeners('connect')
|
||||
this.namespace.close()
|
||||
this.namespace?.close()
|
||||
}
|
||||
|
||||
private readDir(dir) {
|
||||
let children = Java.from(dir.listFiles(new FileFilter({
|
||||
accept: file => file.getName().endsWith('.yml') || file.isDirectory()
|
||||
accept: file => file.getName().endsWith('.yml') || file.getName().endsWith('.js') || file.isDirectory()
|
||||
}))).sort().map(file => {
|
||||
if (file.isDirectory()) {
|
||||
let children = this.readDir(file)
|
||||
@@ -161,15 +143,4 @@ export class MiaoExplorer extends interfaces.Plugin {
|
||||
this.logger.console(`§6客户端 §b${client.id} §c断开连接...`)
|
||||
})
|
||||
}
|
||||
|
||||
@Cmd()
|
||||
msme(sender: any, command: string, args: string[]) {
|
||||
this.logger.log(sender, command, args)
|
||||
sender.sendMessage(JSON.stringify({ command, ...args }))
|
||||
}
|
||||
|
||||
@Tab()
|
||||
tabmsme(_sender: any, _command: string, _args: string[]) {
|
||||
return ['world']
|
||||
}
|
||||
}
|
||||
|
||||
155
packages/plugins/src/MiaoLink.ts
Normal file
155
packages/plugins/src/MiaoLink.ts
Normal file
@@ -0,0 +1,155 @@
|
||||
/// <reference types="@javatypes/bungee-api" />
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
|
||||
import { server, task } from '@ccms/api'
|
||||
import { Autowired } from '@ccms/container'
|
||||
import { Cmd, JSPlugin, Tab, interfaces, PluginConfig, Config } from '@ccms/plugin'
|
||||
|
||||
import * as fs from '@ccms/common/dist/fs'
|
||||
import http from '@ccms/common/dist/http'
|
||||
|
||||
import * as base64 from 'base64-js'
|
||||
|
||||
const Runtime: typeof java.lang.Runtime = Java.type('java.lang.Runtime')
|
||||
const Thread = Java.type('java.lang.Thread')
|
||||
|
||||
const defaultConfig = {
|
||||
id: 0,
|
||||
vkey: ''
|
||||
}
|
||||
|
||||
@JSPlugin({ name: 'MiaoLink', version: '1.0.2', author: 'MiaoWoo', source: __filename })
|
||||
export class MiaoLink extends interfaces.Plugin {
|
||||
@Autowired(task.TaskManager)
|
||||
private task: task.TaskManager
|
||||
@Autowired(server.Server)
|
||||
private server: server.Server
|
||||
|
||||
@Config()
|
||||
private config: PluginConfig & typeof defaultConfig = defaultConfig
|
||||
|
||||
private isWindows = false
|
||||
private clientName: string = 'npc'
|
||||
private client: string = ''
|
||||
private port: number = 0
|
||||
private npc: any
|
||||
|
||||
load() {
|
||||
this.isWindows = process.platform == 'win32' || process.platform.toLowerCase().startsWith('windows')
|
||||
if (this.isWindows) {
|
||||
this.logger.console('§a当前运行于Windows服务器...')
|
||||
this.clientName = "npc.exe"
|
||||
} else {
|
||||
this.logger.console('§a当前运行于Linux服务器...')
|
||||
}
|
||||
}
|
||||
|
||||
bukkitload() {
|
||||
this.port = org.bukkit.Bukkit.getPort()
|
||||
}
|
||||
|
||||
spongeload() {
|
||||
this.logger.console('§4Sponge暂不支持端口映射!')
|
||||
}
|
||||
|
||||
bungeeload() {
|
||||
let server: net.md_5.bungee.api.ProxyServer = base.getInstance().getProxyServer()
|
||||
this.port = server.getConfig().getListeners()[0].getQueryPort()
|
||||
}
|
||||
|
||||
enable() {
|
||||
if (!this.config.vkey) {
|
||||
return this.logger.console('§4服务器尚未绑定 取消自动映射!')
|
||||
}
|
||||
this.cmdconnect(this.server.getConsoleSender())
|
||||
}
|
||||
|
||||
disable() {
|
||||
this.cmddisconnect(this.server.getConsoleSender())
|
||||
}
|
||||
|
||||
@Cmd({ autoMain: true })
|
||||
mlink() { }
|
||||
|
||||
cmdconnect(sender: any, secret?: string) {
|
||||
if (secret) {
|
||||
let configStr = String.fromCharCode(...Array.from(base64.toByteArray(secret)))
|
||||
let config = JSON.parse(configStr)
|
||||
this.config.id = config.id
|
||||
this.config.vkey = config.vkey
|
||||
this.config.save()
|
||||
}
|
||||
this.startClient(sender)
|
||||
}
|
||||
|
||||
cmddisconnect(sender: any) {
|
||||
if (!this.npc || !this.npc.isAlive()) {
|
||||
return this.logger.sender(sender, '§4客户端尚未运行 跳过关闭流程...')
|
||||
}
|
||||
this.logger.sender(sender, '§6已发送关闭客户端指令...')
|
||||
this.npc.destroy()
|
||||
}
|
||||
|
||||
@Tab()
|
||||
tabmlink(_sender: any, _command: string, _args: string[]) {
|
||||
}
|
||||
|
||||
startClient(sender: any, id: number = this.config.id, vkey: string = this.config.vkey) {
|
||||
if (!this.port) {
|
||||
return this.logger.sender(sender, '§4服务器端口获取失败 取消自动映射!')
|
||||
}
|
||||
if (!id || !vkey) {
|
||||
return this.logger.sender(sender, '§4服务器尚未配置 取消自动映射!')
|
||||
}
|
||||
if (this.npc && this.npc.isAlive()) {
|
||||
this.npc.destroy()
|
||||
}
|
||||
this.task.create(() => {
|
||||
this.logger.sender(sender, `§6获取到服务器端口: §3${this.port} §a开始映射端口!`)
|
||||
let client = this.query(id, vkey, this.port)
|
||||
let node = client.node
|
||||
let tunnel = client.tunnel
|
||||
this.client = fs.concat(__dirname, 'MiaoLink', this.clientName)
|
||||
this.download(sender)
|
||||
try {
|
||||
this.npc = Runtime.getRuntime().exec(`${this.client} -server=${node.bridge} -vkey=${vkey} -type=tcp`)
|
||||
this.logger.sender(sender, `§a服务器端口映射成功! §6访问地址: §3${node.address}:${tunnel.port}`)
|
||||
return this.logger.console(`§4客户端已结束运行 退出代码: ${this.npc.waitFor()} 映射关闭!`)
|
||||
} catch (error) {
|
||||
this.logger.sender(sender, `§c服务器端口映射失败! §4ERROR: ${error}`)
|
||||
console.ex(error)
|
||||
}
|
||||
}, this).async().later(5).submit()
|
||||
}
|
||||
|
||||
download(sender: any) {
|
||||
try {
|
||||
if (!fs.exists(this.client)) {
|
||||
this.logger.sender(sender, '§c客户端文件不存在 开始下载客户端...')
|
||||
let temp = this.client + '.tmp'
|
||||
http.download("https://static.c5mc.cn/" + this.clientName, temp)
|
||||
fs.move(temp, this.client, true)
|
||||
if (!this.isWindows) {
|
||||
this.logger.sender(sender, '§a当前处于Linux环境 赋予可执行权限...')
|
||||
Runtime.getRuntime().exec(`chmod +x ${this.client}`)
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
Thread.sleep(500)
|
||||
this.download(sender)
|
||||
}
|
||||
}
|
||||
|
||||
query(id: number, vkey: string, target: number) {
|
||||
let result = this.post(`/client?id=${id}&vkey=${vkey}&target=${target}`)
|
||||
if (result.code != 200) {
|
||||
throw new Error('§4客户端查询失败: ' + result.msg)
|
||||
}
|
||||
return result.data
|
||||
}
|
||||
|
||||
post(path, data = {}) {
|
||||
return http.post("https://nps.yumc.pw/api" + path, data)
|
||||
}
|
||||
}
|
||||
@@ -2,11 +2,11 @@
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
|
||||
import { particle } from '@ccms/api'
|
||||
import { constants, particle } from '@ccms/api'
|
||||
import { Autowired } from '@ccms/container'
|
||||
import { Cmd, JSPlugin, Tab, interfaces } from '@ccms/plugin'
|
||||
|
||||
@JSPlugin({ version: '1.0.0', author: 'MiaoWoo', source: __filename })
|
||||
@JSPlugin({ version: '1.0.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], source: __filename })
|
||||
export class MiaoParticle extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private particleManager: particle.ParticleManager
|
||||
|
||||
@@ -2,30 +2,33 @@
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
|
||||
import { plugin, server, task } from '@ccms/api'
|
||||
import { constants, plugin, server, task } from '@ccms/api'
|
||||
import { Autowired, JSClass } from '@ccms/container'
|
||||
import { Cmd, Config, interfaces, JSPlugin, PluginConfig, Tab } from '@ccms/plugin'
|
||||
import { Cmd, Config, interfaces, JSPlugin, Listener, PluginConfig, Tab } from '@ccms/plugin'
|
||||
|
||||
import type { MiaoReward } from './MiaoReward'
|
||||
import http from '@ccms/common/dist/http'
|
||||
import * as CryptoJS from "crypto-js"
|
||||
|
||||
interface PlayerPointsAPI {
|
||||
look(name: string)
|
||||
give(name: string, amount: number)
|
||||
take(name: string, amount: number)
|
||||
}
|
||||
const Thread = Java.type('java.lang.Thread')
|
||||
|
||||
interface App {
|
||||
appid: string
|
||||
appname: string
|
||||
ratio: number
|
||||
coin_name: string
|
||||
}
|
||||
interface Order {
|
||||
order_id: string
|
||||
amount: string
|
||||
url: string
|
||||
amount: number
|
||||
url?: string
|
||||
}
|
||||
interface Sync {
|
||||
scaned: boolean
|
||||
start?: number
|
||||
left?: number
|
||||
cancelled?: boolean
|
||||
paying?: boolean
|
||||
}
|
||||
|
||||
interface PlaceholderAPI {
|
||||
@@ -36,15 +39,41 @@ interface PlaceholderAPI {
|
||||
|
||||
const defaultConfig = {
|
||||
prefix: '§6[§b喵式支付§6]',
|
||||
name: '',
|
||||
id: '',
|
||||
secret: '',
|
||||
command: 'p give %player_name% %amount%',
|
||||
command: 'points give %player_name% %amount%',
|
||||
check: '%playerpoints_points%',
|
||||
ratio: 100,
|
||||
coinName: '点券'
|
||||
coinName: '点券',
|
||||
reward: {
|
||||
'*': [
|
||||
'msg %player_name% 充值 %amount% 点券成功...'
|
||||
],
|
||||
10: [
|
||||
"points give %player_name% 1",
|
||||
'msg %player_name% 充值 10 点券 奖励 1 点券...'
|
||||
],
|
||||
300: [
|
||||
"points give %player_name% 15",
|
||||
'msg %player_name% 充值 300 点券 奖励 15 点券...'
|
||||
],
|
||||
500: [
|
||||
"points give %player_name% 25",
|
||||
'msg %player_name% 充值 500 点券 奖励 25 点券...'
|
||||
],
|
||||
680: [
|
||||
"points give %player_name% 35",
|
||||
'msg %player_name% 充值 680 点券 奖励 35 点券...'
|
||||
],
|
||||
1280: [
|
||||
"points give %player_name% 70",
|
||||
'msg %player_name% 充值 1280 点券 奖励 70 点券...'
|
||||
],
|
||||
}
|
||||
}
|
||||
|
||||
@JSPlugin({ version: '1.2.0', author: 'MiaoWoo', source: __filename, depends: ['MiaoReward'], nativeDepends: ['PlaceholderAPI'] })
|
||||
@JSPlugin({ version: '1.6.6', author: 'MiaoWoo', source: __filename, servers: [constants.ServerType.Bukkit], depends: ['MiaoReward'], nativeDepends: ['PlaceholderAPI', 'ProtocolLib'] })
|
||||
export class MiaoPay extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private server: server.Server
|
||||
@@ -56,37 +85,42 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
@JSClass('me.clip.placeholderapi.PlaceholderAPI')
|
||||
private PlaceholderAPI: PlaceholderAPI
|
||||
|
||||
private apiGateWay = "https://pay.yumc.pw"
|
||||
private MiaoReward: MiaoReward
|
||||
private appInfo: App
|
||||
|
||||
private cacheMap = new Map<string, Order>();
|
||||
private cacheSyncMap = new Map<string, Sync>();
|
||||
|
||||
private checkSet = new Set<string>();
|
||||
|
||||
@Config()
|
||||
private config: PluginConfig & typeof defaultConfig = defaultConfig
|
||||
|
||||
load() {
|
||||
let needSave = false
|
||||
for (const key of Object.keys(defaultConfig)) {
|
||||
if (!this.config[key]) {
|
||||
this.config[key] = defaultConfig[key]
|
||||
needSave = true
|
||||
}
|
||||
}
|
||||
needSave && this.config.save()
|
||||
}
|
||||
|
||||
cmdtest(sender: org.bukkit.entity.Player) {
|
||||
try {
|
||||
this.logger.sender(sender, this.getPlayerAmount(sender))
|
||||
} catch (error) {
|
||||
return this.logger.sender(sender, error.message)
|
||||
}
|
||||
this.MiaoReward = this.pluginManager.getPlugin('MiaoReward') as MiaoReward
|
||||
}
|
||||
|
||||
enable() {
|
||||
this.MiaoReward = this.pluginManager.getPlugin('MiaoReward') as MiaoReward
|
||||
if (!this.MiaoReward) { return this.logger.error('当前脚本插件需要 MiaoReward 作为前置脚本插件!') }
|
||||
if (!this.config.id || !this.config.secret) { return this.logger.console('§4尚未配置商户信息 将无法正常收款!') }
|
||||
this.initAppInfo()
|
||||
}
|
||||
|
||||
private initAppInfo() {
|
||||
let info = this.httpPost('/apps', { id: this.config.id }, 10)
|
||||
if (info.code == 200) {
|
||||
this.appInfo = info.data
|
||||
this.config.ratio = this.appInfo.ratio
|
||||
this.config.coinName = this.appInfo.coin_name
|
||||
if (this.config.name == this.appInfo.appname) {
|
||||
this.config.name = ''
|
||||
this.config.save()
|
||||
}
|
||||
} else {
|
||||
this.logger.console('§4初始化支付系统失败 请检查配置或网络是否正常!')
|
||||
this.logger.console('§c返回异常: §4' + info.msg)
|
||||
}
|
||||
}
|
||||
|
||||
disable() {
|
||||
@@ -99,32 +133,45 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
@Cmd({ autoMain: true })
|
||||
mpay() { }
|
||||
|
||||
cmdpay(sender: org.bukkit.entity.Player, amount: number) {
|
||||
cmdpay(sender: org.bukkit.entity.Player, amount: number = 0) {
|
||||
if (!sender.getItemInHand) { return this.logger.sender(sender, '§4控制台无法执行此命令!') }
|
||||
if (!this.MiaoReward.serverInfo) { return this.logger.sender(sender, '§4当前服务器尚未配置 请联系管理员先配置MiaoReward!') }
|
||||
if (!this.config.id || !this.config.secret) { return this.logger.sender(sender, '§c当前服务器尚未配置 请联系管理员配置支付密钥!') }
|
||||
if (!this.appInfo) {
|
||||
this.initAppInfo()
|
||||
return this.logger.sender(sender, '§6支付系统初始化中 请稍候重试...')
|
||||
}
|
||||
if (this.cacheMap.has(sender.getName())) {
|
||||
this.logger.sender(sender, '§c您有一笔订单尚未完成 请完成支付或等待订单超时!')
|
||||
let sync = this.cacheSyncMap.get(sender.getName())
|
||||
if (!sync.cancelled) { return }
|
||||
sync.left = 55 - (Math.round(Date.now() / 1000) - sync.start)
|
||||
if (sync.scaned) { return }
|
||||
sync.scaned = true
|
||||
Thread.sleep(1100)
|
||||
sync.scaned = false
|
||||
sync.left = (sync.paying ? 100 : 55) - (Math.round(Date.now() / 1000) - sync.start)
|
||||
let order = this.cacheMap.get(sender.getName())
|
||||
this.MiaoReward.setItemAndTp(sender, order.url, sync, `充值 ${order.amount} ${this.config.coinName}`, `支付宝/微信/QQ 扫码支付`)
|
||||
this.MiaoReward.setItemAndTp(sender, order.url, sync, `充值 ${this.config.ratio * order.amount} ${this.config.coinName}`, `支付宝/微信/QQ 扫码支付`)
|
||||
return
|
||||
}
|
||||
if (amount < 1) { return this.logger.sender(sender, `§c充值异常 §4充值金额不得小于 1 ${this.config.coinName}!`) }
|
||||
if (amount / this.config.ratio > 5000) { return this.logger.sender(sender, `§c充值异常 §4充值金额不得大于 ${this.config.ratio * 3000} ${this.config.coinName}!`) }
|
||||
if (amount != Math.round(amount)) { return this.logger.sender(sender, `§c充值异常 §4充值金额必须为整数!`) }
|
||||
try {
|
||||
this.getPlayerAmount(sender)
|
||||
} catch (error) {
|
||||
return this.logger.sender(sender, error.message)
|
||||
return this.logger.sender(sender, error.message || error)
|
||||
}
|
||||
this.taskManager.create(() => this.createOrderByPlayer(sender, amount)).async().submit()
|
||||
}
|
||||
|
||||
private createOrderByPlayer(sender: org.bukkit.entity.Player, amount: number = 0) {
|
||||
this.MiaoReward.sendTitle(sender, `§6充值 §a${amount} §6${this.config.coinName}`, '§c正在请求充值二维码 请稍候...')
|
||||
let sync = { scaned: false, start: Math.round(Date.now() / 1000) }
|
||||
let sync: any = { scaned: false, start: Math.round(Date.now() / 1000) }
|
||||
let order = this.createOrder(sender, amount)
|
||||
this.cacheMap.set(sender.getName(), order)
|
||||
this.cacheSyncMap.set(sender.getName(), sync)
|
||||
let order_id = order.order_id
|
||||
this.MiaoReward.setItemAndTp(sender, order.url, sync, `充值 ${amount} ${this.config.coinName}`, `支付宝/微信/QQ 扫码支付`)
|
||||
this.logger.sender(sender, [`§3请使用 §b支付宝§3/§a微信§3/§5QQ §3扫描二维码支付!`, `§c如未显示二维码 请打开下列网址完成充值!`, `§6地址: §3${order.url}`])
|
||||
this.taskManager.create(() => {
|
||||
try {
|
||||
let status = this.queryStatus(order_id, 0, 55)
|
||||
@@ -132,15 +179,15 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
this.MiaoReward.sendTitle(sender, '§a已扫码', `§3订单已创建 请及时支付!`)
|
||||
this.MiaoReward.sendActionBar(sender, '§6订单号: §3' + order_id)
|
||||
sync.scaned = true
|
||||
status = this.queryStatus(order_id, 1, 120)
|
||||
sync.start = Math.round(Date.now() / 1000)
|
||||
sync.paying = true
|
||||
status = this.queryStatus(order_id, 1, 100)
|
||||
if (status.code != 200) { throw new Error('§c支付超时 请重新充值!') }
|
||||
if (status.data == 2) {
|
||||
this.MiaoReward.sendTitle(sender, '§a已支付', `§3订单已支付 请等待充值到账...`)
|
||||
this.MiaoReward.sendActionBar(sender, '§6订单号: §3' + order_id)
|
||||
this.logger.sender(sender, `§6订单号: §3${order_id} §a支付成功!`)
|
||||
this.recharge(sender, order_id, order.amount)
|
||||
} else {
|
||||
this.logger.sender(sender, `§c充值系统异常 §4订单状态异常${status.data} §c点券可能未到账 请联系管理员!`)
|
||||
this.recharge(sender, order)
|
||||
}
|
||||
} catch (error) {
|
||||
let cacheOrder = this.cacheMap.get(sender.getName())
|
||||
@@ -156,83 +203,142 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
this.MiaoReward.clearTitle(sender)
|
||||
}
|
||||
|
||||
cmdquery(sender: org.bukkit.entity.Player, id: string) {
|
||||
if (!id) { return this.logger.sender(sender, '§c请输入订单号!') }
|
||||
cmdcheck(sender: org.bukkit.entity.Player, force = 1) {
|
||||
if (this.checkSet.has(sender.getName())) {
|
||||
return this.logger.sender(sender, '§c检查任务执行中 请稍候...')
|
||||
}
|
||||
this.checkSet.add(sender.getName())
|
||||
this.logger.sender(sender, `§3正在检查需要补单充值的订单 请稍候...`)
|
||||
this.taskManager.create(() => {
|
||||
let result = this.queryOrder(id, sender.getName(), sender.getUniqueId().toString())
|
||||
if (result.code != 200) { return this.logger.sender(sender, `§c查询异常! §4ERROR: ${result.msg}`) }
|
||||
let order = result.data
|
||||
this.logger.sender(sender, [
|
||||
`§6商户名称: §3${order.appname}`,
|
||||
`§6订单号: §3${id}`,
|
||||
`§6商品: §b${order.subject}`,
|
||||
`§6金额: §e${order.amount}`,
|
||||
`§6玩家: §a${order.username}`,
|
||||
`§6状态: §c${order.status}`,
|
||||
])
|
||||
if (order.status > 1 && order.status < 4) {
|
||||
this.logger.sender(sender, `§3当前订单已支付 尚未完成充值 开始补单操作...`)
|
||||
this.recharge(sender, id, order.amount)
|
||||
try {
|
||||
let result = this.queryUnconverted(sender.getName(), force)
|
||||
if (result.code != 200) { return this.logger.sender(sender, `§c订单查询失败: ${result.msg}`) }
|
||||
let unconverteds = result.data
|
||||
if (!unconverteds.length) { return this.logger.sender(sender, `§c未发现需要进行补单充值的订单!`) }
|
||||
this.logger.sender(sender, `§3发现 §a${unconverteds.length}笔 §3未充值订单 §c正在充值 请稍候...`)
|
||||
for (const unconverted of unconverteds) {
|
||||
this.logger.sender(sender, `§3正在处理订单 §a${unconverted.order_id} §3请稍候...`)
|
||||
this.recharge(sender, unconverted)
|
||||
Thread.sleep(300)
|
||||
}
|
||||
} finally {
|
||||
this.checkSet.delete(sender.getName())
|
||||
}
|
||||
}).async().submit()
|
||||
}
|
||||
|
||||
/**
|
||||
* 请在异步线程执行此方法
|
||||
* @param amount 订单金额(非点券金额)
|
||||
*/
|
||||
recharge(sender: org.bukkit.entity.Player, order_id: string, amount: number) {
|
||||
let point = amount * this.config.ratio
|
||||
this.taskManager.create(() => {
|
||||
let finish = this.preFinishOrder(order_id)
|
||||
if (finish.code != 200) {
|
||||
return this.logger.console(`§c充值系统异常 订单 §3${order_id} 预标记异常! §4${this.config.coinName}已停止充值 §c请手动补单!`)
|
||||
recharge(sender: org.bukkit.entity.Player, order: Order) {
|
||||
let order_id = order.order_id
|
||||
let amount = order.amount
|
||||
let point = this.safeMultiply(amount, this.config.ratio)
|
||||
if (!sender.isOnline()) { return }
|
||||
let finish = this.preFinishOrder(order_id)
|
||||
if (finish.code != 200) {
|
||||
this.sendError(sender, order_id, amount, '充值预标记异常!')
|
||||
return this.logger.console(`§c充值系统异常 订单 §3${order_id} 预标记异常! §4${this.config.coinName}已停止充值 §c请手动补单!`)
|
||||
}
|
||||
this.taskManager.callSyncMethod(() => {
|
||||
if (!sender.isOnline()) { return this.errorOrder(order_id, "充值前玩家掉线 请重置标记!") }
|
||||
let prePoint = this.getPlayerAmount(sender)
|
||||
let command = this.config.command.replace('%player_name%', sender.getName()).replace('%amount%', `${point}`).replace('%remark%', `${order_id}`)
|
||||
if (!this.server.dispatchConsoleCommand(command)) {
|
||||
return this.sendError(sender, order_id, amount, '充值命令执行异常!')
|
||||
}
|
||||
this.checkRecharge(sender, order_id, amount, prePoint, point)
|
||||
})
|
||||
}
|
||||
|
||||
@Listener()
|
||||
private PlayerJoinEvent(event: org.bukkit.event.player.PlayerJoinEvent) {
|
||||
const player = event.getPlayer()
|
||||
this.cmdcheck(player, 0)
|
||||
}
|
||||
|
||||
private checkRecharge(sender: org.bukkit.entity.Player, order_id: string, amount: number, prePoint: number, point: number) {
|
||||
this.taskManager.create(() => {
|
||||
if (!sender.isOnline()) { return this.errorOrder(order_id, "充值后玩家掉线 请标记已兑换!") }
|
||||
let nowPoint = this.checkNowPoint(sender, point, prePoint)
|
||||
if (nowPoint === false) { return this.sendError(sender, order_id, amount, '充值结果检测异常!') }
|
||||
this.logger.sender(sender, [
|
||||
`§6充值 §a${point} §6${this.config.coinName} §a成功 §6当前账户余额: §3${nowPoint} §6${this.config.coinName}`,
|
||||
`§c如出现未到账的情况 请联系管理员!`
|
||||
])
|
||||
this.rewardOrder(sender, order_id, point)
|
||||
let finish = this.finishOrder(order_id)
|
||||
if (finish.code != 200) {
|
||||
this.errorOrder(order_id, '充值完成标记异常 请到后台标记为已兑换!')
|
||||
return this.logger.console(`§c充值系统异常 订单 §3${order_id} §c完成标记异常! §a请到后台标记为已兑换! §4否则${this.config.coinName}可能重复到账!`)
|
||||
}
|
||||
this.taskManager.callSyncMethod(() => {
|
||||
let prePoint = this.getPlayerAmount(sender)
|
||||
let command = this.config.command.replace('%player_name%', sender.getName()).replace('%amount%', `${point}`).replace('%remark%', `${order_id}`)
|
||||
if (!this.server.dispatchConsoleCommand(command)) {
|
||||
return this.sendError(sender, order_id, amount, '§4充值命令执行异常!')
|
||||
}
|
||||
let nowPoint = this.getPlayerAmount(sender)
|
||||
if (nowPoint != prePoint + point) {
|
||||
return this.sendError(sender, order_id, amount, '§4充值结果检测异常!')
|
||||
} else {
|
||||
this.taskManager.create(() => {
|
||||
this.logger.sender(sender, [
|
||||
`§6充值 §a${point} §6${this.config.coinName} §a成功 §6当前账户余额: §3${nowPoint} §6${this.config.coinName}`,
|
||||
`§c如出现未到账的情况 请联系管理员!`
|
||||
])
|
||||
let finish = this.finishOrder(order_id)
|
||||
if (finish.code != 200) {
|
||||
return this.logger.console(`§c充值系统异常 订单 §3${order_id} 完成标记异常! §4${this.config.coinName}可能重复到账!`)
|
||||
}
|
||||
}).async().submit()
|
||||
}
|
||||
})
|
||||
}).async().submit()
|
||||
}
|
||||
|
||||
private checkNowPoint(sender: org.bukkit.entity.Player, point: number, prePoint: number, times: number = 1) {
|
||||
if (times > 3) { return false }
|
||||
let nowPoint = this.getPlayerAmount(sender)
|
||||
if (nowPoint == prePoint + point) { return nowPoint }
|
||||
Thread.sleep(times * 100)
|
||||
return this.checkNowPoint(sender, point, prePoint, times++)
|
||||
}
|
||||
|
||||
private rewardOrder(sender, order_id, point) {
|
||||
if (!this.config.reward) { return }
|
||||
this.taskManager.callSyncMethod(() => {
|
||||
try {
|
||||
if (this.config.reward['*']) {
|
||||
let rewardCommands: string[] = this.config.reward['*']
|
||||
for (const command of rewardCommands) {
|
||||
this.dispatchConsoleCommand(sender, command, point, order_id)
|
||||
}
|
||||
}
|
||||
let rewardCommands: string[] = this.config.reward[point]
|
||||
if (rewardCommands && rewardCommands.length) {
|
||||
for (const command of rewardCommands) {
|
||||
this.dispatchConsoleCommand(sender, command, point, order_id)
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('§4充值奖励命令执行错误: §c' + error)
|
||||
console.ex(error)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
private dispatchConsoleCommand(sender, command, point, order_id) {
|
||||
return this.server.dispatchConsoleCommand(command.replace('%player_name%', sender.getName()).replace('%amount%', `${point}`).replace('%remark%', `${order_id}`))
|
||||
}
|
||||
|
||||
sendError(sender: org.bukkit.entity.Player, order_id: string, amount: number, error: string) {
|
||||
return this.logger.sender(sender, [
|
||||
this.logger.sender(sender, [
|
||||
`§c========== ${this.config.prefix}§4充值异常 §c==========`,
|
||||
`§6异常订单: §3${order_id}`,
|
||||
`§6订单金额: §3${amount}`,
|
||||
`§6异常原因: §4${error}`,
|
||||
`§6异常账号: §b${sender.getName()}`,
|
||||
`§6异常时间: §a${new Date().toLocaleDateString()} ${new Date().toLocaleTimeString()}`,
|
||||
`§c如果已付款但${this.config.coinName}未到账 请截图发往QQ群!`,
|
||||
`§c如果已付款但${this.config.coinName}未到账 请截图发给腐竹!`,
|
||||
`§c可尝试重新登录 或 执行 §3/mpay check §c手动补单!`,
|
||||
`§c========== ${this.config.prefix}§4充值异常 §c==========`,
|
||||
])
|
||||
this.errorOrder(order_id, error)
|
||||
}
|
||||
|
||||
@Tab()
|
||||
tabmpay() { }
|
||||
|
||||
private safeMultiply(a: number, b: number) {
|
||||
return parseFloat((a * b).toFixed(0))
|
||||
}
|
||||
|
||||
private getPlayerAmount(sender: any): number {
|
||||
let result = this.PlaceholderAPI.setPlaceholders(sender, this.config.check)
|
||||
let amount = parseFloat(result)
|
||||
if (isNaN(amount)) {
|
||||
throw new Error(`§c读取玩家 §3${this.config.coinName} §c异常 §6请检查 §3check §6配置是否正确!\n§6数据解析链路: §3${this.config.check} §6=> §3${result} §6=> §3${amount}`)
|
||||
throw new Error(`§c读取玩家 §3${this.config.coinName} §c异常 §6请检查 §3check §6配置是否正确!
|
||||
§6数据解析链路: §3${this.config.check} §6=> §3${result} §6=> §3${amount}`)
|
||||
}
|
||||
return amount
|
||||
}
|
||||
@@ -242,19 +348,25 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
private preFinishOrder(id: string) {
|
||||
return this.httpPost('/preFinish', { id })
|
||||
return this.httpPost('/preFinish', { id }, 3)
|
||||
}
|
||||
|
||||
private errorOrder(id: string, error: string) {
|
||||
return this.httpPost('/error', { id, error }, 3)
|
||||
}
|
||||
|
||||
private finishOrder(id: string) {
|
||||
return this.httpPost('/finish', { id })
|
||||
return this.httpPost('/finish', { id }, 3)
|
||||
}
|
||||
|
||||
private createOrder(sender: org.bukkit.entity.Player, amount: number) {
|
||||
private createOrder(sender: org.bukkit.entity.Player, amount: number): Order {
|
||||
let serverName = this.appInfo.appname
|
||||
if (this.config.name) { serverName = `${serverName}(${this.config.name})` }
|
||||
let result = this.httpPost('/create', {
|
||||
subject: `${this.MiaoReward.serverInfo.name} 充值 ${amount} ${this.config.coinName}`,
|
||||
totalAmount: amount / this.config.ratio,
|
||||
subject: `${serverName} 充值 ${amount} ${this.config.coinName}`,
|
||||
amount: amount / this.appInfo.ratio,
|
||||
username: sender.getName(),
|
||||
union_id: sender.getUniqueId().toString()
|
||||
unionId: sender.getUniqueId().toString()
|
||||
})
|
||||
if (result.code != 200) {
|
||||
throw new Error(`订单创建失败: ${result.msg}`)
|
||||
@@ -263,23 +375,38 @@ export class MiaoPay extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
private queryOrder(id: string, username: string, uuid: string) {
|
||||
return this.httpPost('/query', { id, username, uuid })
|
||||
return this.httpPost('/query', { id, username, uuid }, 2)
|
||||
}
|
||||
|
||||
private httpPost(method: string, data: any) {
|
||||
private queryUnconverted(username: string, force: number) {
|
||||
return this.httpPost('/unconverted', { username, force }, 2)
|
||||
}
|
||||
|
||||
private httpPost(method: string, data: any, retry = 0) {
|
||||
let startTime = Date.now()
|
||||
data.appid = this.config.id
|
||||
data.timestamp = Math.round(Date.now() / 1000)
|
||||
data.nonce = 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'.replace(/x/g, () => (Math.random() * 16 | 0).toString(16))
|
||||
data.sign = this.sign(data)
|
||||
let url = `https://pay.yumc.pw/api${method}`
|
||||
let result = http.post(url, data)
|
||||
console.debug(`
|
||||
let url = `${this.apiGateWay}/api${method}`
|
||||
try {
|
||||
let result = http.post(url, data)
|
||||
console.debug(`
|
||||
====== HTTP POST ======
|
||||
REQUEST URL : ${url}
|
||||
REQUEST DATA: ${JSON.stringify(data)}
|
||||
RESPONSE : ${JSON.stringify(result)}
|
||||
CAST TIME : ${Date.now() - startTime}`)
|
||||
return result
|
||||
return result
|
||||
} catch (error) {
|
||||
if (retry) {
|
||||
Thread.sleep(retry * 10)
|
||||
return this.httpPost(method, data, --retry)
|
||||
}
|
||||
console.console('§4请求支付中心发生异常 请联系管理员处理此问题!')
|
||||
console.ex(error)
|
||||
return { code: 500, msg: '本地网络错误: ' + error.message, data: error }
|
||||
}
|
||||
}
|
||||
|
||||
private http_build_query(params: any) {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { task, server, constants } from "@ccms/api"
|
||||
import { Autowired, JSClass } from "@ccms/container"
|
||||
import { plugin, interfaces } from "@ccms/plugin"
|
||||
import { plugin, interfaces, Cmd } from "@ccms/plugin"
|
||||
|
||||
let createPacketAdapterFunction = eval(`
|
||||
function(cls, plugin, type, onPacketSending){
|
||||
@@ -10,6 +10,13 @@ function(cls, plugin, type, onPacketSending){
|
||||
}
|
||||
`)
|
||||
|
||||
interface PlaceholderAPI {
|
||||
registerPlaceholderHook: (key: string, onPlaceholderRequest: (player, s) => string) => void
|
||||
unregisterPlaceholderHook: (key: string) => void
|
||||
setPlaceholders: (player: any, str: string) => string
|
||||
}
|
||||
const Pattern = Java.type('java.util.regex.Pattern')
|
||||
|
||||
@plugin({ prefix: 'MPTL', version: '1.0.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], source: __filename })
|
||||
export class MiaoProtocol extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
@@ -24,9 +31,24 @@ export class MiaoProtocol extends interfaces.Plugin {
|
||||
@JSClass('com.comphenix.protocol.ProtocolLibrary')
|
||||
private ProtocolLibrary
|
||||
|
||||
@JSClass('com.comphenix.protocol.wrappers.nbt.NbtFactory')
|
||||
private NbtFactory
|
||||
|
||||
@JSClass('me.clip.placeholderapi.PlaceholderAPI')
|
||||
private PlaceholderAPI: PlaceholderAPI
|
||||
|
||||
private pipeline: any
|
||||
private adapter: any
|
||||
|
||||
@Cmd({ autoMain: true })
|
||||
mptl() {
|
||||
}
|
||||
|
||||
cmdnbt(sender: org.bukkit.entity.Player) {
|
||||
let nbt = this.NbtFactory.fromItemOptional(sender.getItemInHand())
|
||||
console.log(nbt)
|
||||
}
|
||||
|
||||
enable() {
|
||||
let count = 0
|
||||
let wait = this.taskManager.create(() => {
|
||||
@@ -46,20 +68,49 @@ export class MiaoProtocol extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
createPacketAdapter(onPacketSending: (event) => void) {
|
||||
return createPacketAdapterFunction(this.PacketAdapter, base.getInstance(), [this.PacketType.Play.Server.MAP], onPacketSending)
|
||||
return createPacketAdapterFunction(this.PacketAdapter, base.getInstance(), [this.PacketType.Play.Server.CHAT], onPacketSending)
|
||||
}
|
||||
|
||||
initPacketAdapter() {
|
||||
this.adapter = this.createPacketAdapter((event) => {
|
||||
let integers = event.getPacket().getIntegers().getValues()
|
||||
// console.log(`ProtocolLib onPacketSending filter Map
|
||||
// Player: ${event.getPlayer()}
|
||||
// MapId: ${integers.get(0)}
|
||||
// Short: ${event.getPacket().getShorts().read(0)}
|
||||
// Bytes: ${event.getPacket().getByteArrays().read(0).length}
|
||||
// `)
|
||||
//Size: ${integers.get(3)}x${integers.get(4)}
|
||||
// org.bukkit.map.MapPalette.imageToBytes()
|
||||
try {
|
||||
// let wcc = event.getPacket().getChatComponents().read(0)
|
||||
// if (wcc == null) { return }
|
||||
// let json = wcc.getJson()
|
||||
// console.log(json)
|
||||
// let jsonObj = JSON.parse(json)
|
||||
// let result = JSON.stringify(this.colorJson(jsonObj))
|
||||
// console.log(result)
|
||||
// wcc.setJson(result)
|
||||
// event.getPacket().getChatComponents().writeSafely(0, wcc)
|
||||
// let packet = event.getPacket()
|
||||
// let modify = packet.getEntityModifier(event)
|
||||
// console.log(modify.getValues().get(0).getType().getName())
|
||||
// let modify = event.getPacket().getNbtModifier()
|
||||
// let nbt = modify.read(0)
|
||||
// let lines = ["Text1", "Text2", "Text3", "Text4"]
|
||||
// console.log("Before Replace", event.getPlayer().getName(), nbt)
|
||||
// lines.forEach((s: string) => {
|
||||
// let origin = nbt.getString(s)
|
||||
// let replaced = this.PlaceholderAPI.setPlaceholders(event.getPlayer(), origin)
|
||||
// nbt.put(s, replaced)
|
||||
// console.log(event.getPlayer(), origin, replaced)
|
||||
// })
|
||||
// // Arrays.asList("Text1", "Text2", "Text3", "Text4").forEach(s -> nbt.put(s, replace(event.getPlayer(), nbt.getString(s))))
|
||||
// console.log("After Replace", event.getPlayer().getName(), nbt)
|
||||
// modify.write(0, nbt)
|
||||
// let integers = event.getPacket().getIntegers().getValues()
|
||||
// console.log(`ProtocolLib onPacketSending filter Map
|
||||
// Player: ${event.getPlayer()}
|
||||
// MapId: ${integers.get(0)}
|
||||
// Short: ${event.getPacket().getShorts().read(0)}
|
||||
// Bytes: ${event.getPacket().getByteArrays().read(0).length}
|
||||
// `)
|
||||
//Size: ${integers.get(3)}x${integers.get(4)}
|
||||
// org.bukkit.map.MapPalette.imageToBytes()
|
||||
} catch (error) {
|
||||
console.ex(error)
|
||||
}
|
||||
})
|
||||
this.ProtocolLibrary.getProtocolManager().addPacketListener(this.adapter)
|
||||
}
|
||||
|
||||
157
packages/plugins/src/MiaoRGBSupport.ts
Normal file
157
packages/plugins/src/MiaoRGBSupport.ts
Normal file
@@ -0,0 +1,157 @@
|
||||
import { constants } from "@ccms/api"
|
||||
import { JSClass } from "@ccms/container"
|
||||
import { interfaces, JSPlugin } from "@ccms/plugin"
|
||||
|
||||
let createPacketAdapterFunction = eval(`
|
||||
function(cls, plugin, type, onPacketSending){
|
||||
return new cls(plugin, type) {
|
||||
onPacketSending: onPacketSending
|
||||
}
|
||||
}
|
||||
`)
|
||||
const Color = Java.type('java.awt.Color')
|
||||
const Pattern = Java.type('java.util.regex.Pattern')
|
||||
const ChatColor = Java.type('net.md_5.bungee.api.ChatColor')
|
||||
|
||||
@JSPlugin({ prefix: 'MRS', version: '1.0.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], nativeDepends: ['ProtocolLib'], source: __filename })
|
||||
export class MiaoRGBSupport extends interfaces.Plugin {
|
||||
private supportRGB: boolean = false
|
||||
// 用于匹配 '#FFFFFF' 颜色格式
|
||||
private RGBCOLOR_PATTERN = Pattern.compile("(#[a-fA-F0-9]{6}?)([^#?]*)")
|
||||
// 用于匹配彩虹格式
|
||||
private RAINBOW_PATTERN = Pattern.compile("#RAINBOW([0-9]{1,3})([^#?]*)")
|
||||
|
||||
@JSClass('com.comphenix.protocol.events.PacketAdapter')
|
||||
private PacketAdapter: any
|
||||
@JSClass('com.comphenix.protocol.PacketType')
|
||||
private PacketType: any
|
||||
@JSClass('com.comphenix.protocol.ProtocolLibrary')
|
||||
private ProtocolLibrary: any
|
||||
|
||||
private adapter: any
|
||||
|
||||
load() {
|
||||
try {
|
||||
ChatColor.of('#FFFFFF').toString()
|
||||
this.supportRGB = true
|
||||
this.logger.console('§a检测到兼容RGB的服务端 已启动相关支持...')
|
||||
} catch (error) {
|
||||
this.logger.console('§c当前服务端不支持RGB色彩 Error: ' + error)
|
||||
}
|
||||
}
|
||||
|
||||
enable() {
|
||||
if (this.supportRGB) {
|
||||
this.initPacketAdapter()
|
||||
}
|
||||
}
|
||||
|
||||
disable() {
|
||||
if (this.supportRGB) {
|
||||
this.ProtocolLibrary.getProtocolManager().removePacketListener(this.adapter)
|
||||
}
|
||||
}
|
||||
|
||||
createPacketAdapter(onPacketSending: (event) => void) {
|
||||
return createPacketAdapterFunction(this.PacketAdapter, base.getInstance(), [
|
||||
this.PacketType.Play.Server.CHAT,
|
||||
this.PacketType.Play.Server.SCOREBOARD_OBJECTIVE,
|
||||
this.PacketType.Play.Server.SCOREBOARD_SCORE,
|
||||
this.PacketType.Play.Server.SCOREBOARD_TEAM
|
||||
], onPacketSending)
|
||||
}
|
||||
|
||||
colorJson(jsonObj) {
|
||||
return this.processJson(jsonObj, this.RGBCOLOR_PATTERN, (extras, colors) => {
|
||||
return (value, index) => {
|
||||
extras.push({
|
||||
text: value,
|
||||
color: colors[index]
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
rainbowJson(jsonObj) {
|
||||
return this.processJson(jsonObj, this.RAINBOW_PATTERN, (extras, colors) => {
|
||||
return (value: string, index) => {
|
||||
let textArr = value.split("")
|
||||
let rainbowColors = this.createRainbow(textArr.length, colors[index])
|
||||
textArr.forEach((value, index) => {
|
||||
extras.push({
|
||||
text: value,
|
||||
color: rainbowColors[index]
|
||||
})
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
private processJson(jsonObj, PATTERN, process: (extras, colors) => (value, index) => void) {
|
||||
let text: string = jsonObj.text
|
||||
if (jsonObj.extra && jsonObj.extra.length) {
|
||||
for (const extra of jsonObj.extra) {
|
||||
this.processJson(extra, PATTERN, process)
|
||||
}
|
||||
}
|
||||
if (!text) { return jsonObj }
|
||||
var matcher = PATTERN.matcher(text)
|
||||
let colors = []
|
||||
let texts = []
|
||||
let lastStart = 0
|
||||
while (matcher.find()) {
|
||||
if (lastStart == 0) {
|
||||
texts.push(text.substr(lastStart, matcher.start()))
|
||||
lastStart = matcher.end()
|
||||
}
|
||||
colors.push(matcher.group(1))
|
||||
texts.push(matcher.group(2))
|
||||
}
|
||||
if (colors.length) {
|
||||
jsonObj.text = ''
|
||||
let extras = []
|
||||
let firstText = texts.shift()
|
||||
if (firstText) { extras.push({ text: firstText }) }
|
||||
texts.forEach(process(extras, colors))
|
||||
if (jsonObj.extra) {
|
||||
jsonObj.extra = extras.concat(jsonObj.extra)
|
||||
} else {
|
||||
jsonObj.extra = extras
|
||||
}
|
||||
}
|
||||
return jsonObj
|
||||
}
|
||||
|
||||
private createRainbow(step, saturation) {
|
||||
var colors = []
|
||||
var colorStep = (1.00 / step)
|
||||
for (var i = 0; i < step; i++) {
|
||||
colors.push("#" + java.lang.String.format("%08x", Color.getHSBColor((colorStep * i), saturation, saturation).getRGB()).substring(2))
|
||||
}
|
||||
return colors
|
||||
}
|
||||
|
||||
initPacketAdapter() {
|
||||
this.adapter = this.createPacketAdapter((event) => {
|
||||
try {
|
||||
if (!event.getPlayer().hasPermission('MiaoRGBSupport.color')) { return }
|
||||
let ccs = event.getPacket().getChatComponents()
|
||||
let size = ccs.size()
|
||||
for (let i = 0; i < size; i++) {
|
||||
let wcc = ccs.read(i)
|
||||
if (wcc == null) { continue }
|
||||
let json = JSON.parse(wcc.getJson())
|
||||
json = this.colorJson(json)
|
||||
if (event.getPlayer().hasPermission('MiaoRGBSupport.rainbow')) {
|
||||
json = this.rainbowJson(json)
|
||||
}
|
||||
wcc.setJson(JSON.stringify(json))
|
||||
ccs.write(i, wcc)
|
||||
}
|
||||
} catch (error) {
|
||||
console.ex(error)
|
||||
}
|
||||
})
|
||||
this.ProtocolLibrary.getProtocolManager().addPacketListener(this.adapter)
|
||||
}
|
||||
}
|
||||
253
packages/plugins/src/MiaoRebate.ts
Normal file
253
packages/plugins/src/MiaoRebate.ts
Normal file
@@ -0,0 +1,253 @@
|
||||
/// <reference types="@javatypes/bungee-api" />
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
|
||||
import { constants, plugin, server, task } from '@ccms/api'
|
||||
import { Autowired, JSClass } from '@ccms/container'
|
||||
import { Cmd, Config, interfaces, JSPlugin, Listener, PluginConfig, Tab } from '@ccms/plugin'
|
||||
|
||||
import type { MiaoReward } from './MiaoReward'
|
||||
import http from '@ccms/common/dist/http'
|
||||
|
||||
interface Sync {
|
||||
scaned: boolean
|
||||
start?: number
|
||||
left?: number
|
||||
cancelled?: boolean
|
||||
paying?: boolean
|
||||
}
|
||||
|
||||
interface PlaceholderAPI {
|
||||
registerPlaceholderHook: (key: string, onPlaceholderRequest: (player, s) => string) => void
|
||||
unregisterPlaceholderHook: (key: string) => void
|
||||
setPlaceholders: (player: any, str: string) => string
|
||||
}
|
||||
|
||||
const defaultConfig = {
|
||||
prefix: '§6[§b外卖系统§6]§r',
|
||||
check: '%playerpoints_points%',
|
||||
command: 'points give %player_name% %amount%',
|
||||
ratio: 1,
|
||||
coinName: '点券',
|
||||
joinTip: true,
|
||||
owner: {
|
||||
userid: '',
|
||||
ccid: '',
|
||||
openid: ''
|
||||
}
|
||||
}
|
||||
|
||||
@JSPlugin({ version: '1.0.1', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], depends: ['MiaoReward'], nativeDepends: ['PlaceholderAPI'], source: __filename })
|
||||
export class MiaoRebate extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private server: server.Server
|
||||
@Autowired()
|
||||
private taskManager: task.TaskManager
|
||||
@Autowired()
|
||||
private pluginManager: plugin.PluginManager
|
||||
|
||||
@JSClass('me.clip.placeholderapi.PlaceholderAPI')
|
||||
private PlaceholderAPI: PlaceholderAPI
|
||||
|
||||
private apiGateWay = "https://rebate.yumc.pw"
|
||||
private MiaoReward: MiaoReward
|
||||
|
||||
@Config({ default: defaultConfig })
|
||||
private config: PluginConfig & typeof defaultConfig = defaultConfig
|
||||
|
||||
load() {
|
||||
this.logger.prefix = this.config.prefix
|
||||
}
|
||||
|
||||
enable() {
|
||||
this.MiaoReward = this.pluginManager.getPlugin('MiaoReward') as MiaoReward
|
||||
if (!this.MiaoReward) { return this.logger.error(`当前脚本插件需要 MiaoReward 作为前置脚本插件!`) }
|
||||
}
|
||||
|
||||
disable() {
|
||||
}
|
||||
|
||||
@Cmd({ autoMain: true })
|
||||
mre() { }
|
||||
|
||||
cmdbind(sender: org.bukkit.entity.Player, server: boolean) {
|
||||
if (!sender.getItemInHand) { return this.logger.sender(sender, `§c手持物品检测异常 请检查是否在客户端执行命令!`) }
|
||||
if (server) { return this.bindServer(sender) }
|
||||
if (!sender.getItemInHand) { return this.logger.sender(sender, `§c手持物品检测异常 请检查是否在客户端执行命令!`) }
|
||||
if (!this.config.owner.openid || !this.config.owner.userid) { return this.logger.sender(sender, `§4当前服务器尚未绑定管理员账号 请联系管理员完成绑定!`) }
|
||||
this.MiaoReward.sendTitle(sender, `§a获取二维码中`, `§6请稍候...`)
|
||||
let scan = this.qrCreate(sender, `绑定成功 请返回游戏查看!`, {
|
||||
v: 1,
|
||||
type: "invite",
|
||||
user: this.config.owner.openid,
|
||||
userid: this.config.owner.userid,
|
||||
})
|
||||
this.createScanTask(sender, scan.url, `微信扫码绑定账号`, `微信扫码 点击关注 绑定账号`, (sender) => {
|
||||
this.MiaoReward.sendTitle(sender, `§a绑定成功!`, `§6已绑定用户: §b${this.qrGet(scan.token).user.username}`)
|
||||
})
|
||||
}
|
||||
|
||||
cmddraw(sender: org.bukkit.entity.Player, amount: number) {
|
||||
if (!sender.getItemInHand) { return this.logger.sender(sender, `§c手持物品检测异常 请检查是否在客户端执行命令!`) }
|
||||
if (!this.config.owner.openid || !this.config.owner.userid) { return this.logger.sender(sender, `§4当前服务器尚未绑定管理员账号 请联系管理员完成绑定!`) }
|
||||
amount = Number(amount)
|
||||
if (!Number.isInteger(amount)) {
|
||||
return this.logger.sender(sender, `§4兑换金额必须是数字!`)
|
||||
}
|
||||
if (amount < 1) {
|
||||
return this.logger.sender(sender, `§4兑换金额必须大于1!`)
|
||||
}
|
||||
this.MiaoReward.sendTitle(sender, `§a获取二维码中`, `§6请稍候...`)
|
||||
let scan = this.qrCreate(sender, ``, {
|
||||
v: 1,
|
||||
type: "draw",
|
||||
ccid: this.config.owner.ccid,
|
||||
userid: this.config.owner.userid,
|
||||
amount
|
||||
})
|
||||
this.createScanTask(sender, scan.url, `微信扫码兑换奖励`, `微信扫码兑换奖励`, (sender) => {
|
||||
let result = this.qrGet(scan.token)
|
||||
this.MiaoReward.sendTitle(sender, `§a扫码成功`, `§a兑换奖励中 §b具体结果请查看公众号消息...`)
|
||||
if (!(result = result.result)) {
|
||||
return this.sendError(sender, amount, `§4服务器返回数据异常!`)
|
||||
}
|
||||
if (!result.success) {
|
||||
return this.sendError(sender, amount, `§c` + result.message)
|
||||
}
|
||||
this.logger.sender(sender, `§a` + result.message)
|
||||
this.taskManager.callSyncMethod(() => {
|
||||
let point = this.safeMultiply(amount, this.config.ratio)
|
||||
let command = this.config.command.replace(`%player_name%`, sender.getName()).replace(`%amount%`, `${point}`)
|
||||
if (!this.server.dispatchConsoleCommand(command)) {
|
||||
return this.sendError(sender, amount, `§4充值命令执行异常!`)
|
||||
}
|
||||
let nowPoint = this.getPlayerAmount(sender)
|
||||
this.logger.sender(sender, [
|
||||
`§6充值 §a${point} §6${this.config.coinName} §a成功 §6当前账户余额: §3${nowPoint} §6${this.config.coinName}`,
|
||||
`§c如出现未到账的情况 请联系管理员!`
|
||||
])
|
||||
})
|
||||
})
|
||||
this.MiaoReward.clearTitle(sender)
|
||||
}
|
||||
private safeMultiply(a: number, b: number) {
|
||||
return parseFloat((a * b).toFixed(0))
|
||||
}
|
||||
sendError(sender: org.bukkit.entity.Player, amount: number, error: string) {
|
||||
return this.logger.sender(sender, [
|
||||
`§c========== ${this.config.prefix}§4兑换异常 §c==========`,
|
||||
`§6兑换圈币: §3${amount}`,
|
||||
`§6异常原因: §4${error}`,
|
||||
`§6异常账号: §b${sender.getName()}`,
|
||||
`§6异常时间: §a${new Date().toLocaleDateString()} ${new Date().toLocaleTimeString()}`,
|
||||
`§c如果已扣除圈币但${this.config.coinName}未到账 请截图发给腐竹!`,
|
||||
`§c========== ${this.config.prefix}§4兑换异常 §c==========`,
|
||||
])
|
||||
}
|
||||
|
||||
private getPlayerAmount(sender: any): number {
|
||||
let result = this.PlaceholderAPI.setPlaceholders(sender, this.config.check)
|
||||
let amount = parseFloat(result)
|
||||
if (isNaN(amount)) {
|
||||
throw new Error(`§c读取玩家 §3${this.config.coinName} §c异常 §6请检查 §3check §6配置是否正确!
|
||||
§6数据解析链路: §3${this.config.check} §6=> §3${result} §6=> §3${amount}`)
|
||||
}
|
||||
return amount
|
||||
}
|
||||
private bindServer(sender: org.bukkit.entity.Player) {
|
||||
if (!sender.isOp()) { return this.logger.sender(sender, `§4您没有配置服务器的权限!`) }
|
||||
if (this.config.owner.openid || this.config.owner.userid) {
|
||||
this.logger.sender(sender, `§c更换管理员账号 历史绑定数据将不会迁移!`)
|
||||
}
|
||||
let scan = this.qrCreate(sender, `绑定成功 请返回游戏查看!`)
|
||||
this.createScanTask(sender, scan.url, `微信扫码绑定账号`, `微信扫码 点击关注 绑定账号`, (sender) => {
|
||||
let result = this.qrGet(scan.token)
|
||||
let user = result.user
|
||||
this.config.owner.userid = user.id
|
||||
this.config.owner.ccid = user.ccid
|
||||
this.config.owner.openid = result.openid
|
||||
this.config.save()
|
||||
this.MiaoReward.sendTitle(sender, `§a绑定成功!`, `§6已绑定用户: §b${user.username}`)
|
||||
})
|
||||
this.MiaoReward.clearTitle(sender)
|
||||
}
|
||||
|
||||
private createScanTask(sender: org.bukkit.entity.Player, qrcode: string, name: string, tip: string, task: (sender: org.bukkit.entity.Player) => void) {
|
||||
let sync: any = { scaned: false, start: Math.round(Date.now() / 1000) }
|
||||
this.MiaoReward.setItemAndTp(sender, qrcode, sync, name, tip)
|
||||
this.taskManager.create(() => {
|
||||
try {
|
||||
task(sender)
|
||||
} catch (error) {
|
||||
if (!sync.cancelled) {
|
||||
this.logger.sender(sender, `§c` + error)
|
||||
}
|
||||
} finally {
|
||||
sync.scaned = true
|
||||
sender.updateInventory()
|
||||
}
|
||||
}).async().submit()
|
||||
this.MiaoReward.clearTitle(sender)
|
||||
}
|
||||
|
||||
@Listener()
|
||||
private PlayerJoinEvent(event: org.bukkit.event.player.PlayerJoinEvent) {
|
||||
if (this.config.joinTip) {
|
||||
this.taskManager.create(() => {
|
||||
this.logger.sender(event.getPlayer(), [
|
||||
`§b本服已和§a饿了么§6美团§c达成战略合作!`,
|
||||
`§3/mre bind §a扫码§e免费赠送§a外卖红包!`
|
||||
])
|
||||
}).later(30).submit()
|
||||
}
|
||||
}
|
||||
|
||||
@Tab()
|
||||
tabmre(sender: any, _command: any, args: string | any[]) {
|
||||
if (args.length === 2 && args[0] === "bind" && sender.isOp()) return [`server`]
|
||||
}
|
||||
|
||||
private qrCreate(sender: org.bukkit.entity.Player, message: string, data: any = {}) {
|
||||
let create = this.httpPost(`/qr/create/type/login/message/${encodeURIComponent(message)}`, data)
|
||||
if (create.code != 200) {
|
||||
return this.logger.sender(sender, `§c获取链接异常: ` + create.msg)
|
||||
}
|
||||
return create.data
|
||||
}
|
||||
|
||||
private qrGet(token: string) {
|
||||
let get = this.httpPost('/qr/get', { token })
|
||||
if (get.code != 200) {
|
||||
throw new Error(get.msg)
|
||||
}
|
||||
return get.data
|
||||
}
|
||||
|
||||
private httpPost(method: string, data: any = {}) {
|
||||
let startTime = Date.now()
|
||||
let url = `${this.apiGateWay}${method}`
|
||||
let result = http.post(url, data)
|
||||
console.debug(`
|
||||
====== HTTP POST ======
|
||||
REQUEST URL : ${url}
|
||||
REQUEST DATA: ${JSON.stringify(data)}
|
||||
RESPONSE : ${JSON.stringify(result)}
|
||||
CAST TIME : ${Date.now() - startTime}`)
|
||||
return result
|
||||
}
|
||||
|
||||
private cmdhelp(sender: any) {
|
||||
let help = [
|
||||
`§6====== ${this.config.prefix} §a帮助菜单 §6======`,
|
||||
`§6/mre bind §a绑定账号`,
|
||||
`§6/mre draw §e<兑换数量> §a兑换${this.config.coinName}`
|
||||
]
|
||||
if (sender.isOp()) {
|
||||
help = help.concat([
|
||||
`§c由于您是管理员 以为您展示额外命令`,
|
||||
`§6/mre bind server §a绑定服务器`,
|
||||
])
|
||||
}
|
||||
this.logger.sender(sender, help)
|
||||
}
|
||||
}
|
||||
@@ -12,10 +12,19 @@ const Bytes = Java.type('byte[]')
|
||||
|
||||
interface PlaceholderAPI {
|
||||
registerPlaceholderHook: (key: string, onPlaceholderRequest: (player, s) => string) => void
|
||||
registerExpansion: (expansion: PlaceholderExpansion) => void
|
||||
unregisterPlaceholderHook: (key: string) => void
|
||||
setPlaceholders: (player: any, str: string) => string
|
||||
}
|
||||
|
||||
interface PlaceholderExpansion {
|
||||
getIdentifier: () => string
|
||||
getAuthor: () => string
|
||||
getVersion: () => string
|
||||
persist: () => string
|
||||
onPlaceholderRequest: (player, s) => string
|
||||
}
|
||||
|
||||
interface UserInfo {
|
||||
balance: number
|
||||
sign: string
|
||||
@@ -54,14 +63,15 @@ const defaultConfig = {
|
||||
prefix: '§6[§b广告系统§6]§r',
|
||||
serverId: '',
|
||||
serverToken: '',
|
||||
drawCommand: 'p give %player_name% %amount%',
|
||||
drawCommand: 'points give %player_name% %amount%',
|
||||
coinName: '点券',
|
||||
joinTip: true
|
||||
}
|
||||
|
||||
@JSPlugin({ prefix: 'MRD', version: '1.5.1', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], nativeDepends: ['ProtocolLib', 'PlaceholderAPI'], source: __filename })
|
||||
@JSPlugin({ prefix: 'MRD', version: '1.6.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], nativeDepends: ['ProtocolLib', 'PlaceholderAPI'], source: __filename })
|
||||
export class MiaoReward extends interfaces.Plugin {
|
||||
public serverInfo: ServerInfo
|
||||
private notifyError = true
|
||||
private cacheBindUuid = ''
|
||||
private zeroMapView = undefined
|
||||
private playerImageCache = new Map<string, any>()
|
||||
@@ -69,6 +79,7 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
private playerInfoCache = new Map<string, UserInfo>()
|
||||
|
||||
private downgrade = false
|
||||
private subversion = 0
|
||||
|
||||
@Autowired()
|
||||
private chat: chat.Chat
|
||||
@@ -91,6 +102,9 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
private PlaceholderAPI: PlaceholderAPI
|
||||
@JSClass('me.clip.placeholderapi.PlaceholderHook')
|
||||
private PlaceholderHook: any
|
||||
@JSClass('me.clip.placeholderapi.expansion.PlaceholderExpansion')
|
||||
private PlaceholderExpansion: any
|
||||
private expansion: any
|
||||
|
||||
@JSClass('com.comphenix.protocol.ProtocolLibrary')
|
||||
private ProtocolLibrary: any
|
||||
@@ -108,17 +122,18 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
load() {
|
||||
this.config.prefix = this.config.prefix || '§6[§b广告系统§6]§r'
|
||||
this.config.drawCommand = this.config.drawCommand || 'p give %player_name% %amount%'
|
||||
if (!this.config.coinName) {
|
||||
if (this.config.coinName == undefined) {
|
||||
this.config.coinName = '点券'
|
||||
this.config.save()
|
||||
}
|
||||
if (!this.config.joinTip) {
|
||||
if (this.config.joinTip == undefined) {
|
||||
this.config.joinTip = true
|
||||
this.config.save()
|
||||
}
|
||||
//@ts-ignore
|
||||
this.logger.prefix = this.config.prefix
|
||||
this.downgrade = this.Bukkit.server.class.name.split('.')[3] == "v1_7_R4"
|
||||
this.subversion = parseInt(this.Bukkit.server.class.name.split('.')[3].split('_')[1])
|
||||
this.updateServerInfo(null, () => this.updateOnlinePlayersInfo())
|
||||
}
|
||||
|
||||
@@ -191,21 +206,28 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
if (!this.PlaceholderAPI) {
|
||||
console.console("§cCan't found me.clip.placeholderapi.PlaceholderAPI variable will not be replaced!")
|
||||
} else {
|
||||
this.PlaceholderAPI.registerPlaceholderHook("mrd", new this.PlaceholderHook({
|
||||
onPlaceholderRequest: (player: any, s: string) => {
|
||||
if (!this.playerInfoCache.has(player.getName())) { return '数据加载中' }
|
||||
let data = this.playerInfoCache.get(player.getName())
|
||||
if (!data) { return '用户未绑定' }
|
||||
switch (s.toLowerCase()) {
|
||||
case "balance": return data.balance
|
||||
case "sign": return data.sign
|
||||
case "video": return data.video
|
||||
case "box": return data.box
|
||||
case "block": return data.block
|
||||
default: return "未知的参数: " + s
|
||||
}
|
||||
}
|
||||
}))
|
||||
this.expansion = new this.PlaceholderExpansion({
|
||||
getIdentifier: () => 'mrd',
|
||||
persist: () => true,
|
||||
getAuthor: () => 'MiaoWoo',
|
||||
getVersion: () => '1.0.0',
|
||||
onPlaceholderRequest: this.onPlaceholderRequest.bind(this)
|
||||
})
|
||||
this.taskManager.create(() => this.expansion.register()).submit()
|
||||
}
|
||||
}
|
||||
|
||||
private onPlaceholderRequest(player: any, s: string) {
|
||||
if (!this.playerInfoCache.has(player.getName())) { return '数据加载中' }
|
||||
let data = this.playerInfoCache.get(player.getName())
|
||||
if (!data) { return '用户未绑定' }
|
||||
switch (s.toLowerCase()) {
|
||||
case "balance": return data.balance
|
||||
case "sign": return data.sign
|
||||
case "video": return data.video
|
||||
case "box": return data.box
|
||||
case "block": return data.block
|
||||
default: return "未知的参数: " + s
|
||||
}
|
||||
}
|
||||
|
||||
@@ -228,29 +250,50 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
if (!this.ProtocolLibrary) {
|
||||
return this.logger.console(`§4服务器未安装 ProtocolLib 无法扫码功能 请安装后重试!`)
|
||||
}
|
||||
this.adapter = this.createPacketAdapter((event) => {
|
||||
let writer = undefined
|
||||
if (this.downgrade) {
|
||||
writer = (packet, bytes) => {
|
||||
// let xbytes = new Bytes(131)
|
||||
let origin = packet.getByteArrays().read(0)
|
||||
// xbytes[1] = origin[1]
|
||||
// xbytes[2] = origin[2]
|
||||
for (let y = 0; y < 128; ++y) {
|
||||
origin[y + 3] = bytes[y * 128 + origin[1]]
|
||||
}
|
||||
packet.getByteArrays().write(0, origin)
|
||||
}
|
||||
} else if (this.subversion < 17) {
|
||||
writer = (packet, bytes) => {
|
||||
packet.getByteArrays().write(0, bytes)
|
||||
packet.getIntegers().write(3, 128)
|
||||
packet.getIntegers().write(4, 128)
|
||||
}
|
||||
} else if (this.subversion > 16) {
|
||||
writer = (packet, bytes) => {
|
||||
let b = packet.getModifier().read(4)
|
||||
if (b) {
|
||||
let bi = Java.type(b.class.name)
|
||||
packet.getModifier().write(4, new bi(b.a, b.b, 128, 128, bytes))
|
||||
}
|
||||
}
|
||||
}
|
||||
if (writer) {
|
||||
this.adapter = this.createPacketAdapter(this.getPacketAdapter(writer))
|
||||
this.ProtocolLibrary.getProtocolManager().addPacketListener(this.adapter)
|
||||
} else {
|
||||
console.console('§4当前服务器不支持虚拟地图发包 将无法使用扫码功能!')
|
||||
}
|
||||
}
|
||||
|
||||
private getPacketAdapter(writer: (packet: any, bytes: number[]) => void) {
|
||||
return (event) => {
|
||||
let integers = event.getPacket().getIntegers().getValues()
|
||||
let mapId = integers.get(0)
|
||||
let player = event.getPlayer()
|
||||
if (mapId == this.zeroMapView.getId() && this.playerImageCache.has(player.getName())) {
|
||||
let bytes = this.playerImageCache.get(player.getName())
|
||||
if (!this.downgrade) {
|
||||
event.getPacket().getByteArrays().write(0, bytes)
|
||||
event.getPacket().getIntegers().write(3, 128)
|
||||
event.getPacket().getIntegers().write(4, 128)
|
||||
} else {
|
||||
// let xbytes = new Bytes(131)
|
||||
let origin = event.getPacket().getByteArrays().read(0)
|
||||
// xbytes[1] = origin[1]
|
||||
// xbytes[2] = origin[2]
|
||||
for (let y = 0; y < 128; ++y) {
|
||||
origin[y + 3] = bytes[y * 128 + origin[1]]
|
||||
}
|
||||
event.getPacket().getByteArrays().write(0, origin)
|
||||
}
|
||||
writer(event.getPacket(), this.playerImageCache.get(player.getName()))
|
||||
}
|
||||
})
|
||||
this.ProtocolLibrary.getProtocolManager().addPacketListener(this.adapter)
|
||||
}
|
||||
}
|
||||
|
||||
private sendWindowItems(player: org.bukkit.entity.Player, mapItem: any) {
|
||||
@@ -261,10 +304,19 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
java.util.Arrays.fill(arritemStack, new org.bukkit.inventory.ItemStack(org.bukkit.Material.AIR))
|
||||
arritemStack[36 + player.getInventory().getHeldItemSlot()] = mapItem
|
||||
var packetContainer = protocolManager.createPacket(this.PacketType.Play.Server.WINDOW_ITEMS)
|
||||
if (packetContainer.getItemArrayModifier().size() > 0) {
|
||||
try {
|
||||
packetContainer.getItemArrayModifier().write(0, arritemStack)
|
||||
} else {
|
||||
packetContainer.getItemListModifier().write(0, java.util.Arrays.asList(arritemStack))
|
||||
} catch (error) {
|
||||
try {
|
||||
packetContainer.getItemListModifier().write(0, java.util.Arrays.asList(arritemStack))
|
||||
} catch (error) {
|
||||
if (this.notifyError) {
|
||||
console.console('§4发送虚拟物品包失败 可能是ProtocolLib版本不兼容!')
|
||||
console.ex(error)
|
||||
this.notifyError = false
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
protocolManager.sendServerPacket(player, packetContainer)
|
||||
} catch (ex) {
|
||||
@@ -273,8 +325,7 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
}
|
||||
|
||||
disable() {
|
||||
if (!this.ProtocolLibrary) return
|
||||
this.PlaceholderAPI?.unregisterPlaceholderHook("mrd")
|
||||
try { this.expansion?.unregister() } catch (error) { }
|
||||
this.adapter && this.ProtocolLibrary.getProtocolManager().removePacketListener(this.adapter)
|
||||
Java.from(this.server.getOnlinePlayers()).forEach(p => this.checkAndClear(p))
|
||||
this.channelOff?.off()
|
||||
@@ -420,6 +471,7 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
this.logger.sender(sender, draw.msg.split('\n').map(s => s.replace('点券', this.config.coinName)))
|
||||
this.sendBroadcast(sender, `${this.config.prefix}§6玩家 §b${sender.getName()} §6成功将 §a${amount}喵币 §6兑换成 §c${draw.data}${this.config.coinName}!`)
|
||||
this.sendBroadcast(sender, `${this.config.prefix}§c/mrd help §b查看广告系统帮助 §6快来一起看广告赚${this.config.coinName}吧!`)
|
||||
this.queryUser(sender)
|
||||
}).submit()
|
||||
}
|
||||
|
||||
@@ -611,6 +663,7 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
sync.cancelled = false
|
||||
let task = this.taskManager.create(() => {
|
||||
try {
|
||||
console.log(JSON.stringify(sync))
|
||||
if (sync.scaned || !sender.isOnline() || !this.isHoldQrCodeItem(sender) || --sync.left < 0) {
|
||||
if (sync.left < 0) {
|
||||
this.logger.sender(sender, '§c二维码已过期 请重新获取 如已扫码请忽略!')
|
||||
@@ -626,20 +679,7 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
}
|
||||
}, this).async().later(20).timer(20).submit()
|
||||
this.playerTaskCache.set(sender.getName(), task)
|
||||
if (this.downgrade) {
|
||||
this.logger.sender(sender, '§c低版本客户端 二维码渲染中 请等待 3 秒 稍候扫码!')
|
||||
let waitTask = this.taskManager.create(() => {
|
||||
let temp = sender.getLocation()
|
||||
temp.setPitch(-90)
|
||||
sender.teleport(temp)
|
||||
}, this).later(0).timer(20).submit()
|
||||
this.taskManager.create(() => {
|
||||
waitTask.cancel()
|
||||
let temp = sender.getLocation()
|
||||
temp.setPitch(90)
|
||||
sender.teleport(temp)
|
||||
}).later(80).submit()
|
||||
}
|
||||
if (this.downgrade) { this.downgradeTask(sender) }
|
||||
this.playerImageCache.set(sender.getName(), org.bukkit.map.MapPalette.imageToBytes(this.createQrcode(content)))
|
||||
if (!this.downgrade) {
|
||||
let temp = sender.getLocation()
|
||||
@@ -648,9 +688,25 @@ export class MiaoReward extends interfaces.Plugin {
|
||||
}
|
||||
this.sendWindowItems(sender, this.createQrCodeMapItem(name))
|
||||
sender.sendMap(this.zeroMapView)
|
||||
this.taskManager.create(() => this.sendWindowItems(sender, this.createQrCodeMapItem(name))).later(20).async().submit()
|
||||
}).submit()
|
||||
}
|
||||
|
||||
private downgradeTask(sender) {
|
||||
this.logger.sender(sender, '§c低版本客户端 二维码渲染中 请等待 3 秒 稍候扫码!')
|
||||
let waitTask = this.taskManager.create(() => {
|
||||
let temp = sender.getLocation()
|
||||
temp.setPitch(-90)
|
||||
sender.teleport(temp)
|
||||
}, this).later(0).timer(20).submit()
|
||||
this.taskManager.create(() => {
|
||||
waitTask.cancel()
|
||||
let temp = sender.getLocation()
|
||||
temp.setPitch(90)
|
||||
sender.teleport(temp)
|
||||
}).later(80).submit()
|
||||
}
|
||||
|
||||
private queryUser(sender: org.bukkit.entity.Player, sync = false) {
|
||||
if (!this.serverInfo) { return this.logger.sender(sender, '§4当前服务器尚未配置绑定ID 请联系腐竹进行配置!') }
|
||||
let result = this.httpPost(`https://reward.yumc.pw/server/queryUser`, {
|
||||
|
||||
@@ -1,102 +1,220 @@
|
||||
/// <reference types="@javatypes/bungee-api" />
|
||||
/// <reference types="@javatypes/bukkit-api" />
|
||||
/// <reference types="@javatypes/sponge-api" />
|
||||
/// <reference types="typescript" />
|
||||
// @ts-ignore
|
||||
require.clear('websocket/client')
|
||||
import { server } from '@ccms/api'
|
||||
import { Autowired, Container, ContainerInstance } from '@ccms/container'
|
||||
import { Cmd, JSPlugin, Tab, interfaces, PluginConfig, Config } from '@ccms/plugin'
|
||||
|
||||
import { EventEmitter } from 'events'
|
||||
|
||||
import { constants, server } from '@ccms/api'
|
||||
import { Autowired, JSClass } from '@ccms/container'
|
||||
import { Cmd, JSPlugin, Tab, interfaces, PluginConfig, Config, Listener } from '@ccms/plugin'
|
||||
import { WebSocket } from '@ccms/websocket'
|
||||
|
||||
const Thread = Java.type('java.lang.Thread')
|
||||
const ChatColor = Java.type('org.bukkit.ChatColor')
|
||||
|
||||
const defaultConfig = {
|
||||
version: 1,
|
||||
address: '',
|
||||
token: ''
|
||||
token: '',
|
||||
group_id: '',
|
||||
admin_id: '',
|
||||
message: {
|
||||
join: "玩家: %player_name% 加入了服务器!",
|
||||
quit: "玩家: %player_name% 退出了服务器!",
|
||||
chat: "%player_name%: ",
|
||||
group: "&6[&c服务器群&6] &b%sender_nickname%&6(&a%sender_user_id%&6)&r: "
|
||||
}
|
||||
}
|
||||
//https://github3.mk-proxy.ml/-----https://github.com/Mrs4s/go-cqhttp/releases/download/v0.9.34/go-cqhttp-v0.9.34-linux-amd64
|
||||
@JSPlugin({ version: '1.0.0', author: 'MiaoWoo', source: __filename })
|
||||
|
||||
interface RobotConfig {
|
||||
address: string,
|
||||
token: string,
|
||||
timeout: number
|
||||
}
|
||||
|
||||
interface PlaceholderAPI {
|
||||
registerPlaceholderHook: (key: string, onPlaceholderRequest: (player, s) => string) => void
|
||||
unregisterPlaceholderHook: (key: string) => void
|
||||
setPlaceholders: (player: any, str: string) => string
|
||||
}
|
||||
|
||||
class Robot extends EventEmitter {
|
||||
private config: RobotConfig
|
||||
|
||||
private websocket: WebSocket
|
||||
|
||||
private invokeCount = 1;
|
||||
private apiResultCache = [];
|
||||
|
||||
constructor(config: RobotConfig) {
|
||||
super()
|
||||
this.config = config
|
||||
}
|
||||
|
||||
sleep(ms) {
|
||||
Thread.sleep(ms)
|
||||
}
|
||||
|
||||
invoke(action, params) {
|
||||
if (this.websocket.readyState != WebSocket.OPEN) { throw new Error('client not connect!') }
|
||||
let startTime: number = new Date().getTime()
|
||||
let request = { action, params, echo: this.invokeCount++ }
|
||||
this.websocket.send(JSON.stringify(request))
|
||||
while (startTime + this.config.timeout > new Date().getTime()) {
|
||||
if (this.apiResultCache[request.echo]) {
|
||||
let result = this.apiResultCache[request.echo]
|
||||
delete this.apiResultCache[request.echo]
|
||||
if ((result.status === "ok" && result.retcode !== 0) && (result.status === "async" && result !== 1)) {
|
||||
throw Error(`Invoke API Error! Response ${JSON.stringify(result)}`)
|
||||
}
|
||||
return result.data
|
||||
}
|
||||
this.sleep(50)
|
||||
}
|
||||
throw Error(`Invoke API Timeout! Request ${JSON.stringify(request)}`)
|
||||
}
|
||||
|
||||
connect() {
|
||||
this.websocket = new WebSocket(this.config.address, '', { Authorization: `Bearer ${this.config.token}` })
|
||||
this.websocket.onopen = () => {
|
||||
this.emit('connect')
|
||||
}
|
||||
this.websocket.onmessage = (event) => {
|
||||
let robotEvent = JSON.parse(event.data)
|
||||
if (robotEvent.post_type == "meta_event") { return }
|
||||
if (robotEvent.post_type) {
|
||||
this.emit(robotEvent.post_type, robotEvent)
|
||||
}
|
||||
}
|
||||
this.websocket.onclose = (event) => {
|
||||
this.emit('close', event)
|
||||
}
|
||||
this.websocket.onerror = (event) => {
|
||||
this.emit('error', event)
|
||||
}
|
||||
}
|
||||
|
||||
disconnect(reason = '') {
|
||||
if (this.websocket) {
|
||||
this.websocket.close(0, reason)
|
||||
}
|
||||
}
|
||||
|
||||
sendGroupMessage(group_id, message) {
|
||||
this.websocket.send(JSON.stringify({
|
||||
action: "send_msg",
|
||||
params: { group_id, message }
|
||||
}))
|
||||
}
|
||||
|
||||
sendPrivateMessage(user_id, message) {
|
||||
this.websocket.send(JSON.stringify({
|
||||
action: "send_msg",
|
||||
params: { user_id, message }
|
||||
}))
|
||||
}
|
||||
}
|
||||
|
||||
@JSPlugin({ version: '1.0.0', author: 'MiaoWoo', servers: [constants.ServerType.Bukkit], source: __filename, nativeDepends: ['PlaceholderAPI'] })
|
||||
export class MiaoRobot extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private server: server.Server
|
||||
|
||||
private client: WebSocket
|
||||
@JSClass('me.clip.placeholderapi.PlaceholderAPI')
|
||||
private PlaceholderAPI: PlaceholderAPI
|
||||
|
||||
private robot: Robot
|
||||
|
||||
@Config()
|
||||
private config: PluginConfig & typeof defaultConfig = defaultConfig
|
||||
|
||||
load() {
|
||||
}
|
||||
|
||||
private downloadRobot() {
|
||||
//https://api.github.com/repos/Mrs4s/go-cqhttp/releases?per_page=1&page=1
|
||||
this.logger.prefix = ''
|
||||
}
|
||||
|
||||
enable() {
|
||||
if (!this.config.address || !this.config.token) {
|
||||
return this.logger
|
||||
if (this.config.address && this.config.token) {
|
||||
this.cmdconnect(this.server.getConsoleSender())
|
||||
if (!this.config.group_id) {
|
||||
this.logger.console('§c机器人尚未配置绑定服务器群 部分功能将无法使用!')
|
||||
}
|
||||
} else {
|
||||
this.logger.console('§c机器人尚未配置 请参照帖子内容配置机器人!')
|
||||
}
|
||||
this.cmdconnect(this.server.getConsoleSender(), this.config.address, this.config.token)
|
||||
}
|
||||
|
||||
disable() {
|
||||
this.cmdclose(this.server.getConsoleSender())
|
||||
if (this.robot) {
|
||||
this.cmdclose(this.server.getConsoleSender())
|
||||
}
|
||||
}
|
||||
|
||||
@Cmd({ autoMain: true })
|
||||
mbot() { }
|
||||
|
||||
cmdconnect(sender: org.bukkit.entity.Player, address: string = this.config.address, token: string = this.config.token) {
|
||||
if (this.client && this.client.readyState == WebSocket.OPEN) {
|
||||
this.client.close()
|
||||
this.client = undefined
|
||||
}
|
||||
if (!address || !token) {
|
||||
return this.logger.sender(sender, '§4错误 请配置服务器地址和Token!')
|
||||
}
|
||||
try {
|
||||
this.client = new WebSocket(address, '', { Authorization: `Bearer ${token}` })
|
||||
this.initRobot(this.client)
|
||||
} catch (error) {
|
||||
console.ex(error)
|
||||
}
|
||||
this.cmdclose(sender)
|
||||
this.initRobot(sender)
|
||||
}
|
||||
|
||||
private initRobot(client: WebSocket) {
|
||||
client.onopen = () => {
|
||||
this.logger.console(`§3连接到 §b${client.url} §a成功!`)
|
||||
}
|
||||
client.onmessage = (event) => {
|
||||
let messageEvent = JSON.parse(event.data)
|
||||
switch (messageEvent.post_type) {
|
||||
case "message":
|
||||
this.logger.console(`§6接收到 §3群 §b${messageEvent.group_id} §2成员 §a${messageEvent.sender.nickname} §6的消息: §r${messageEvent.message}`)
|
||||
break
|
||||
initRobot(sender) {
|
||||
this.robot = new Robot({ ...this.config, timeout: 60 })
|
||||
this.robot.on('connect', () => {
|
||||
this.logger.sender(sender, '§a机器人链接成功!')
|
||||
})
|
||||
this.robot.on('message', (event) => {
|
||||
if (event.message_type == "group" && event.group_id == this.config.group_id) {
|
||||
let message: string = event.message
|
||||
message = message.replace(/.*\[CQ:image\,file=(.*),url=(.*),.*]/g, '[图片]')
|
||||
message = this.config.message.group
|
||||
.replace(/%sender_nickname%/g, event.sender.nickname)
|
||||
.replace(/%sender_card%/g, event.sender.card)
|
||||
.replace(/%sender_title%/g, event.sender.title)
|
||||
.replace(/%sender_user_id%/g, event.sender.user_id) + message
|
||||
message = ChatColor.translateAlternateColorCodes('&', message)
|
||||
this.server.getOnlinePlayers().forEach(p => this.logger.sender(p, message))
|
||||
this.logger.console(message)
|
||||
}
|
||||
}
|
||||
client.onclose = (event) => {
|
||||
this.logger.console(`§4连接已断开 §6Code: §3${event.code} §6原因: §c${event.reason}!`)
|
||||
}
|
||||
client.onerror = (event) => {
|
||||
this.logger.console(`§4发生错误: §r${event.error}`)
|
||||
console.ex(event.error)
|
||||
}
|
||||
})
|
||||
this.robot.connect()
|
||||
}
|
||||
|
||||
cmdclose(sender: org.bukkit.entity.Player) {
|
||||
if (this.client) {
|
||||
this.client.close(0, 'plugin close socket')
|
||||
if (this.robot) {
|
||||
this.robot.disconnect()
|
||||
this.logger.sender(sender, '§c机器人已断开链接!')
|
||||
}
|
||||
}
|
||||
|
||||
cmdsend(sender: org.bukkit.entity.Player, text: string) {
|
||||
if (this.client) {
|
||||
this.client.send(text)
|
||||
this.logger.sender(sender, '§a发送成功!')
|
||||
}
|
||||
this?.robot.sendGroupMessage(this.config.group_id, text)
|
||||
this.logger.sender(sender, '§a发送成功!')
|
||||
}
|
||||
|
||||
@Tab()
|
||||
tabmbot(_sender: any, _command: string, _args: string[]) {
|
||||
return []
|
||||
}
|
||||
|
||||
@Listener()
|
||||
private PlayerJoinEvent(event: org.bukkit.event.player.PlayerJoinEvent) {
|
||||
if (this.robot && this.config.group_id) {
|
||||
this.robot.sendGroupMessage(this.config.group_id, this.PlaceholderAPI.setPlaceholders(event.getPlayer(), this.config.message.join))
|
||||
}
|
||||
}
|
||||
@Listener()
|
||||
private PlayerQuitEvent(event: org.bukkit.event.player.PlayerQuitEvent) {
|
||||
if (this.robot && this.config.group_id) {
|
||||
this.robot.sendGroupMessage(this.config.group_id, this.PlaceholderAPI.setPlaceholders(event.getPlayer(), this.config.message.quit))
|
||||
}
|
||||
}
|
||||
@Listener()
|
||||
private AsyncPlayerChatEvent(event: org.bukkit.event.player.AsyncPlayerChatEvent) {
|
||||
if (this.robot && this.config.group_id) {
|
||||
this.robot.sendGroupMessage(this.config.group_id, this.PlaceholderAPI.setPlaceholders(event.getPlayer(), this.config.message.chat) + event.getMessage())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -11,8 +11,9 @@ let help = [
|
||||
'§6========= §6[§aMiaoScriptPackageManager§6] 帮助 §aBy §bMiaoWoo §6=========',
|
||||
'§6/mspm §ainstall §e<插件名称> §6- §3安装仓库插件',
|
||||
'§6/mspm §aload §e<插件名称> §6- §3安装本地插件',
|
||||
'§6/mspm §aunload §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)',
|
||||
@@ -44,13 +45,15 @@ let langMap = {
|
||||
'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} §a发现新版本 §3{new_version} §6当前版本 §3{old_version}!',
|
||||
'cloud.update.exists': '§6插件 §b{name} §6版本 §3{old_version} §a发现更新 §3{new_version} §r{changelog}§6!',
|
||||
'cloud.update.tip': `§6发现存在 §b{count}个 §6需要更新的插件 请使用 §aupdate §6或 §cupgrade §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更新成功!',
|
||||
'update.finish': '§6插件 §b{name} §6版本 §3{version} §a更新完成!',
|
||||
'update.tip': '§6插件 §b{name} §a更新完成 §6请使用 §areload §6命令重载生效!',
|
||||
'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请重启服务器完成更新...',
|
||||
@@ -106,7 +109,7 @@ class SpongeFakeSender extends FakeSender {
|
||||
}
|
||||
}
|
||||
|
||||
@JSPlugin({ prefix: 'PM', version: '1.3.1', author: 'MiaoWoo', source: __filename })
|
||||
@JSPlugin({ prefix: 'PM', version: '1.5.1', author: 'MiaoWoo', source: __filename })
|
||||
export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
@Autowired()
|
||||
private pluginManager: pluginApi.PluginManager
|
||||
@@ -133,7 +136,6 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
public serverName: string
|
||||
private translate: Translate
|
||||
private channelOff: { off: () => void }
|
||||
private subCommandCache = []
|
||||
|
||||
load() {
|
||||
this.translate = new Translate({
|
||||
@@ -141,7 +143,6 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
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] })
|
||||
@@ -232,7 +233,7 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
this.logger.sender(sender, `§6[§3BPM§6][§a${this.serverName}§6] §6命令 §b/mspm ${args.join?.(' ')} §a发布成功!`)
|
||||
}
|
||||
|
||||
@Cmd({ servers: [constants.ServerType.Bungee] })
|
||||
@Cmd({ alias: ["bmspm"], 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()
|
||||
@@ -332,6 +333,8 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
try {
|
||||
this.i18n(sender, 'upgrade.start')
|
||||
base.delete(enginePath)
|
||||
// @ts-ignore
|
||||
require.setUpgradeMode?.(true)
|
||||
this.cmdrestart(sender)
|
||||
} catch (ex) {
|
||||
if (global.debug) {
|
||||
@@ -359,6 +362,15 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
}
|
||||
}
|
||||
|
||||
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)
|
||||
@@ -395,6 +407,9 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
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) {
|
||||
@@ -422,12 +437,23 @@ export class MiaoScriptPackageManager extends interfaces.Plugin {
|
||||
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) })
|
||||
this.i18n(sender, 'run.result', { result: result == undefined ? this.translate.translate('run.noresult') : typeof result == "string" ? result : this.stringify(result) })
|
||||
} catch (ex) {
|
||||
this.logger.sender(sender, this.logger.stack(ex))
|
||||
}
|
||||
}
|
||||
|
||||
private stringify(object) {
|
||||
let seen = []
|
||||
return JSON.stringify(object, function (key, val) {
|
||||
if (typeof val == "object") {
|
||||
if (seen.indexOf(val) >= 0) return
|
||||
seen.push(val)
|
||||
}
|
||||
return val
|
||||
})
|
||||
}
|
||||
|
||||
private runCode(code: string, sender: any, _this: any) {
|
||||
let paramNames = [
|
||||
'sender',
|
||||
@@ -453,7 +479,7 @@ return eval(${JSON.stringify(code)});`)
|
||||
return tfunc.apply(_this, params)
|
||||
}
|
||||
|
||||
cmddeploy(sender: any, name: any) {
|
||||
cmddeploy(sender: string, name: string, changelog: string = '') {
|
||||
if (!process.env.AccessToken) { return this.i18n(sender, 'deploy.token.not.exists') }
|
||||
this.taskManager.create(() => {
|
||||
if (this.checkPlugin(sender, name)) {
|
||||
@@ -462,7 +488,8 @@ return eval(${JSON.stringify(code)});`)
|
||||
name,
|
||||
author: plugin.description.author,
|
||||
version: plugin.description.version,
|
||||
source: base.read((plugin.description.source || plugin.description.loadMetadata.file).toString())
|
||||
source: base.read((plugin.description.source || plugin.description.loadMetadata.file).toString()),
|
||||
changelog: changelog.replace('&', '§')
|
||||
})
|
||||
this.i18n(sender, result.code == 200 ? 'deploy.success' : 'deploy.fail', { name, version: plugin.description.version, msg: result.msg })
|
||||
}
|
||||
@@ -473,7 +500,11 @@ return eval(${JSON.stringify(code)});`)
|
||||
if (this.checkCloudPlugin(sender, name)) {
|
||||
this.download(sender, name, true, () => {
|
||||
this.i18n(sender, 'update.finish', { name, version: this.packageCache[name].version })
|
||||
callback?.()
|
||||
if (callback) {
|
||||
callback()
|
||||
} else {
|
||||
this.i18n(sender, 'update.tip', { name, version: this.packageCache[name].version })
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -513,13 +544,23 @@ return eval(${JSON.stringify(code)});`)
|
||||
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 })
|
||||
let updateCount = 0
|
||||
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 })
|
||||
this.i18n(sender, 'cloud.update.exists', {
|
||||
name: p.description.name,
|
||||
new_version: cloudPlugin.version,
|
||||
old_version: p.description.version,
|
||||
changelog: cloudPlugin.changelog || ''
|
||||
})
|
||||
updateCount++
|
||||
}
|
||||
})
|
||||
if (updateCount) {
|
||||
this.i18n(sender, 'cloud.update.tip', { count: updateCount })
|
||||
}
|
||||
}).async().submit()
|
||||
}
|
||||
|
||||
@@ -529,7 +570,7 @@ return eval(${JSON.stringify(code)});`)
|
||||
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)
|
||||
http.download(pluginPkg.url + '?t=' + Date.now(), pluginFile)
|
||||
this.i18n(sender, 'download.finish', { name, version: pluginPkg.version })
|
||||
callback?.()
|
||||
}).async().submit()
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "@ccms/ployfill",
|
||||
"version": "0.13.0",
|
||||
"description": "MiaoScript ployfill package",
|
||||
"name": "@ccms/polyfill",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript polyfill package",
|
||||
"author": "MiaoWoo <admin@yumc.pw>",
|
||||
"homepage": "https://github.com/circlecloud/ms.git",
|
||||
"license": "ISC",
|
||||
@@ -14,14 +14,14 @@
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/i18n": "^0.13.0",
|
||||
"@ccms/nodejs": "^0.13.0",
|
||||
"core-js": "^3.7.0"
|
||||
"@ccms/i18n": "^0.17.0",
|
||||
"@ccms/nodejs": "^0.17.0",
|
||||
"core-js": "^3.16.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
@@ -1,9 +1,10 @@
|
||||
/// <reference types="@ccms/nashorn" />
|
||||
import '@ccms/nodejs'
|
||||
import i18n from '@ccms/i18n'
|
||||
let ployfillStartTime = new Date().getTime()
|
||||
let polyfillStartTime = new Date().getTime()
|
||||
i18n.initialize()
|
||||
console.i18n("ms.ployfill.initialize")
|
||||
console.i18n("ms.polyfill.initialize")
|
||||
import './openjdk-nashorn-shim'
|
||||
import './es5-ext'
|
||||
import './node-shim'
|
||||
import 'core-js'
|
||||
@@ -12,4 +13,4 @@ process.on('exit', () => require.disable())
|
||||
global.setGlobal('Proxy', require('./proxy').Proxy)
|
||||
global.setGlobal('XMLHttpRequest', require('./xml-http-request').XMLHttpRequest)
|
||||
global.setGlobal('Blob', require('blob-polyfill').Blob)
|
||||
console.i18n("ms.ployfill.completed", { time: (new Date().getTime() - ployfillStartTime) / 1000 })
|
||||
console.i18n("ms.polyfill.completed", { time: (new Date().getTime() - polyfillStartTime) / 1000 })
|
||||
@@ -2,7 +2,6 @@ import { EventEmitter } from 'events'
|
||||
|
||||
const System = Java.type('java.lang.System')
|
||||
const Thread = Java.type('java.lang.Thread')
|
||||
const Runnable = Java.type('java.lang.Runnable')
|
||||
const InterruptedException = Java.type('java.lang.InterruptedException')
|
||||
const ThreadGroup = Java.type("java.lang.ThreadGroup")
|
||||
const AtomicInteger = Java.type("java.util.concurrent.atomic.AtomicInteger")
|
||||
@@ -40,22 +39,38 @@ class Process extends EventEmitter {
|
||||
return super.on(event, (...args) => {
|
||||
try {
|
||||
listener(...args)
|
||||
} catch (error) {
|
||||
} catch (origin) {
|
||||
try {
|
||||
super.emit('error', error)
|
||||
super.emit('error', origin)
|
||||
} catch (error) {
|
||||
console.ex(origin)
|
||||
console.ex(error)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
nextTick(func: Function) {
|
||||
microTaskPool.execute(func)
|
||||
nextTick(func: Function, ...args: any[]) {
|
||||
microTaskPool.execute(() => {
|
||||
try {
|
||||
func(args)
|
||||
} catch (origin) {
|
||||
try {
|
||||
super.emit('error', origin)
|
||||
} catch (error) {
|
||||
console.ex(origin)
|
||||
console.ex(error)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
exit(code: number) {
|
||||
console.log(`process exit by code ${code}!`)
|
||||
this.emit('exit', code)
|
||||
}
|
||||
|
||||
toString() {
|
||||
return "[object process]"
|
||||
}
|
||||
}
|
||||
|
||||
class EventLoop {
|
||||
@@ -191,10 +206,11 @@ class EventLoop {
|
||||
}
|
||||
}
|
||||
global.setGlobal('process', new Process(), {})
|
||||
Object.defineProperty(process, require('core-js/es/symbol/to-string-tag'), { value: '[object process]' })
|
||||
const eventLoop = new EventLoop()
|
||||
global.setGlobal('eventLoop', eventLoop, {})
|
||||
Object.defineProperty(process, 'eventLoop', { value: eventLoop })
|
||||
eventLoop.startEventLoop()
|
||||
global.setGlobal('queueMicrotask', (func: any) => microTaskPool.execute(func), {})
|
||||
global.setGlobal('queueMicrotask', (func: any, ...args: any[]) => process.nextTick(func, args), {})
|
||||
global.setGlobal('setTimeout', eventLoop.setTimeout.bind(eventLoop), {})
|
||||
global.setGlobal('clearTimeout', eventLoop.clearTimeout.bind(eventLoop), {})
|
||||
global.setGlobal('setInterval', eventLoop.setInterval.bind(eventLoop), {})
|
||||
7
packages/polyfill/src/openjdk-nashorn-shim.ts
Normal file
7
packages/polyfill/src/openjdk-nashorn-shim.ts
Normal file
@@ -0,0 +1,7 @@
|
||||
// fix OpenJDK Nahsorn setPrototypeOf Bug
|
||||
try {
|
||||
Java.type('org.openjdk.nashorn.api.scripting.NashornScriptEngine')
|
||||
Object.setPrototypeOf = ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b })
|
||||
} catch (error) {
|
||||
}
|
||||
export { }
|
||||
@@ -1,4 +1,4 @@
|
||||
import { ProxyHandle } from '@ccms/nashorn'
|
||||
import type { ProxyHandle } from '@ccms/nashorn'
|
||||
|
||||
// Nashorn JSAdapter See https://wiki.openjdk.java.net/display/Nashorn/Nashorn+extensions#Nashornextensions-JSAdapterconstructor
|
||||
let createProxy = eval(`
|
||||
@@ -1,8 +1,6 @@
|
||||
import '@ccms/nashorn'
|
||||
|
||||
const URL = Java.type("java.net.URL")
|
||||
const Files = Java.type("java.nio.file.Files")
|
||||
const StandardCopyOption = Java.type("java.nio.file.StandardCopyOption")
|
||||
const JavaString = Java.type("java.lang.String")
|
||||
const SecureRandom = Java.type("java.security.SecureRandom")
|
||||
const SSLContext = Java.type("javax.net.ssl.SSLContext")
|
||||
@@ -13,8 +11,12 @@ const X509TrustManager = Java.type("javax.net.ssl.X509TrustManager")
|
||||
const SocketTimeoutException = Java.type('java.net.SocketTimeoutException')
|
||||
|
||||
const Callable = Java.type('java.util.concurrent.Callable')
|
||||
const TimeUnit = Java.type('java.util.concurrent.TimeUnit')
|
||||
const Executors = Java.type('java.util.concurrent.Executors')
|
||||
|
||||
const ByteArrayOutputStream = Java.type("java.io.ByteArrayOutputStream")
|
||||
const ByteArray = Java.type("byte[]")
|
||||
|
||||
const UTF_8 = "UTF-8"
|
||||
|
||||
const TrustAnyHostnameVerifier = new HostnameVerifier({ verify: () => true })
|
||||
@@ -72,7 +74,7 @@ type HttpHeader = { [key: string]: string }
|
||||
const executor = Executors.newCachedThreadPool()
|
||||
|
||||
export class XMLHttpRequest {
|
||||
private _timeout: number
|
||||
private _timeout: number = 120000;
|
||||
private _responseType: ResponseType = 'text';
|
||||
private _withCredentials: boolean
|
||||
|
||||
@@ -170,6 +172,7 @@ export class XMLHttpRequest {
|
||||
this._connection.setConnectTimeout(this._timeout)
|
||||
this._connection.setReadTimeout(this._timeout)
|
||||
|
||||
this.setRequestHeader('X-Requested-With', 'XMLHttpRequest')
|
||||
this.setReadyState(ReadyState.OPENED)
|
||||
}
|
||||
send(body?: string | object): Future<string> {
|
||||
@@ -178,7 +181,7 @@ export class XMLHttpRequest {
|
||||
}
|
||||
if (this._readyState !== ReadyState.OPENED) { throw new Error(`Error Status ${this._readyState}!`) }
|
||||
let future = executor.submit(new Callable({ call: () => this._send(body) }))
|
||||
if (!this._async) { future.get() }
|
||||
if (!this._async) { future.get(this._timeout, TimeUnit.MILLISECONDS) }
|
||||
return future
|
||||
}
|
||||
get() {
|
||||
@@ -248,8 +251,16 @@ export class XMLHttpRequest {
|
||||
}
|
||||
|
||||
private readOutput(input: any) {
|
||||
var tempFile = Files.createTempFile('xhr', '.response')
|
||||
Files.copy(input, tempFile, StandardCopyOption['REPLACE_EXISTING']); tempFile.toFile().deleteOnExit()
|
||||
return new JavaString(Files.readAllBytes(tempFile), 'UTF-8')
|
||||
let output = new ByteArrayOutputStream()
|
||||
let buffer = new ByteArray(1024)
|
||||
try {
|
||||
let n: number
|
||||
while ((n = input.read(buffer)) != -1) {
|
||||
output.write(buffer, 0, n)
|
||||
}
|
||||
return output.toString(UTF_8)
|
||||
} finally {
|
||||
output.close()
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/protocol",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript protocol package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,6 +22,6 @@
|
||||
"devDependencies": {
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/sponge",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript api package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -22,11 +22,11 @@
|
||||
"@javatypes/sponge-api": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,12 +1,35 @@
|
||||
import { server } from '@ccms/api'
|
||||
|
||||
const Sponge = org.spongepowered.api.Sponge
|
||||
const Sponge: typeof org.spongepowered.api.Sponge = Java.type('org.spongepowered.api.Sponge')
|
||||
|
||||
export class SpongeNativePluginManager extends server.NativePluginManager {
|
||||
private spongePluginManager: org.spongepowered.api.plugin.PluginManager
|
||||
|
||||
constructor() {
|
||||
super()
|
||||
this.spongePluginManager = Sponge.getPluginManager()
|
||||
}
|
||||
|
||||
list(): server.NativePlugin[] {
|
||||
return Java.from(this.spongePluginManager.getPlugins()).map(plugin => this.convert(plugin))
|
||||
}
|
||||
has(name: string) {
|
||||
return !!this.get(name)
|
||||
return !!this.spongePluginManager.getPlugin(name).orElse(null)
|
||||
}
|
||||
get(name: string) {
|
||||
return Sponge.getPluginManager().getPlugin(name).orElse(null)
|
||||
return this.convert(this.spongePluginManager.getPlugin(name).orElse(null))
|
||||
}
|
||||
|
||||
private convert(plugin: org.spongepowered.api.plugin.PluginContainer): server.NativePlugin {
|
||||
if (!plugin) return plugin as any
|
||||
return {
|
||||
name: plugin.getName(),
|
||||
version: plugin.getVersion().get() as string,
|
||||
authors: Java.from(plugin.getAuthors() as string[]),
|
||||
depends: Java.from(plugin.getDependencies()),
|
||||
softDepends: [],
|
||||
enable: true,
|
||||
origin: plugin
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -3,7 +3,8 @@ import { provideSingleton } from '@ccms/container'
|
||||
|
||||
import * as reflect from '@ccms/common/dist/reflect'
|
||||
|
||||
const Sponge = org.spongepowered.api.Sponge
|
||||
const Sponge: typeof org.spongepowered.api.Sponge = org.spongepowered.api.Sponge
|
||||
const Text: typeof org.spongepowered.api.text.Text = org.spongepowered.api.text.Text
|
||||
const File = Java.type("java.io.File")
|
||||
|
||||
@provideSingleton(server.Server)
|
||||
@@ -30,6 +31,12 @@ export class SpongeServer extends server.ReflectServer {
|
||||
getService(service: string) {
|
||||
return Sponge.getServiceManager().provide(base.getClass(service)).orElse(null)
|
||||
}
|
||||
broadcast(message: string, permission: string) {
|
||||
return Sponge.getServer().getBroadcastChannel().permission(permission).send(Text.of(message) as any)
|
||||
}
|
||||
broadcastMessage(message: string) {
|
||||
return Sponge.getServer().getBroadcastChannel().TO_ALL.send(Text.of(message) as any)
|
||||
}
|
||||
dispatchCommand(sender: string | any, command: string): boolean {
|
||||
if (typeof sender === 'string') {
|
||||
sender = this.getPlayer(sender)
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/spring",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript spring package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -21,12 +21,12 @@
|
||||
"devDependencies": {
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/common": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0",
|
||||
"@ccms/database": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/common": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0",
|
||||
"@ccms/database": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/web",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript web package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -26,10 +26,10 @@
|
||||
"@javatypes/tomcat": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
},
|
||||
"dependencies": {
|
||||
"@ccms/api": "^0.13.0",
|
||||
"@ccms/container": "^0.13.0"
|
||||
"@ccms/api": "^0.17.0",
|
||||
"@ccms/container": "^0.17.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@ccms/websocket",
|
||||
"version": "0.13.0",
|
||||
"version": "0.17.0",
|
||||
"description": "MiaoScript websocket package",
|
||||
"keywords": [
|
||||
"miaoscript",
|
||||
@@ -18,11 +18,15 @@
|
||||
"build": "yarn clean && tsc",
|
||||
"test": "echo \"Error: run tests from root\" && exit 1"
|
||||
},
|
||||
"dependencies": {
|
||||
"backo2": "^1.0.2",
|
||||
"parseuri": "^0.0.6"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@ccms/nashorn": "^0.13.0",
|
||||
"@ccms/nashorn": "^0.17.0",
|
||||
"@javatypes/tomcat-websocket-api": "^0.0.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"rimraf": "^3.0.2",
|
||||
"typescript": "^4.0.5"
|
||||
"typescript": "^4.3.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -23,7 +23,7 @@ export class WebSocketManager {
|
||||
}
|
||||
}
|
||||
|
||||
export const managers = new WebSocketManager()
|
||||
export const manager = new WebSocketManager()
|
||||
|
||||
export class WebSocket extends EventEmitter {
|
||||
public static CONNECTING = 0
|
||||
@@ -31,6 +31,7 @@ export class WebSocket extends EventEmitter {
|
||||
public static CLOSING = 2
|
||||
public static CLOSED = 3
|
||||
public binaryType: 'blob' | 'arraybuffer'
|
||||
protected manager: WebSocketManager
|
||||
|
||||
protected _url: string
|
||||
protected _headers: WebSocketHeader = {}
|
||||
@@ -39,11 +40,13 @@ export class WebSocket extends EventEmitter {
|
||||
|
||||
constructor(url: string, subProtocol: string = '', headers: WebSocketHeader = {}) {
|
||||
super()
|
||||
this.manager = manager
|
||||
this._url = url
|
||||
this._headers = headers
|
||||
try {
|
||||
let TransportImpl = require('./netty').NettyWebSocket
|
||||
this.client = new TransportImpl(url, subProtocol, headers)
|
||||
console.debug('create websocket from ' + this.client.constructor.name)
|
||||
} catch (error) {
|
||||
console.error('create websocket impl error: ' + error)
|
||||
console.ex(error)
|
||||
@@ -51,12 +54,12 @@ export class WebSocket extends EventEmitter {
|
||||
}
|
||||
this.client.on('open', (event) => {
|
||||
this.onopen?.(event)
|
||||
managers.add(this)
|
||||
manager.add(this)
|
||||
})
|
||||
this.client.on('message', (event) => this.onmessage?.(event))
|
||||
this.client.on('close', (event) => {
|
||||
this.onclose?.(event)
|
||||
managers.del(this)
|
||||
manager.del(this)
|
||||
})
|
||||
this.client.on('error', (event) => this.onerror?.(event))
|
||||
setTimeout(() => this.client.connect(), 20)
|
||||
@@ -96,3 +99,4 @@ export class WebSocket extends EventEmitter {
|
||||
this.removeAllListeners()
|
||||
}
|
||||
}
|
||||
global.setGlobal('WebSocket', WebSocket)
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
import { EventEmitter } from 'events'
|
||||
import { NettyWebSocket } from '.'
|
||||
import { WebSocketClientHandlerAdapter } from './adapter/handler'
|
||||
|
||||
@@ -6,6 +5,7 @@ const CharsetUtil = Java.type('io.netty.util.CharsetUtil')
|
||||
const TextWebSocketFrame = Java.type('io.netty.handler.codec.http.websocketx.TextWebSocketFrame')
|
||||
const CloseWebSocketFrame = Java.type('io.netty.handler.codec.http.websocketx.CloseWebSocketFrame')
|
||||
const FullHttpResponse = Java.type('io.netty.handler.codec.http.FullHttpResponse')
|
||||
const DefaultChannelPromise = Java.type('io.netty.channel.DefaultChannelPromise')
|
||||
|
||||
export class WebSocketClientHandler extends WebSocketClientHandlerAdapter {
|
||||
public handshaker: any
|
||||
@@ -20,16 +20,20 @@ export class WebSocketClientHandler extends WebSocketClientHandlerAdapter {
|
||||
return true
|
||||
}
|
||||
handlerAdded(ctx: any) {
|
||||
console.trace(`${ctx} handlerAdded`)
|
||||
this.handshakeFuture = ctx.newPromise()
|
||||
console.debug(`${ctx} handlerAdded`)
|
||||
if (ctx.newPromise) {
|
||||
this.handshakeFuture = ctx.newPromise()
|
||||
} else {
|
||||
this.handshakeFuture = new DefaultChannelPromise(ctx.channel(), ctx.executor())
|
||||
}
|
||||
}
|
||||
channelActive(ctx: any) {
|
||||
console.trace(`${ctx} channelActive`)
|
||||
console.debug(`${ctx} channelActive`)
|
||||
this.handshaker.handshake(ctx.channel())
|
||||
}
|
||||
channelInactive(ctx: any) {
|
||||
console.trace(`${ctx} channelInactive`)
|
||||
this.client.onclose({ code: 0, reason: 'server connection channel inactive!' })
|
||||
console.debug(`${ctx} channelInactive`)
|
||||
this.client.onclose({ code: 0, reason: 'client connection channel inactive!' })
|
||||
}
|
||||
channelRead0(ctx: any, msg: any) {
|
||||
console.trace(`${ctx} channelRead0 ${msg}`)
|
||||
@@ -50,11 +54,11 @@ export class WebSocketClientHandler extends WebSocketClientHandlerAdapter {
|
||||
if (frame instanceof TextWebSocketFrame) {
|
||||
this.client.onmessage({ data: frame.text() })
|
||||
} else if (frame instanceof CloseWebSocketFrame) {
|
||||
this.client.onclose({ code: 0, reason: 'server send CloseWebSocketFrame!' })
|
||||
this.client.onclose({ code: 0, reason: 'server close connection!' })
|
||||
}
|
||||
}
|
||||
exceptionCaught(ctx: any, cause: Error) {
|
||||
console.trace(`${ctx} exceptionCaught ${cause}`)
|
||||
console.debug(`${ctx} exceptionCaught ${cause}`)
|
||||
this.client.onerror({ error: cause })
|
||||
if (!this.handshakeFuture.isDone()) {
|
||||
this.handshakeFuture.setFailure(cause)
|
||||
|
||||
@@ -4,16 +4,12 @@ import { Transport } from '../transport'
|
||||
import { WebSocketClientHandler } from './handler'
|
||||
|
||||
const URI = Java.type('java.net.URI')
|
||||
const Epoll = Java.type('io.netty.channel.epoll.Epoll')
|
||||
const Bootstrap = Java.type('io.netty.bootstrap.Bootstrap')
|
||||
const ChannelFutureListener = Java.type('io.netty.channel.ChannelFutureListener')
|
||||
|
||||
const NioEventLoopGroup = Java.type('io.netty.channel.nio.NioEventLoopGroup')
|
||||
const NioSocketChannel = Java.type('io.netty.channel.socket.nio.NioSocketChannel')
|
||||
|
||||
const EpollEventLoopGroup = Java.type('io.netty.channel.epoll.EpollEventLoopGroup')
|
||||
const EpollSocketChannel = Java.type('io.netty.channel.epoll.EpollSocketChannel')
|
||||
|
||||
const WebSocketClientHandshakerFactory = Java.type('io.netty.handler.codec.http.websocketx.WebSocketClientHandshakerFactory')
|
||||
const WebSocketVersion = Java.type('io.netty.handler.codec.http.websocketx.WebSocketVersion')
|
||||
|
||||
@@ -25,22 +21,77 @@ const CloseWebSocketFrame = Java.type('io.netty.handler.codec.http.websocketx.Cl
|
||||
const ChannelInitializer = Java.type('io.netty.channel.ChannelInitializer')
|
||||
const DefaultHttpHeaders = Java.type('io.netty.handler.codec.http.DefaultHttpHeaders')
|
||||
|
||||
const epull = Epoll.isAvailable()
|
||||
const group = epull ? new EpollEventLoopGroup() : new NioEventLoopGroup()
|
||||
const socketChannelClass = epull ? EpollSocketChannel.class : NioSocketChannel.class
|
||||
const AtomicInteger = Java.type("java.util.concurrent.atomic.AtomicInteger")
|
||||
const channelCount = new AtomicInteger(0)
|
||||
|
||||
var SslContextBuilder: any
|
||||
var InsecureTrustManagerFactory: any
|
||||
var SSLContext: any
|
||||
var SslHandler: any
|
||||
try {
|
||||
SslContextBuilder = Java.type('io.netty.handler.ssl.SslContextBuilder')
|
||||
InsecureTrustManagerFactory = Java.type('io.netty.handler.ssl.util.InsecureTrustManagerFactory')
|
||||
} catch (error) {
|
||||
SSLContext = Java.type('javax.net.ssl.SSLContext')
|
||||
SslHandler = Java.type('io.netty.handler.ssl.SslHandler')
|
||||
}
|
||||
|
||||
var group: any
|
||||
var socketChannelClass: any
|
||||
try {
|
||||
const Epoll = Java.type('io.netty.channel.epoll.Epoll')
|
||||
const epull = Epoll.isAvailable()
|
||||
const EpollEventLoopGroup = Java.type('io.netty.channel.epoll.EpollEventLoopGroup')
|
||||
const EpollSocketChannel = Java.type('io.netty.channel.epoll.EpollSocketChannel')
|
||||
group = epull ? new EpollEventLoopGroup() : new NioEventLoopGroup()
|
||||
socketChannelClass = epull ? EpollSocketChannel.class : NioSocketChannel.class
|
||||
} catch (error) {
|
||||
group = new NioEventLoopGroup()
|
||||
socketChannelClass = NioSocketChannel.class
|
||||
}
|
||||
|
||||
process.on('exit', () => group.shutdownGracefully())
|
||||
|
||||
export class NettyWebSocket extends Transport {
|
||||
private _uri: any
|
||||
private _schema: string
|
||||
private _host: string
|
||||
private _port: number
|
||||
private channel: any
|
||||
private b = new Bootstrap();
|
||||
|
||||
constructor(url: string, subProtocol: string = '', headers: WebSocketHeader = {}) {
|
||||
super(url, subProtocol, headers)
|
||||
if (!url) {
|
||||
throw new Error("Failed to construct 'WebSocket': The URL '" + url + "' is invalid.")
|
||||
}
|
||||
this._uri = URI.create(this._url)
|
||||
this._schema = this._uri.getScheme() ?? 'ws'
|
||||
if (["wss", "ws"].indexOf(this._schema) == -1) {
|
||||
throw new Error("Failed to construct 'WebSocket': The URL's scheme must be either 'ws' or 'wss'. '" + this._schema + "' is not allowed.")
|
||||
}
|
||||
this._host = this._uri.getHost()
|
||||
if (!this._host) {
|
||||
throw new Error("Failed to construct 'WebSocket': The Host '" + this._host + "' is invalid.")
|
||||
}
|
||||
this._port = this._uri.getPort()
|
||||
if (this._port == -1) {
|
||||
if (this._schema == "wss") {
|
||||
this._port = 443
|
||||
} else if (this._schema == "ws") {
|
||||
this._port = 80
|
||||
}
|
||||
}
|
||||
console.debug(`constructor NettyWebSocket url: ${url} scheme: ${this._schema} host: ${this._host} port: ${this._port} header: ${JSON.stringify(headers)}`)
|
||||
}
|
||||
getId() {
|
||||
return this.channel?.id() + ''
|
||||
if (this.channel?.id) {
|
||||
return this.channel?.id() + ''
|
||||
}
|
||||
return 'NettyWebSocket#' + channelCount.incrementAndGet()
|
||||
}
|
||||
doConnect() {
|
||||
console.debug('client NettyWebSocket doConnect', this._url)
|
||||
let uri = URI.create(this._url)
|
||||
let headers = new DefaultHttpHeaders()
|
||||
for (const key of Object.getOwnPropertyNames(this._headers || {})) {
|
||||
@@ -56,20 +107,32 @@ export class NettyWebSocket extends Transport {
|
||||
.handler(new ChannelInitializer({
|
||||
initChannel: (ch: any) => {
|
||||
let pipeline = ch.pipeline()
|
||||
if (this._schema == "wss") {
|
||||
if (SslContextBuilder) {
|
||||
let sslCtx = SslContextBuilder.forClient().trustManager(InsecureTrustManagerFactory.INSTANCE).build()
|
||||
pipeline.addLast(sslCtx.newHandler(ch.alloc(), this._host, this._port))
|
||||
} else {
|
||||
let sslEngine = SSLContext.getDefault().createSSLEngine()
|
||||
sslEngine.setUseClientMode(true)
|
||||
pipeline.addLast("ssl", new SslHandler(sslEngine))
|
||||
}
|
||||
}
|
||||
pipeline.addLast("http-codec", new HttpClientCodec())
|
||||
pipeline.addLast("aggregator", new HttpObjectAggregator(65536))
|
||||
pipeline.addLast("websocket", handler.getHandler())
|
||||
}
|
||||
}))
|
||||
this.b.connect(uri.getHost(), uri.getPort()).addListener(new ChannelFutureListener((future: any) => {
|
||||
this.b.connect(this._host, this._port).addListener(new ChannelFutureListener((future: any) => {
|
||||
this.channel = future.sync().channel()
|
||||
this.onconnection({})
|
||||
handler.handshakeFuture.addListener(new ChannelFutureListener((future: any) => {
|
||||
try {
|
||||
future.sync()
|
||||
// only trigger onconnect when not have error
|
||||
this.onconnect({})
|
||||
} catch (error) {
|
||||
console.debug(error)
|
||||
// ignore error exceptionCaught from handler
|
||||
// this.onerror({ error })
|
||||
}
|
||||
}))
|
||||
}))
|
||||
|
||||
@@ -32,9 +32,11 @@ export abstract class Transport extends EventEmitter {
|
||||
}
|
||||
|
||||
connect() {
|
||||
console.debug(`client Transport connect`)
|
||||
try {
|
||||
this.doConnect()
|
||||
} catch (error) {
|
||||
console.ex(error)
|
||||
this.onerror({ error })
|
||||
}
|
||||
}
|
||||
@@ -55,6 +57,8 @@ export abstract class Transport extends EventEmitter {
|
||||
this.doClose(code, reason)
|
||||
} catch (error) {
|
||||
this.onerror({ error })
|
||||
} finally {
|
||||
this.removeAllListeners()
|
||||
}
|
||||
} else {
|
||||
console.debug(`${this.id} call close but state is ${this.readyStatus}`)
|
||||
@@ -67,7 +71,6 @@ export abstract class Transport extends EventEmitter {
|
||||
}
|
||||
|
||||
onconnect(event: Event) {
|
||||
console.debug(`${this.id} call onconnect`)
|
||||
if (this.readyStatus != WebSocket.OPEN) {
|
||||
this.readyStatus = WebSocket.OPEN
|
||||
this.emit('open', event)
|
||||
|
||||
1
packages/websocket/src/debug.ts
Normal file
1
packages/websocket/src/debug.ts
Normal file
@@ -0,0 +1 @@
|
||||
export = (namepsace) => (...args) => { }//console.debug(namepsace, ...args)
|
||||
16
packages/websocket/src/engine.io-client/index.ts
Normal file
16
packages/websocket/src/engine.io-client/index.ts
Normal file
@@ -0,0 +1,16 @@
|
||||
import { Socket } from './socket'
|
||||
|
||||
export default (uri, opts) => new Socket(uri, opts)
|
||||
|
||||
/**
|
||||
* Expose deps for legacy compatibility
|
||||
* and standalone browser access.
|
||||
*/
|
||||
const protocol = Socket.protocol // this is an int
|
||||
export { Socket, protocol }
|
||||
// module.exports.Transport = require("./transport")
|
||||
// module.exports.transports = require("./transports/index")
|
||||
// module.exports.parser = require("../engine.io-parser")
|
||||
export * from './transport'
|
||||
export * from './transports/index'
|
||||
export * from '../engine.io-parser'
|
||||
688
packages/websocket/src/engine.io-client/socket.ts
Normal file
688
packages/websocket/src/engine.io-client/socket.ts
Normal file
@@ -0,0 +1,688 @@
|
||||
import transports from "./transports"
|
||||
// const transports = require("./transports/index")
|
||||
const Emitter = require("component-emitter")
|
||||
const debug = (...args: any) => console.debug('engine.io-client:socket', ...args)//require("debug")("engine.io-client:socket")
|
||||
import parser from "../engine.io-parser"
|
||||
const parseuri = require("parseuri")
|
||||
const parseqs = require("parseqs")
|
||||
import { installTimerFunctions } from "./util"
|
||||
|
||||
export class Socket extends Emitter {
|
||||
/**
|
||||
* Socket constructor.
|
||||
*
|
||||
* @param {String|Object} uri or options
|
||||
* @param {Object} options
|
||||
* @api public
|
||||
*/
|
||||
constructor(uri, opts: any = {}) {
|
||||
super()
|
||||
|
||||
if (uri && "object" === typeof uri) {
|
||||
opts = uri
|
||||
uri = null
|
||||
}
|
||||
|
||||
if (uri) {
|
||||
uri = parseuri(uri)
|
||||
opts.hostname = uri.host
|
||||
opts.secure = uri.protocol === "https" || uri.protocol === "wss"
|
||||
opts.port = uri.port
|
||||
if (uri.query) opts.query = uri.query
|
||||
} else if (opts.host) {
|
||||
opts.hostname = parseuri(opts.host).host
|
||||
}
|
||||
|
||||
installTimerFunctions(this, opts)
|
||||
|
||||
this.secure =
|
||||
null != opts.secure
|
||||
? opts.secure
|
||||
: typeof location !== "undefined" && "https:" === location.protocol
|
||||
|
||||
if (opts.hostname && !opts.port) {
|
||||
// if no port is specified manually, use the protocol default
|
||||
opts.port = this.secure ? "443" : "80"
|
||||
}
|
||||
|
||||
this.hostname =
|
||||
opts.hostname ||
|
||||
(typeof location !== "undefined" ? location.hostname : "localhost")
|
||||
this.port =
|
||||
opts.port ||
|
||||
(typeof location !== "undefined" && location.port
|
||||
? location.port
|
||||
: this.secure
|
||||
? 443
|
||||
: 80)
|
||||
|
||||
this.transports = ["websocket"]
|
||||
this.readyState = ""
|
||||
this.writeBuffer = []
|
||||
this.prevBufferLen = 0
|
||||
|
||||
this.opts = Object.assign(
|
||||
{
|
||||
path: "/engine.io",
|
||||
agent: false,
|
||||
withCredentials: false,
|
||||
upgrade: true,
|
||||
jsonp: true,
|
||||
timestampParam: "t",
|
||||
rememberUpgrade: false,
|
||||
rejectUnauthorized: true,
|
||||
perMessageDeflate: {
|
||||
threshold: 1024
|
||||
},
|
||||
transportOptions: {},
|
||||
closeOnBeforeunload: true
|
||||
},
|
||||
opts
|
||||
)
|
||||
|
||||
this.opts.path = this.opts.path.replace(/\/$/, "") + "/"
|
||||
|
||||
if (typeof this.opts.query === "string") {
|
||||
this.opts.query = parseqs.decode(this.opts.query)
|
||||
}
|
||||
|
||||
// set on handshake
|
||||
this.id = null
|
||||
this.upgrades = null
|
||||
this.pingInterval = null
|
||||
this.pingTimeout = null
|
||||
|
||||
// set on heartbeat
|
||||
this.pingTimeoutTimer = null
|
||||
|
||||
if (typeof addEventListener === "function") {
|
||||
if (this.opts.closeOnBeforeunload) {
|
||||
// Firefox closes the connection when the "beforeunload" event is emitted but not Chrome. This event listener
|
||||
// ensures every browser behaves the same (no "disconnect" event at the Socket.IO level when the page is
|
||||
// closed/reloaded)
|
||||
addEventListener(
|
||||
"beforeunload",
|
||||
() => {
|
||||
if (this.transport) {
|
||||
// silently close the transport
|
||||
this.transport.removeAllListeners()
|
||||
this.transport.close()
|
||||
}
|
||||
},
|
||||
false
|
||||
)
|
||||
}
|
||||
if (this.hostname !== "localhost") {
|
||||
this.offlineEventListener = () => {
|
||||
this.onClose("transport close")
|
||||
}
|
||||
addEventListener("offline", this.offlineEventListener, false)
|
||||
}
|
||||
}
|
||||
|
||||
this.open()
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates transport of the given type.
|
||||
*
|
||||
* @param {String} transport name
|
||||
* @return {Transport}
|
||||
* @api private
|
||||
*/
|
||||
createTransport(name, opt?) {
|
||||
if (name != 'websocket') {
|
||||
throw new Error('Only Support WebSocket in MiaoScript!')
|
||||
}
|
||||
debug('creating transport "%s"', name)
|
||||
const query: any = clone(this.opts.query)
|
||||
|
||||
// append engine.io protocol identifier
|
||||
query.EIO = parser.protocol
|
||||
|
||||
// transport name
|
||||
query.transport = name
|
||||
|
||||
// session id if we already have one
|
||||
if (this.id) query.sid = this.id
|
||||
|
||||
const opts = Object.assign(
|
||||
{},
|
||||
this.opts.transportOptions[name],
|
||||
this.opts,
|
||||
{
|
||||
query,
|
||||
socket: this,
|
||||
hostname: this.hostname,
|
||||
secure: this.secure,
|
||||
port: this.port
|
||||
}
|
||||
)
|
||||
|
||||
debug("options: %j", JSON.stringify(opts))
|
||||
debug("new func", transports[name])
|
||||
return new transports[name](opts)
|
||||
}
|
||||
|
||||
/**
|
||||
* Initializes transport to use and starts probe.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
open() {
|
||||
let transport
|
||||
if (
|
||||
this.opts.rememberUpgrade &&
|
||||
Socket.priorWebsocketSuccess &&
|
||||
this.transports.indexOf("websocket") !== -1
|
||||
) {
|
||||
transport = "websocket"
|
||||
} else if (0 === this.transports.length) {
|
||||
// Emit error on next tick so it can be listened to
|
||||
this.setTimeoutFn(() => {
|
||||
this.emit("error", "No transports available")
|
||||
}, 0)
|
||||
return
|
||||
} else {
|
||||
transport = this.transports[0]
|
||||
}
|
||||
this.readyState = "opening"
|
||||
|
||||
// Retry with the next transport if the transport is disabled (jsonp: false)
|
||||
try {
|
||||
transport = this.createTransport(transport)
|
||||
} catch (e) {
|
||||
debug("error while creating transport: %s", e)
|
||||
this.transports.shift()
|
||||
this.open()
|
||||
return
|
||||
}
|
||||
|
||||
transport.open()
|
||||
this.setTransport(transport)
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the current transport. Disables the existing one (if any).
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
setTransport(transport) {
|
||||
debug("setting transport %s", transport.name)
|
||||
|
||||
if (this.transport) {
|
||||
debug("clearing existing transport %s", this.transport.name)
|
||||
this.transport.removeAllListeners()
|
||||
}
|
||||
|
||||
// set up transport
|
||||
this.transport = transport
|
||||
|
||||
// set up transport listeners
|
||||
transport
|
||||
.on("drain", this.onDrain.bind(this))
|
||||
.on("packet", this.onPacket.bind(this))
|
||||
.on("error", this.onError.bind(this))
|
||||
.on("close", () => {
|
||||
this.onClose("transport close")
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* Probes a transport.
|
||||
*
|
||||
* @param {String} transport name
|
||||
* @api private
|
||||
*/
|
||||
probe(name) {
|
||||
debug('probing transport "%s"', name)
|
||||
let transport = this.createTransport(name, { probe: 1 })
|
||||
let failed = false
|
||||
|
||||
Socket.priorWebsocketSuccess = false
|
||||
|
||||
const onTransportOpen = () => {
|
||||
if (failed) return
|
||||
|
||||
debug('probe transport "%s" opened', name)
|
||||
transport.send([{ type: "ping", data: "probe" }])
|
||||
transport.once("packet", msg => {
|
||||
if (failed) return
|
||||
if ("pong" === msg.type && "probe" === msg.data) {
|
||||
debug('probe transport "%s" pong', name)
|
||||
this.upgrading = true
|
||||
this.emit("upgrading", transport)
|
||||
if (!transport) return
|
||||
Socket.priorWebsocketSuccess = "websocket" === transport.name
|
||||
|
||||
debug('pausing current transport "%s"', this.transport.name)
|
||||
this.transport.pause(() => {
|
||||
if (failed) return
|
||||
if ("closed" === this.readyState) return
|
||||
debug("changing transport and sending upgrade packet")
|
||||
|
||||
cleanup()
|
||||
|
||||
this.setTransport(transport)
|
||||
transport.send([{ type: "upgrade" }])
|
||||
this.emit("upgrade", transport)
|
||||
transport = null
|
||||
this.upgrading = false
|
||||
this.flush()
|
||||
})
|
||||
} else {
|
||||
debug('probe transport "%s" failed', name)
|
||||
const err: any = new Error("probe error")
|
||||
err.transport = transport.name
|
||||
this.emit("upgradeError", err)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
function freezeTransport() {
|
||||
if (failed) return
|
||||
|
||||
// Any callback called by transport should be ignored since now
|
||||
failed = true
|
||||
|
||||
cleanup()
|
||||
|
||||
transport.close()
|
||||
transport = null
|
||||
}
|
||||
|
||||
// Handle any error that happens while probing
|
||||
const onerror = err => {
|
||||
const error: any = new Error("probe error: " + err)
|
||||
error.transport = transport.name
|
||||
|
||||
freezeTransport()
|
||||
|
||||
debug('probe transport "%s" failed because of error: %s', name, err)
|
||||
|
||||
this.emit("upgradeError", error)
|
||||
}
|
||||
|
||||
function onTransportClose() {
|
||||
onerror("transport closed")
|
||||
}
|
||||
|
||||
// When the socket is closed while we're probing
|
||||
function onclose() {
|
||||
onerror("socket closed")
|
||||
}
|
||||
|
||||
// When the socket is upgraded while we're probing
|
||||
function onupgrade(to) {
|
||||
if (transport && to.name !== transport.name) {
|
||||
debug('"%s" works - aborting "%s"', to.name, transport.name)
|
||||
freezeTransport()
|
||||
}
|
||||
}
|
||||
|
||||
// Remove all listeners on the transport and on self
|
||||
const cleanup = () => {
|
||||
transport.removeListener("open", onTransportOpen)
|
||||
transport.removeListener("error", onerror)
|
||||
transport.removeListener("close", onTransportClose)
|
||||
this.removeListener("close", onclose)
|
||||
this.removeListener("upgrading", onupgrade)
|
||||
}
|
||||
|
||||
transport.once("open", onTransportOpen)
|
||||
transport.once("error", onerror)
|
||||
transport.once("close", onTransportClose)
|
||||
|
||||
this.once("close", onclose)
|
||||
this.once("upgrading", onupgrade)
|
||||
|
||||
transport.open()
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when connection is deemed open.
|
||||
*
|
||||
* @api public
|
||||
*/
|
||||
onOpen() {
|
||||
debug("socket open")
|
||||
this.readyState = "open"
|
||||
Socket.priorWebsocketSuccess = "websocket" === this.transport.name
|
||||
this.emit("open")
|
||||
this.flush()
|
||||
|
||||
// we check for `readyState` in case an `open`
|
||||
// listener already closed the socket
|
||||
if (
|
||||
"open" === this.readyState &&
|
||||
this.opts.upgrade &&
|
||||
this.transport.pause
|
||||
) {
|
||||
debug("starting upgrade probes")
|
||||
let i = 0
|
||||
const l = this.upgrades.length
|
||||
for (; i < l; i++) {
|
||||
this.probe(this.upgrades[i])
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Handles a packet.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onPacket(packet) {
|
||||
if (
|
||||
"opening" === this.readyState ||
|
||||
"open" === this.readyState ||
|
||||
"closing" === this.readyState
|
||||
) {
|
||||
debug('socket receive: type "%s", data "%s"', packet.type, packet.data)
|
||||
|
||||
this.emit("packet", packet)
|
||||
|
||||
// Socket is live - any packet counts
|
||||
this.emit("heartbeat")
|
||||
|
||||
switch (packet.type) {
|
||||
case "open":
|
||||
this.onHandshake(JSON.parse(packet.data))
|
||||
break
|
||||
|
||||
case "ping":
|
||||
this.resetPingTimeout()
|
||||
this.sendPacket("pong")
|
||||
this.emit("ping")
|
||||
this.emit("pong")
|
||||
break
|
||||
|
||||
case "error":
|
||||
const err: any = new Error("server error")
|
||||
err.code = packet.data
|
||||
this.onError(err)
|
||||
break
|
||||
|
||||
case "message":
|
||||
this.emit("data", packet.data)
|
||||
this.emit("message", packet.data)
|
||||
break
|
||||
}
|
||||
} else {
|
||||
debug('packet received with socket readyState "%s"', this.readyState)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon handshake completion.
|
||||
*
|
||||
* @param {Object} handshake obj
|
||||
* @api private
|
||||
*/
|
||||
onHandshake(data) {
|
||||
this.emit("handshake", data)
|
||||
this.id = data.sid
|
||||
this.transport.query.sid = data.sid
|
||||
this.upgrades = this.filterUpgrades(data.upgrades)
|
||||
this.pingInterval = data.pingInterval
|
||||
this.pingTimeout = data.pingTimeout
|
||||
this.onOpen()
|
||||
// In case open handler closes socket
|
||||
if ("closed" === this.readyState) return
|
||||
this.resetPingTimeout()
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets and resets ping timeout timer based on server pings.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
resetPingTimeout() {
|
||||
this.clearTimeoutFn(this.pingTimeoutTimer)
|
||||
this.pingTimeoutTimer = this.setTimeoutFn(() => {
|
||||
this.onClose("ping timeout")
|
||||
}, this.pingInterval + this.pingTimeout)
|
||||
if (this.opts.autoUnref) {
|
||||
this.pingTimeoutTimer.unref()
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called on `drain` event
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onDrain() {
|
||||
this.writeBuffer.splice(0, this.prevBufferLen)
|
||||
|
||||
// setting prevBufferLen = 0 is very important
|
||||
// for example, when upgrading, upgrade packet is sent over,
|
||||
// and a nonzero prevBufferLen could cause problems on `drain`
|
||||
this.prevBufferLen = 0
|
||||
|
||||
if (0 === this.writeBuffer.length) {
|
||||
this.emit("drain")
|
||||
} else {
|
||||
this.flush()
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Flush write buffers.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
flush() {
|
||||
if (
|
||||
"closed" !== this.readyState &&
|
||||
this.transport.writable &&
|
||||
!this.upgrading &&
|
||||
this.writeBuffer.length
|
||||
) {
|
||||
debug("flushing %d packets in socket", this.writeBuffer.length)
|
||||
this.transport.send(this.writeBuffer)
|
||||
// keep track of current length of writeBuffer
|
||||
// splice writeBuffer and callbackBuffer on `drain`
|
||||
this.prevBufferLen = this.writeBuffer.length
|
||||
this.emit("flush")
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Sends a message.
|
||||
*
|
||||
* @param {String} message.
|
||||
* @param {Function} callback function.
|
||||
* @param {Object} options.
|
||||
* @return {Socket} for chaining.
|
||||
* @api public
|
||||
*/
|
||||
write(msg, options, fn) {
|
||||
this.sendPacket("message", msg, options, fn)
|
||||
return this
|
||||
}
|
||||
|
||||
send(msg, options, fn) {
|
||||
this.sendPacket("message", msg, options, fn)
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Sends a packet.
|
||||
*
|
||||
* @param {String} packet type.
|
||||
* @param {String} data.
|
||||
* @param {Object} options.
|
||||
* @param {Function} callback function.
|
||||
* @api private
|
||||
*/
|
||||
sendPacket(type, data?, options?, fn?) {
|
||||
if ("function" === typeof data) {
|
||||
fn = data
|
||||
data = undefined
|
||||
}
|
||||
|
||||
if ("function" === typeof options) {
|
||||
fn = options
|
||||
options = null
|
||||
}
|
||||
|
||||
if ("closing" === this.readyState || "closed" === this.readyState) {
|
||||
return
|
||||
}
|
||||
|
||||
options = options || {}
|
||||
options.compress = false !== options.compress
|
||||
|
||||
const packet = {
|
||||
type: type,
|
||||
data: data,
|
||||
options: options
|
||||
}
|
||||
this.emit("packetCreate", packet)
|
||||
this.writeBuffer.push(packet)
|
||||
if (fn) this.once("flush", fn)
|
||||
this.flush()
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes the connection.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
close() {
|
||||
const close = () => {
|
||||
this.onClose("forced close")
|
||||
debug("socket closing - telling transport to close")
|
||||
this.transport.close()
|
||||
}
|
||||
|
||||
const cleanupAndClose = () => {
|
||||
this.removeListener("upgrade", cleanupAndClose)
|
||||
this.removeListener("upgradeError", cleanupAndClose)
|
||||
close()
|
||||
}
|
||||
|
||||
const waitForUpgrade = () => {
|
||||
// wait for upgrade to finish since we can't send packets while pausing a transport
|
||||
this.once("upgrade", cleanupAndClose)
|
||||
this.once("upgradeError", cleanupAndClose)
|
||||
}
|
||||
|
||||
if ("opening" === this.readyState || "open" === this.readyState) {
|
||||
this.readyState = "closing"
|
||||
|
||||
if (this.writeBuffer.length) {
|
||||
this.once("drain", () => {
|
||||
if (this.upgrading) {
|
||||
waitForUpgrade()
|
||||
} else {
|
||||
close()
|
||||
}
|
||||
})
|
||||
} else if (this.upgrading) {
|
||||
waitForUpgrade()
|
||||
} else {
|
||||
close()
|
||||
}
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport error
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onError(err) {
|
||||
debug("socket error %j", err)
|
||||
Socket.priorWebsocketSuccess = false
|
||||
this.emit("error", err)
|
||||
this.onClose("transport error", err)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport close.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onClose(reason, desc?) {
|
||||
if (
|
||||
"opening" === this.readyState ||
|
||||
"open" === this.readyState ||
|
||||
"closing" === this.readyState
|
||||
) {
|
||||
debug('socket close with reason: "%s"', reason)
|
||||
|
||||
// clear timers
|
||||
this.clearTimeoutFn(this.pingIntervalTimer)
|
||||
this.clearTimeoutFn(this.pingTimeoutTimer)
|
||||
|
||||
// stop event from firing again for transport
|
||||
this.transport.removeAllListeners("close")
|
||||
|
||||
// ensure transport won't stay open
|
||||
this.transport.close()
|
||||
|
||||
// ignore further transport communication
|
||||
this.transport.removeAllListeners()
|
||||
|
||||
if (typeof removeEventListener === "function") {
|
||||
removeEventListener("offline", this.offlineEventListener, false)
|
||||
}
|
||||
|
||||
// set ready state
|
||||
this.readyState = "closed"
|
||||
|
||||
// clear session id
|
||||
this.id = null
|
||||
|
||||
// emit close event
|
||||
this.emit("close", reason, desc)
|
||||
|
||||
// clean buffers after, so users can still
|
||||
// grab the buffers on `close` event
|
||||
this.writeBuffer = []
|
||||
this.prevBufferLen = 0
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Filters upgrades, returning only those matching client transports.
|
||||
*
|
||||
* @param {Array} server upgrades
|
||||
* @api private
|
||||
*
|
||||
*/
|
||||
filterUpgrades(upgrades) {
|
||||
const filteredUpgrades = []
|
||||
let i = 0
|
||||
const j = upgrades.length
|
||||
for (; i < j; i++) {
|
||||
if (~this.transports.indexOf(upgrades[i]))
|
||||
filteredUpgrades.push(upgrades[i])
|
||||
}
|
||||
return filteredUpgrades
|
||||
}
|
||||
}
|
||||
|
||||
Socket.priorWebsocketSuccess = false
|
||||
|
||||
/**
|
||||
* Protocol version.
|
||||
*
|
||||
* @api public
|
||||
*/
|
||||
|
||||
Socket.protocol = parser.protocol // this is an int
|
||||
|
||||
function clone(obj) {
|
||||
const o = {}
|
||||
for (let i in obj) {
|
||||
if (obj.hasOwnProperty(i)) {
|
||||
o[i] = obj[i]
|
||||
}
|
||||
}
|
||||
return o
|
||||
}
|
||||
119
packages/websocket/src/engine.io-client/transport.ts
Normal file
119
packages/websocket/src/engine.io-client/transport.ts
Normal file
@@ -0,0 +1,119 @@
|
||||
import parser from "../engine.io-parser"
|
||||
const Emitter = require("component-emitter")
|
||||
import { installTimerFunctions } from "./util"
|
||||
const debug = (...args: any) => console.debug('engine.io-client:transport', ...args)//require("debug")("engine.io-client:transport")
|
||||
|
||||
export class Transport extends Emitter {
|
||||
/**
|
||||
* Transport abstract constructor.
|
||||
*
|
||||
* @param {Object} options.
|
||||
* @api private
|
||||
*/
|
||||
constructor(opts) {
|
||||
super()
|
||||
installTimerFunctions(this, opts)
|
||||
|
||||
this.opts = opts
|
||||
this.query = opts.query
|
||||
this.readyState = ""
|
||||
this.socket = opts.socket
|
||||
}
|
||||
|
||||
/**
|
||||
* Emits an error.
|
||||
*
|
||||
* @param {String} str
|
||||
* @return {Transport} for chaining
|
||||
* @api public
|
||||
*/
|
||||
onError(msg, desc) {
|
||||
const err: any = new Error(msg)
|
||||
err.type = "TransportError"
|
||||
err.description = desc
|
||||
this.emit("error", err)
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Opens the transport.
|
||||
*
|
||||
* @api public
|
||||
*/
|
||||
open() {
|
||||
if ("closed" === this.readyState || "" === this.readyState) {
|
||||
this.readyState = "opening"
|
||||
this.doOpen()
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes the transport.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
close() {
|
||||
if ("opening" === this.readyState || "open" === this.readyState) {
|
||||
this.doClose()
|
||||
this.onClose()
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Sends multiple packets.
|
||||
*
|
||||
* @param {Array} packets
|
||||
* @api private
|
||||
*/
|
||||
send(packets) {
|
||||
if ("open" === this.readyState) {
|
||||
this.write(packets)
|
||||
} else {
|
||||
// this might happen if the transport was silently closed in the beforeunload event handler
|
||||
debug("transport is not open, discarding packets")
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon open
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onOpen() {
|
||||
this.readyState = "open"
|
||||
this.writable = true
|
||||
this.emit("open")
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with data.
|
||||
*
|
||||
* @param {String} data
|
||||
* @api private
|
||||
*/
|
||||
onData(data) {
|
||||
const packet = parser.decodePacket(data, this.socket.binaryType)
|
||||
this.onPacket(packet)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with a decoded packet.
|
||||
*/
|
||||
onPacket(packet) {
|
||||
this.emit("packet", packet)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon close.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onClose() {
|
||||
this.readyState = "closed"
|
||||
this.emit("close")
|
||||
}
|
||||
}
|
||||
4
packages/websocket/src/engine.io-client/transports/index.ts
Executable file
4
packages/websocket/src/engine.io-client/transports/index.ts
Executable file
@@ -0,0 +1,4 @@
|
||||
import { WS } from "./websocket"
|
||||
export default {
|
||||
'websocket': WS
|
||||
}
|
||||
259
packages/websocket/src/engine.io-client/transports/websocket.ts
Normal file
259
packages/websocket/src/engine.io-client/transports/websocket.ts
Normal file
@@ -0,0 +1,259 @@
|
||||
import { Transport } from '../transport'
|
||||
// const Transport = require("../transport")
|
||||
import parser from '../../engine.io-parser'
|
||||
// const parser = require("../engine.io-parser")
|
||||
const parseqs = require("parseqs")
|
||||
const yeast = require("yeast")
|
||||
import { pick } from '../util'
|
||||
// const { pick } = require("../util")
|
||||
import { WebSocket } from '../../client'
|
||||
const usingBrowserWebSocket = true
|
||||
// const {
|
||||
// WebSocket,
|
||||
// usingBrowserWebSocket,
|
||||
// defaultBinaryType,
|
||||
// nextTick
|
||||
// } = require("./websocket-constructor")
|
||||
|
||||
const debug = (...args: any) => console.debug('engine.io-client:websocket', ...args)//require("debug")("engine.io-client:websocket")
|
||||
|
||||
// detect ReactNative environment
|
||||
const isReactNative =
|
||||
typeof navigator !== "undefined" &&
|
||||
typeof navigator.product === "string" &&
|
||||
navigator.product.toLowerCase() === "reactnative"
|
||||
|
||||
export class WS extends Transport {
|
||||
/**
|
||||
* WebSocket transport constructor.
|
||||
*
|
||||
* @api {Object} connection options
|
||||
* @api public
|
||||
*/
|
||||
constructor(opts) {
|
||||
super(opts)
|
||||
|
||||
this.supportsBinary = !opts.forceBase64
|
||||
}
|
||||
|
||||
/**
|
||||
* Transport name.
|
||||
*
|
||||
* @api public
|
||||
*/
|
||||
get name() {
|
||||
return "websocket"
|
||||
}
|
||||
|
||||
/**
|
||||
* Opens socket.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
doOpen() {
|
||||
if (!this.check()) {
|
||||
// let probe timeout
|
||||
return
|
||||
}
|
||||
|
||||
const uri = this.uri()
|
||||
const protocols = this.opts.protocols
|
||||
|
||||
// React Native only supports the 'headers' option, and will print a warning if anything else is passed
|
||||
const opts = isReactNative
|
||||
? {}
|
||||
: pick(
|
||||
this.opts,
|
||||
"agent",
|
||||
"perMessageDeflate",
|
||||
"pfx",
|
||||
"key",
|
||||
"passphrase",
|
||||
"cert",
|
||||
"ca",
|
||||
"ciphers",
|
||||
"rejectUnauthorized",
|
||||
"localAddress",
|
||||
"protocolVersion",
|
||||
"origin",
|
||||
"maxPayload",
|
||||
"family",
|
||||
"checkServerIdentity"
|
||||
)
|
||||
|
||||
if (this.opts.extraHeaders) {
|
||||
opts.headers = this.opts.extraHeaders
|
||||
}
|
||||
|
||||
try {
|
||||
this.ws = new WebSocket(uri, protocols)
|
||||
// usingBrowserWebSocket && !isReactNative
|
||||
// ? protocols
|
||||
// ? new WebSocket(uri, protocols)
|
||||
// : new WebSocket(uri)
|
||||
// : new WebSocket(uri, protocols, opts)
|
||||
} catch (err) {
|
||||
return this.emit("error", err)
|
||||
}
|
||||
|
||||
this.ws.binaryType = this.socket.binaryType || 'arraybuffer'
|
||||
|
||||
this.addEventListeners()
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds event listeners to the socket
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
addEventListeners() {
|
||||
this.ws.onopen = () => {
|
||||
if (this.opts.autoUnref) {
|
||||
this.ws._socket.unref()
|
||||
}
|
||||
this.onOpen()
|
||||
}
|
||||
this.ws.onclose = this.onClose.bind(this)
|
||||
this.ws.onmessage = ev => this.onData(ev.data)
|
||||
this.ws.onerror = e => this.onError("websocket error", e)
|
||||
}
|
||||
|
||||
/**
|
||||
* Writes data to socket.
|
||||
*
|
||||
* @param {Array} array of packets.
|
||||
* @api private
|
||||
*/
|
||||
write(packets) {
|
||||
this.writable = false
|
||||
|
||||
// encodePacket efficient as it uses WS framing
|
||||
// no need for encodePayload
|
||||
for (let i = 0; i < packets.length; i++) {
|
||||
const packet = packets[i]
|
||||
const lastPacket = i === packets.length - 1
|
||||
|
||||
parser.encodePacket(packet, this.supportsBinary, data => {
|
||||
// always create a new object (GH-437)
|
||||
const opts: any = {}
|
||||
if (!usingBrowserWebSocket) {
|
||||
if (packet.options) {
|
||||
opts.compress = packet.options.compress
|
||||
}
|
||||
|
||||
if (this.opts.perMessageDeflate) {
|
||||
const len =
|
||||
"string" === typeof data ? Buffer.byteLength(data) : data.length
|
||||
if (len < this.opts.perMessageDeflate.threshold) {
|
||||
opts.compress = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Sometimes the websocket has already been closed but the browser didn't
|
||||
// have a chance of informing us about it yet, in that case send will
|
||||
// throw an error
|
||||
try {
|
||||
if (usingBrowserWebSocket) {
|
||||
// TypeError is thrown when passing the second argument on Safari
|
||||
this.ws.send(data)
|
||||
} else {
|
||||
this.ws.send(data, opts)
|
||||
}
|
||||
} catch (e) {
|
||||
debug("websocket closed before onclose event")
|
||||
}
|
||||
|
||||
if (lastPacket) {
|
||||
// fake drain
|
||||
// defer to next tick to allow Socket to clear writeBuffer
|
||||
process.nextTick(() => {
|
||||
this.writable = true
|
||||
this.emit("drain")
|
||||
}, this.setTimeoutFn)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon close
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onClose() {
|
||||
Transport.prototype.onClose.call(this)
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes socket.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
doClose() {
|
||||
if (typeof this.ws !== "undefined") {
|
||||
this.ws.close()
|
||||
this.ws = null
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Generates uri for connection.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
uri() {
|
||||
let query = this.query || {}
|
||||
const schema = this.opts.secure ? "wss" : "ws"
|
||||
let port = ""
|
||||
|
||||
// avoid port if default for schema
|
||||
if (
|
||||
this.opts.port &&
|
||||
(("wss" === schema && Number(this.opts.port) !== 443) ||
|
||||
("ws" === schema && Number(this.opts.port) !== 80))
|
||||
) {
|
||||
port = ":" + this.opts.port
|
||||
}
|
||||
|
||||
// append timestamp to URI
|
||||
if (this.opts.timestampRequests) {
|
||||
query[this.opts.timestampParam] = yeast()
|
||||
}
|
||||
|
||||
// communicate binary support capabilities
|
||||
if (!this.supportsBinary) {
|
||||
query.b64 = 1
|
||||
}
|
||||
|
||||
query = parseqs.encode(query)
|
||||
|
||||
// prepend ? to query
|
||||
if (query.length) {
|
||||
query = "?" + query
|
||||
}
|
||||
|
||||
const ipv6 = this.opts.hostname.indexOf(":") !== -1
|
||||
return (
|
||||
schema +
|
||||
"://" +
|
||||
(ipv6 ? "[" + this.opts.hostname + "]" : this.opts.hostname) +
|
||||
port +
|
||||
this.opts.path +
|
||||
query
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Feature detection for WebSocket.
|
||||
*
|
||||
* @return {Boolean} whether this transport is available.
|
||||
* @api public
|
||||
*/
|
||||
check() {
|
||||
return (
|
||||
!!WebSocket &&
|
||||
!("__initialize" in WebSocket && this.name === WS.prototype.name)
|
||||
)
|
||||
}
|
||||
}
|
||||
23
packages/websocket/src/engine.io-client/util.ts
Normal file
23
packages/websocket/src/engine.io-client/util.ts
Normal file
@@ -0,0 +1,23 @@
|
||||
const pick = (obj, ...attr) => {
|
||||
return attr.reduce((acc, k) => {
|
||||
if (obj.hasOwnProperty(k)) {
|
||||
acc[k] = obj[k]
|
||||
}
|
||||
return acc
|
||||
}, {})
|
||||
}
|
||||
|
||||
// Keep a reference to the real timeout functions so they can be used when overridden
|
||||
const NATIVE_SET_TIMEOUT = setTimeout
|
||||
const NATIVE_CLEAR_TIMEOUT = clearTimeout
|
||||
|
||||
const installTimerFunctions = (obj, opts) => {
|
||||
if (opts.useNativeTimers) {
|
||||
obj.setTimeoutFn = NATIVE_SET_TIMEOUT.bind(globalThis)
|
||||
obj.clearTimeoutFn = NATIVE_CLEAR_TIMEOUT.bind(globalThis)
|
||||
} else {
|
||||
obj.setTimeoutFn = setTimeout.bind(globalThis)
|
||||
obj.clearTimeoutFn = clearTimeout.bind(globalThis)
|
||||
}
|
||||
}
|
||||
export { pick, installTimerFunctions }
|
||||
21
packages/websocket/src/engine.io-parser/commons.ts
Normal file
21
packages/websocket/src/engine.io-parser/commons.ts
Normal file
@@ -0,0 +1,21 @@
|
||||
const PACKET_TYPES = Object.create(null) // no Map = no polyfill
|
||||
PACKET_TYPES["open"] = "0"
|
||||
PACKET_TYPES["close"] = "1"
|
||||
PACKET_TYPES["ping"] = "2"
|
||||
PACKET_TYPES["pong"] = "3"
|
||||
PACKET_TYPES["message"] = "4"
|
||||
PACKET_TYPES["upgrade"] = "5"
|
||||
PACKET_TYPES["noop"] = "6"
|
||||
|
||||
const PACKET_TYPES_REVERSE = Object.create(null)
|
||||
Object.keys(PACKET_TYPES).forEach(key => {
|
||||
PACKET_TYPES_REVERSE[PACKET_TYPES[key]] = key
|
||||
})
|
||||
|
||||
const ERROR_PACKET = { type: "error", data: "parser error" }
|
||||
|
||||
export = {
|
||||
PACKET_TYPES,
|
||||
PACKET_TYPES_REVERSE,
|
||||
ERROR_PACKET
|
||||
}
|
||||
48
packages/websocket/src/engine.io-parser/decodePacket.ts
Normal file
48
packages/websocket/src/engine.io-parser/decodePacket.ts
Normal file
@@ -0,0 +1,48 @@
|
||||
const { PACKET_TYPES_REVERSE, ERROR_PACKET } = require("./commons")
|
||||
|
||||
export const decodePacket = (encodedPacket, binaryType) => {
|
||||
if (typeof encodedPacket !== "string") {
|
||||
return {
|
||||
type: "message",
|
||||
data: mapBinary(encodedPacket, binaryType)
|
||||
}
|
||||
}
|
||||
const type = encodedPacket.charAt(0)
|
||||
if (type === "b") {
|
||||
const buffer = Buffer.from(encodedPacket.substring(1), "base64")
|
||||
return {
|
||||
type: "message",
|
||||
data: mapBinary(buffer, binaryType)
|
||||
}
|
||||
}
|
||||
if (!PACKET_TYPES_REVERSE[type]) {
|
||||
return ERROR_PACKET
|
||||
}
|
||||
return encodedPacket.length > 1
|
||||
? {
|
||||
type: PACKET_TYPES_REVERSE[type],
|
||||
data: encodedPacket.substring(1)
|
||||
}
|
||||
: {
|
||||
type: PACKET_TYPES_REVERSE[type]
|
||||
}
|
||||
}
|
||||
|
||||
const mapBinary = (data, binaryType) => {
|
||||
switch (binaryType) {
|
||||
case "arraybuffer":
|
||||
return Buffer.isBuffer(data) ? toArrayBuffer(data) : data
|
||||
case "nodebuffer":
|
||||
default:
|
||||
return data // assuming the data is already a Buffer
|
||||
}
|
||||
}
|
||||
|
||||
const toArrayBuffer = buffer => {
|
||||
const arrayBuffer = new ArrayBuffer(buffer.length)
|
||||
const view = new Uint8Array(arrayBuffer)
|
||||
for (let i = 0; i < buffer.length; i++) {
|
||||
view[i] = buffer[i]
|
||||
}
|
||||
return arrayBuffer
|
||||
}
|
||||
26
packages/websocket/src/engine.io-parser/encodePacket.ts
Normal file
26
packages/websocket/src/engine.io-parser/encodePacket.ts
Normal file
@@ -0,0 +1,26 @@
|
||||
const { PACKET_TYPES } = require("./commons")
|
||||
|
||||
export const encodePacket = ({ type, data }, supportsBinary, callback) => {
|
||||
console.trace('encodePacket', type, JSON.stringify(data))
|
||||
if (data instanceof ArrayBuffer || ArrayBuffer.isView(data)) {
|
||||
const buffer = toBuffer(data)
|
||||
return callback(encodeBuffer(buffer, supportsBinary))
|
||||
}
|
||||
// plain string
|
||||
return callback(PACKET_TYPES[type] + (data || ""))
|
||||
}
|
||||
|
||||
const toBuffer = data => {
|
||||
if (Buffer.isBuffer(data)) {
|
||||
return data
|
||||
} else if (data instanceof ArrayBuffer) {
|
||||
return Buffer.from(data)
|
||||
} else {
|
||||
return Buffer.from(data.buffer, data.byteOffset, data.byteLength)
|
||||
}
|
||||
}
|
||||
|
||||
// only 'message' packets can contain binary, so the type prefix is not needed
|
||||
const encodeBuffer = (data, supportsBinary) => {
|
||||
return supportsBinary ? data : "b" + data.toString("base64")
|
||||
}
|
||||
42
packages/websocket/src/engine.io-parser/index.ts
Normal file
42
packages/websocket/src/engine.io-parser/index.ts
Normal file
@@ -0,0 +1,42 @@
|
||||
import { encodePacket } from "./encodePacket"
|
||||
import { decodePacket } from "./decodePacket"
|
||||
|
||||
const SEPARATOR = String.fromCharCode(30) // see https://en.wikipedia.org/wiki/Delimiter#ASCII_delimited_text
|
||||
|
||||
const encodePayload = (packets, callback) => {
|
||||
// some packets may be added to the array while encoding, so the initial length must be saved
|
||||
const length = packets.length
|
||||
const encodedPackets = new Array(length)
|
||||
let count = 0
|
||||
|
||||
packets.forEach((packet, i) => {
|
||||
// force base64 encoding for binary packets
|
||||
encodePacket(packet, false, encodedPacket => {
|
||||
encodedPackets[i] = encodedPacket
|
||||
if (++count === length) {
|
||||
callback(encodedPackets.join(SEPARATOR))
|
||||
}
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
const decodePayload = (encodedPayload, binaryType) => {
|
||||
const encodedPackets = encodedPayload.split(SEPARATOR)
|
||||
const packets = []
|
||||
for (let i = 0; i < encodedPackets.length; i++) {
|
||||
const decodedPacket = decodePacket(encodedPackets[i], binaryType)
|
||||
packets.push(decodedPacket)
|
||||
if (decodedPacket.type === "error") {
|
||||
break
|
||||
}
|
||||
}
|
||||
return packets
|
||||
}
|
||||
|
||||
export default {
|
||||
protocol: 4,
|
||||
encodePacket,
|
||||
encodePayload,
|
||||
decodePacket,
|
||||
decodePayload
|
||||
}
|
||||
27
packages/websocket/src/engine.io/index.ts
Normal file
27
packages/websocket/src/engine.io/index.ts
Normal file
@@ -0,0 +1,27 @@
|
||||
/**
|
||||
* Module dependencies.
|
||||
*/
|
||||
|
||||
import * as server from "../server"
|
||||
// const http = require("http")
|
||||
// const Server = require("./server")
|
||||
import { Server } from './server'
|
||||
|
||||
/**
|
||||
* Captures upgrade requests for a http.Server.
|
||||
*
|
||||
* @param {http.Server} server
|
||||
* @param {Object} options
|
||||
* @return {Server} engine server
|
||||
* @api public
|
||||
*/
|
||||
|
||||
function attach(srv, options) {
|
||||
const engine = new Server(options)
|
||||
engine.attach(server.attach(srv, options), options)
|
||||
return engine
|
||||
}
|
||||
|
||||
export = {
|
||||
attach
|
||||
}
|
||||
690
packages/websocket/src/engine.io/server.ts
Normal file
690
packages/websocket/src/engine.io/server.ts
Normal file
@@ -0,0 +1,690 @@
|
||||
const qs = require("querystring")
|
||||
const parse = require("url").parse
|
||||
// const base64id = require("base64id")
|
||||
import transports from './transports'
|
||||
import { EventEmitter } from 'events'
|
||||
// const EventEmitter = require("events").EventEmitter
|
||||
import { Socket } from './socket'
|
||||
// const debug = require("debug")("engine")
|
||||
const debug = function (...args) { }
|
||||
// const cookieMod = require("cookie")
|
||||
|
||||
// const DEFAULT_WS_ENGINE = require("ws").Server;
|
||||
import { WebSocketServer } from '../server'
|
||||
import { Transport } from './transport'
|
||||
const DEFAULT_WS_ENGINE = WebSocketServer
|
||||
|
||||
import { Request } from '../server/request'
|
||||
import { WebSocketClient } from '../server/client'
|
||||
|
||||
export class Server extends EventEmitter {
|
||||
public static errors = {
|
||||
UNKNOWN_TRANSPORT: 0,
|
||||
UNKNOWN_SID: 1,
|
||||
BAD_HANDSHAKE_METHOD: 2,
|
||||
BAD_REQUEST: 3,
|
||||
FORBIDDEN: 4,
|
||||
UNSUPPORTED_PROTOCOL_VERSION: 5
|
||||
}
|
||||
|
||||
public static errorMessages = {
|
||||
0: "Transport unknown",
|
||||
1: "Session ID unknown",
|
||||
2: "Bad handshake method",
|
||||
3: "Bad request",
|
||||
4: "Forbidden",
|
||||
5: "Unsupported protocol version"
|
||||
}
|
||||
|
||||
private clients = {}
|
||||
private clientsCount = 0
|
||||
public opts: any
|
||||
|
||||
private corsMiddleware: any
|
||||
|
||||
private ws: any
|
||||
private perMessageDeflate: any
|
||||
|
||||
constructor(opts: any = {}) {
|
||||
super()
|
||||
this.opts = Object.assign(
|
||||
{
|
||||
wsEngine: DEFAULT_WS_ENGINE,
|
||||
pingTimeout: 20000,
|
||||
pingInterval: 25000,
|
||||
upgradeTimeout: 10000,
|
||||
maxHttpBufferSize: 1e6,
|
||||
transports: Object.keys(transports),
|
||||
allowUpgrades: true,
|
||||
httpCompression: {
|
||||
threshold: 1024
|
||||
},
|
||||
cors: false,
|
||||
allowEIO3: false
|
||||
},
|
||||
opts
|
||||
)
|
||||
|
||||
// if (opts.cookie) {
|
||||
// this.opts.cookie = Object.assign(
|
||||
// {
|
||||
// name: "io",
|
||||
// path: "/",
|
||||
// httpOnly: opts.cookie.path !== false,
|
||||
// sameSite: "lax"
|
||||
// },
|
||||
// opts.cookie
|
||||
// )
|
||||
// }
|
||||
|
||||
// if (this.opts.cors) {
|
||||
// this.corsMiddleware = require("cors")(this.opts.cors)
|
||||
// }
|
||||
|
||||
// if (opts.perMessageDeflate) {
|
||||
// this.opts.perMessageDeflate = Object.assign(
|
||||
// {
|
||||
// threshold: 1024
|
||||
// },
|
||||
// opts.perMessageDeflate
|
||||
// )
|
||||
// }
|
||||
|
||||
// this.init()
|
||||
}
|
||||
|
||||
// /**
|
||||
// * Initialize websocket server
|
||||
// *
|
||||
// * @api private
|
||||
// */
|
||||
// init() {
|
||||
// if (!~this.opts.transports.indexOf("websocket")) return
|
||||
|
||||
// if (this.ws) this.ws.close()
|
||||
|
||||
// this.ws = new this.opts.wsEngine({
|
||||
// noServer: true,
|
||||
// clientTracking: false,
|
||||
// perMessageDeflate: this.opts.perMessageDeflate,
|
||||
// maxPayload: this.opts.maxHttpBufferSize
|
||||
// })
|
||||
|
||||
// if (typeof this.ws.on === "function") {
|
||||
// this.ws.on("headers", (headersArray, req) => {
|
||||
// // note: 'ws' uses an array of headers, while Engine.IO uses an object (response.writeHead() accepts both formats)
|
||||
// // we could also try to parse the array and then sync the values, but that will be error-prone
|
||||
// const additionalHeaders = {}
|
||||
|
||||
// const isInitialRequest = !req._query.sid
|
||||
// if (isInitialRequest) {
|
||||
// this.emit("initial_headers", additionalHeaders, req)
|
||||
// }
|
||||
|
||||
// this.emit("headers", additionalHeaders, req)
|
||||
|
||||
// Object.keys(additionalHeaders).forEach(key => {
|
||||
// headersArray.push(`${key}: ${additionalHeaders[key]}`)
|
||||
// })
|
||||
// })
|
||||
// }
|
||||
// }
|
||||
|
||||
/**
|
||||
* Returns a list of available transports for upgrade given a certain transport.
|
||||
*
|
||||
* @return {Array}
|
||||
* @api public
|
||||
*/
|
||||
upgrades(transport): Array<any> {
|
||||
if (!this.opts.allowUpgrades) return []
|
||||
return transports[transport].upgradesTo || []
|
||||
}
|
||||
|
||||
// /**
|
||||
// * Verifies a request.
|
||||
// *
|
||||
// * @param {http.IncomingMessage}
|
||||
// * @return {Boolean} whether the request is valid
|
||||
// * @api private
|
||||
// */
|
||||
// verify(req, upgrade, fn) {
|
||||
// // transport check
|
||||
// const transport = req._query.transport
|
||||
// if (!~this.opts.transports.indexOf(transport)) {
|
||||
// debug('unknown transport "%s"', transport)
|
||||
// return fn(Server.errors.UNKNOWN_TRANSPORT, { transport })
|
||||
// }
|
||||
|
||||
// // 'Origin' header check
|
||||
// const isOriginInvalid = checkInvalidHeaderChar(req.headers.origin)
|
||||
// if (isOriginInvalid) {
|
||||
// const origin = req.headers.origin
|
||||
// req.headers.origin = null
|
||||
// debug("origin header invalid")
|
||||
// return fn(Server.errors.BAD_REQUEST, {
|
||||
// name: "INVALID_ORIGIN",
|
||||
// origin
|
||||
// })
|
||||
// }
|
||||
|
||||
// // sid check
|
||||
// const sid = req._query.sid
|
||||
// if (sid) {
|
||||
// if (!this.clients.hasOwnProperty(sid)) {
|
||||
// debug('unknown sid "%s"', sid)
|
||||
// return fn(Server.errors.UNKNOWN_SID, {
|
||||
// sid
|
||||
// })
|
||||
// }
|
||||
// const previousTransport = this.clients[sid].transport.name
|
||||
// if (!upgrade && previousTransport !== transport) {
|
||||
// debug("bad request: unexpected transport without upgrade")
|
||||
// return fn(Server.errors.BAD_REQUEST, {
|
||||
// name: "TRANSPORT_MISMATCH",
|
||||
// transport,
|
||||
// previousTransport
|
||||
// })
|
||||
// }
|
||||
// } else {
|
||||
// // handshake is GET only
|
||||
// if ("GET" !== req.method) {
|
||||
// return fn(Server.errors.BAD_HANDSHAKE_METHOD, {
|
||||
// method: req.method
|
||||
// })
|
||||
// }
|
||||
|
||||
// if (!this.opts.allowRequest) return fn()
|
||||
|
||||
// return this.opts.allowRequest(req, (message, success) => {
|
||||
// if (!success) {
|
||||
// return fn(Server.errors.FORBIDDEN, {
|
||||
// message
|
||||
// })
|
||||
// }
|
||||
// fn()
|
||||
// })
|
||||
// }
|
||||
|
||||
// fn()
|
||||
// }
|
||||
|
||||
/**
|
||||
* Prepares a request by processing the query string.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
prepare(req) {
|
||||
// try to leverage pre-existing `req._query` (e.g: from connect)
|
||||
if (!req._query) {
|
||||
req._query = ~req.url.indexOf("?") ? qs.parse(parse(req.url).query) : {}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes all clients.
|
||||
*
|
||||
* @api public
|
||||
*/
|
||||
close() {
|
||||
debug("closing all open clients")
|
||||
for (let i in this.clients) {
|
||||
if (this.clients.hasOwnProperty(i)) {
|
||||
this.clients[i].close(true)
|
||||
}
|
||||
}
|
||||
if (this.ws) {
|
||||
debug("closing webSocketServer")
|
||||
this.ws.close()
|
||||
// don't delete this.ws because it can be used again if the http server starts listening again
|
||||
}
|
||||
return this
|
||||
}
|
||||
|
||||
// /**
|
||||
// * Handles an Engine.IO HTTP request.
|
||||
// *
|
||||
// * @param {http.IncomingMessage} request
|
||||
// * @param {http.ServerResponse|http.OutgoingMessage} response
|
||||
// * @api public
|
||||
// */
|
||||
// handleRequest(req, res) {
|
||||
// debug('handling "%s" http request "%s"', req.method, req.url)
|
||||
// this.prepare(req)
|
||||
// req.res = res
|
||||
|
||||
// const callback = (errorCode, errorContext) => {
|
||||
// if (errorCode !== undefined) {
|
||||
// this.emit("connection_error", {
|
||||
// req,
|
||||
// code: errorCode,
|
||||
// message: Server.errorMessages[errorCode],
|
||||
// context: errorContext
|
||||
// })
|
||||
// abortRequest(res, errorCode, errorContext)
|
||||
// return
|
||||
// }
|
||||
|
||||
// if (req._query.sid) {
|
||||
// debug("setting new request for existing client")
|
||||
// this.clients[req._query.sid].transport.onRequest(req)
|
||||
// } else {
|
||||
// const closeConnection = (errorCode, errorContext) =>
|
||||
// abortRequest(res, errorCode, errorContext)
|
||||
// this.handshake(req._query.transport, req, closeConnection)
|
||||
// }
|
||||
// }
|
||||
|
||||
// if (this.corsMiddleware) {
|
||||
// this.corsMiddleware.call(null, req, res, () => {
|
||||
// this.verify(req, false, callback)
|
||||
// })
|
||||
// } else {
|
||||
// this.verify(req, false, callback)
|
||||
// }
|
||||
// }
|
||||
|
||||
/**
|
||||
* generate a socket id.
|
||||
* Overwrite this method to generate your custom socket id
|
||||
*
|
||||
* @param {Object} request object
|
||||
* @api public
|
||||
*/
|
||||
generateId(req) {
|
||||
return req.id
|
||||
}
|
||||
|
||||
/**
|
||||
* Handshakes a new client.
|
||||
*
|
||||
* @param {String} transport name
|
||||
* @param {Object} request object
|
||||
* @param {Function} closeConnection
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
// @java-patch sync handshake
|
||||
handshake(transportName, req, closeConnection: (code: number) => void) {
|
||||
console.debug('engine.io server handshake transport', transportName, 'from', req.url)
|
||||
const protocol = req._query.EIO === "4" ? 4 : 3 // 3rd revision by default
|
||||
if (protocol === 3 && !this.opts.allowEIO3) {
|
||||
debug("unsupported protocol version")
|
||||
this.emit("connection_error", {
|
||||
req,
|
||||
code: Server.errors.UNSUPPORTED_PROTOCOL_VERSION,
|
||||
message:
|
||||
Server.errorMessages[Server.errors.UNSUPPORTED_PROTOCOL_VERSION],
|
||||
context: {
|
||||
protocol
|
||||
}
|
||||
})
|
||||
closeConnection(Server.errors.UNSUPPORTED_PROTOCOL_VERSION)
|
||||
return
|
||||
}
|
||||
|
||||
let id
|
||||
try {
|
||||
id = this.generateId(req)
|
||||
} catch (e) {
|
||||
console.debug("error while generating an id")
|
||||
this.emit("connection_error", {
|
||||
req,
|
||||
code: Server.errors.BAD_REQUEST,
|
||||
message: Server.errorMessages[Server.errors.BAD_REQUEST],
|
||||
context: {
|
||||
name: "ID_GENERATION_ERROR",
|
||||
error: e
|
||||
}
|
||||
})
|
||||
closeConnection(Server.errors.BAD_REQUEST)
|
||||
return
|
||||
}
|
||||
|
||||
console.debug('engine.io server handshaking client "' + id + '"')
|
||||
|
||||
try {
|
||||
var transport: Transport = new transports[transportName](req)
|
||||
if ("websocket" !== transportName) {
|
||||
throw new Error('Unsupport polling at MiaoScript!')
|
||||
}
|
||||
// if ("polling" === transportName) {
|
||||
// transport.maxHttpBufferSize = this.opts.maxHttpBufferSize
|
||||
// transport.httpCompression = this.opts.httpCompression
|
||||
// } else if ("websocket" === transportName) {
|
||||
transport.perMessageDeflate = this.opts.perMessageDeflate
|
||||
// }
|
||||
|
||||
if (req._query && req._query.b64) {
|
||||
transport.supportsBinary = false
|
||||
} else {
|
||||
transport.supportsBinary = true
|
||||
}
|
||||
} catch (e) {
|
||||
console.ex(e)
|
||||
this.emit("connection_error", {
|
||||
req,
|
||||
code: Server.errors.BAD_REQUEST,
|
||||
message: Server.errorMessages[Server.errors.BAD_REQUEST],
|
||||
context: {
|
||||
name: "TRANSPORT_HANDSHAKE_ERROR",
|
||||
error: e
|
||||
}
|
||||
})
|
||||
closeConnection(Server.errors.BAD_REQUEST)
|
||||
return
|
||||
}
|
||||
console.debug(`engine.io server create socket ${id} from transport ${transport.name} protocol ${protocol}`)
|
||||
const socket = new Socket(id, this, transport, req, protocol)
|
||||
|
||||
transport.on("headers", (headers, req) => {
|
||||
const isInitialRequest = !req._query.sid
|
||||
|
||||
if (isInitialRequest) {
|
||||
if (this.opts.cookie) {
|
||||
headers["Set-Cookie"] = [
|
||||
// cookieMod.serialize(this.opts.cookie.name, id, this.opts.cookie)
|
||||
]
|
||||
}
|
||||
this.emit("initial_headers", headers, req)
|
||||
}
|
||||
this.emit("headers", headers, req)
|
||||
})
|
||||
|
||||
transport.onRequest(req)
|
||||
|
||||
this.clients[id] = socket
|
||||
this.clientsCount++
|
||||
|
||||
socket.once("close", () => {
|
||||
delete this.clients[id]
|
||||
this.clientsCount--
|
||||
})
|
||||
this.emit("connection", socket)
|
||||
}
|
||||
|
||||
// /**
|
||||
// * Handles an Engine.IO HTTP Upgrade.
|
||||
// *
|
||||
// * @api public
|
||||
// */
|
||||
// handleUpgrade(req, socket, upgradeHead) {
|
||||
// this.prepare(req)
|
||||
|
||||
// this.verify(req, true, (errorCode, errorContext) => {
|
||||
// if (errorCode) {
|
||||
// this.emit("connection_error", {
|
||||
// req,
|
||||
// code: errorCode,
|
||||
// message: Server.errorMessages[errorCode],
|
||||
// context: errorContext
|
||||
// })
|
||||
// abortUpgrade(socket, errorCode, errorContext)
|
||||
// return
|
||||
// }
|
||||
|
||||
// const head = Buffer.from(upgradeHead) // eslint-disable-line node/no-deprecated-api
|
||||
// upgradeHead = null
|
||||
|
||||
// // delegate to ws
|
||||
// this.ws.handleUpgrade(req, socket, head, websocket => {
|
||||
// this.onWebSocket(req, socket, websocket)
|
||||
// })
|
||||
// })
|
||||
// }
|
||||
|
||||
/**
|
||||
* Called upon a ws.io connection.
|
||||
*
|
||||
* @param {ws.Socket} websocket
|
||||
* @api private
|
||||
*/
|
||||
onWebSocket(req: Request, socket, websocket: WebSocketClient) {
|
||||
websocket.on("error", onUpgradeError)
|
||||
|
||||
if (
|
||||
transports[req._query.transport] !== undefined &&
|
||||
!transports[req._query.transport].prototype.handlesUpgrades
|
||||
) {
|
||||
console.debug("transport doesnt handle upgraded requests")
|
||||
websocket.close()
|
||||
return
|
||||
}
|
||||
|
||||
// get client id
|
||||
const id = req._query.sid
|
||||
|
||||
// keep a reference to the ws.Socket
|
||||
req.websocket = websocket
|
||||
|
||||
if (id) {
|
||||
const client = this.clients[id]
|
||||
if (!client) {
|
||||
console.debug("upgrade attempt for closed client")
|
||||
websocket.close()
|
||||
} else if (client.upgrading) {
|
||||
console.debug("transport has already been trying to upgrade")
|
||||
websocket.close()
|
||||
} else if (client.upgraded) {
|
||||
console.debug("transport had already been upgraded")
|
||||
websocket.close()
|
||||
} else {
|
||||
console.debug("upgrading existing transport")
|
||||
|
||||
// transport error handling takes over
|
||||
websocket.removeListener("error", onUpgradeError)
|
||||
|
||||
const transport = new transports[req._query.transport](req)
|
||||
if (req._query && req._query.b64) {
|
||||
transport.supportsBinary = false
|
||||
} else {
|
||||
transport.supportsBinary = true
|
||||
}
|
||||
transport.perMessageDeflate = this.perMessageDeflate
|
||||
client.maybeUpgrade(transport)
|
||||
}
|
||||
} else {
|
||||
// transport error handling takes over
|
||||
websocket.removeListener("error", onUpgradeError)
|
||||
|
||||
// const closeConnection = (errorCode, errorContext) =>
|
||||
// abortUpgrade(socket, errorCode, errorContext)
|
||||
this.handshake(req._query.transport, req, () => { })
|
||||
}
|
||||
|
||||
function onUpgradeError() {
|
||||
console.debug("websocket error before upgrade")
|
||||
// websocket.close() not needed
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Captures upgrade requests for a http.Server.
|
||||
*
|
||||
* @param {http.Server} server
|
||||
* @param {Object} options
|
||||
* @api public
|
||||
*/
|
||||
attach(server, options: any = {}) {
|
||||
// let path = (options.path || "/engine.io").replace(/\/$/, "")
|
||||
|
||||
// const destroyUpgradeTimeout = options.destroyUpgradeTimeout || 1000
|
||||
|
||||
// // normalize path
|
||||
// path += "/"
|
||||
|
||||
// function check(req) {
|
||||
// return path === req.url.substr(0, path.length)
|
||||
// }
|
||||
|
||||
// cache and clean up listeners
|
||||
// const listeners = server.listeners("request").slice(0)
|
||||
// server.removeAllListeners("request")
|
||||
server.on("close", this.close.bind(this))
|
||||
// server.on("listening", this.init.bind(this))
|
||||
// @java-patch transfer to Netty Server
|
||||
server.on("connect", (request: Request, websocket: WebSocketClient) => {
|
||||
console.debug('Engine.IO connect client from', request.url)
|
||||
this.prepare(request)
|
||||
this.onWebSocket(request, undefined, websocket)
|
||||
})
|
||||
// set server as ws server
|
||||
this.ws = server
|
||||
|
||||
// // add request handler
|
||||
// server.on("request", (req, res) => {
|
||||
// if (check(req)) {
|
||||
// debug('intercepting request for path "%s"', path)
|
||||
// this.handleRequest(req, res)
|
||||
// } else {
|
||||
// let i = 0
|
||||
// const l = listeners.length
|
||||
// for (; i < l; i++) {
|
||||
// listeners[i].call(server, req, res)
|
||||
// }
|
||||
// }
|
||||
// })
|
||||
|
||||
// if (~this.opts.transports.indexOf("websocket")) {
|
||||
// server.on("upgrade", (req, socket, head) => {
|
||||
// if (check(req)) {
|
||||
// this.handleUpgrade(req, socket, head)
|
||||
// } else if (false !== options.destroyUpgrade) {
|
||||
// // default node behavior is to disconnect when no handlers
|
||||
// // but by adding a handler, we prevent that
|
||||
// // and if no eio thing handles the upgrade
|
||||
// // then the socket needs to die!
|
||||
// setTimeout(function () {
|
||||
// if (socket.writable && socket.bytesWritten <= 0) {
|
||||
// return socket.end()
|
||||
// }
|
||||
// }, destroyUpgradeTimeout)
|
||||
// }
|
||||
// })
|
||||
// }
|
||||
}
|
||||
}
|
||||
|
||||
// /**
|
||||
// * Close the HTTP long-polling request
|
||||
// *
|
||||
// * @param res - the response object
|
||||
// * @param errorCode - the error code
|
||||
// * @param errorContext - additional error context
|
||||
// *
|
||||
// * @api private
|
||||
// */
|
||||
|
||||
// function abortRequest(res, errorCode, errorContext) {
|
||||
// const statusCode = errorCode === Server.errors.FORBIDDEN ? 403 : 400
|
||||
// const message =
|
||||
// errorContext && errorContext.message
|
||||
// ? errorContext.message
|
||||
// : Server.errorMessages[errorCode]
|
||||
|
||||
// res.writeHead(statusCode, { "Content-Type": "application/json" })
|
||||
// res.end(
|
||||
// JSON.stringify({
|
||||
// code: errorCode,
|
||||
// message
|
||||
// })
|
||||
// )
|
||||
// }
|
||||
|
||||
// /**
|
||||
// * Close the WebSocket connection
|
||||
// *
|
||||
// * @param {net.Socket} socket
|
||||
// * @param {string} errorCode - the error code
|
||||
// * @param {object} errorContext - additional error context
|
||||
// *
|
||||
// * @api private
|
||||
// */
|
||||
|
||||
// function abortUpgrade(socket, errorCode, errorContext: any = {}) {
|
||||
// socket.on("error", () => {
|
||||
// debug("ignoring error from closed connection")
|
||||
// })
|
||||
// if (socket.writable) {
|
||||
// const message = errorContext.message || Server.errorMessages[errorCode]
|
||||
// const length = Buffer.byteLength(message)
|
||||
// socket.write(
|
||||
// "HTTP/1.1 400 Bad Request\r\n" +
|
||||
// "Connection: close\r\n" +
|
||||
// "Content-type: text/html\r\n" +
|
||||
// "Content-Length: " +
|
||||
// length +
|
||||
// "\r\n" +
|
||||
// "\r\n" +
|
||||
// message
|
||||
// )
|
||||
// }
|
||||
// socket.destroy()
|
||||
// }
|
||||
|
||||
// module.exports = Server
|
||||
|
||||
/* eslint-disable */
|
||||
|
||||
// /**
|
||||
// * From https://github.com/nodejs/node/blob/v8.4.0/lib/_http_common.js#L303-L354
|
||||
// *
|
||||
// * True if val contains an invalid field-vchar
|
||||
// * field-value = *( field-content / obs-fold )
|
||||
// * field-content = field-vchar [ 1*( SP / HTAB ) field-vchar ]
|
||||
// * field-vchar = VCHAR / obs-text
|
||||
// *
|
||||
// * checkInvalidHeaderChar() is currently designed to be inlinable by v8,
|
||||
// * so take care when making changes to the implementation so that the source
|
||||
// * code size does not exceed v8's default max_inlined_source_size setting.
|
||||
// **/
|
||||
// // prettier-ignore
|
||||
// const validHdrChars = [
|
||||
// 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, // 0 - 15
|
||||
// 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 16 - 31
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 32 - 47
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 48 - 63
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 64 - 79
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 80 - 95
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 96 - 111
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, // 112 - 127
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 128 ...
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
// 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // ... 255
|
||||
// ]
|
||||
|
||||
// function checkInvalidHeaderChar(val) {
|
||||
// val += ""
|
||||
// if (val.length < 1) return false
|
||||
// if (!validHdrChars[val.charCodeAt(0)]) {
|
||||
// debug('invalid header, index 0, char "%s"', val.charCodeAt(0))
|
||||
// return true
|
||||
// }
|
||||
// if (val.length < 2) return false
|
||||
// if (!validHdrChars[val.charCodeAt(1)]) {
|
||||
// debug('invalid header, index 1, char "%s"', val.charCodeAt(1))
|
||||
// return true
|
||||
// }
|
||||
// if (val.length < 3) return false
|
||||
// if (!validHdrChars[val.charCodeAt(2)]) {
|
||||
// debug('invalid header, index 2, char "%s"', val.charCodeAt(2))
|
||||
// return true
|
||||
// }
|
||||
// if (val.length < 4) return false
|
||||
// if (!validHdrChars[val.charCodeAt(3)]) {
|
||||
// debug('invalid header, index 3, char "%s"', val.charCodeAt(3))
|
||||
// return true
|
||||
// }
|
||||
// for (let i = 4; i < val.length; ++i) {
|
||||
// if (!validHdrChars[val.charCodeAt(i)]) {
|
||||
// debug('invalid header, index "%i", char "%s"', i, val.charCodeAt(i))
|
||||
// return true
|
||||
// }
|
||||
// }
|
||||
// return false
|
||||
// }
|
||||
530
packages/websocket/src/engine.io/socket.ts
Normal file
530
packages/websocket/src/engine.io/socket.ts
Normal file
@@ -0,0 +1,530 @@
|
||||
import { EventEmitter } from "events"
|
||||
import { Server } from "./server"
|
||||
import { Transport } from "./transport"
|
||||
import type { Request } from "../server/request"
|
||||
// const debug = require("debug")("engine:socket")
|
||||
|
||||
export class Socket extends EventEmitter {
|
||||
public id: string
|
||||
private server: Server
|
||||
private upgrading = false
|
||||
private upgraded = false
|
||||
public readyState = "opening"
|
||||
private writeBuffer = []
|
||||
private packetsFn = []
|
||||
private sentCallbackFn = []
|
||||
private cleanupFn = []
|
||||
public request: Request
|
||||
public protocol: number
|
||||
public remoteAddress: any
|
||||
public transport: Transport
|
||||
|
||||
private checkIntervalTimer: NodeJS.Timeout
|
||||
private upgradeTimeoutTimer: NodeJS.Timeout
|
||||
private pingTimeoutTimer: NodeJS.Timeout
|
||||
private pingIntervalTimer: NodeJS.Timeout
|
||||
|
||||
/**
|
||||
* Client class (abstract).
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
constructor(id: string, server: Server, transport: Transport, req: Request, protocol: number) {
|
||||
super()
|
||||
this.id = id
|
||||
this.server = server
|
||||
this.request = req
|
||||
this.protocol = protocol
|
||||
|
||||
// Cache IP since it might not be in the req later
|
||||
if (req.websocket && req.websocket._socket) {
|
||||
this.remoteAddress = req.websocket._socket.remoteAddress
|
||||
} else {
|
||||
this.remoteAddress = req.connection.remoteAddress
|
||||
}
|
||||
|
||||
this.checkIntervalTimer = null
|
||||
this.upgradeTimeoutTimer = null
|
||||
this.pingTimeoutTimer = null
|
||||
this.pingIntervalTimer = null
|
||||
|
||||
this.setTransport(transport)
|
||||
this.onOpen()
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport considered open.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onOpen() {
|
||||
this.readyState = "open"
|
||||
|
||||
// sends an `open` packet
|
||||
this.transport.sid = this.id
|
||||
this.sendPacket(
|
||||
"open",
|
||||
JSON.stringify({
|
||||
sid: this.id,
|
||||
upgrades: this.getAvailableUpgrades(),
|
||||
pingInterval: this.server.opts.pingInterval,
|
||||
pingTimeout: this.server.opts.pingTimeout
|
||||
})
|
||||
)
|
||||
|
||||
if (this.server.opts.initialPacket) {
|
||||
this.sendPacket("message", this.server.opts.initialPacket)
|
||||
}
|
||||
|
||||
this.emit("open")
|
||||
|
||||
if (this.protocol === 3) {
|
||||
// in protocol v3, the client sends a ping, and the server answers with a pong
|
||||
this.resetPingTimeout(
|
||||
this.server.opts.pingInterval + this.server.opts.pingTimeout
|
||||
)
|
||||
} else {
|
||||
// in protocol v4, the server sends a ping, and the client answers with a pong
|
||||
this.schedulePing()
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport packet.
|
||||
*
|
||||
* @param {Object} packet
|
||||
* @api private
|
||||
*/
|
||||
onPacket(packet: { type: any; data: any }) {
|
||||
if ("open" !== this.readyState) {
|
||||
console.debug("packet received with closed socket")
|
||||
return
|
||||
}
|
||||
// export packet event
|
||||
// debug(`received packet ${packet.type}`)
|
||||
this.emit("packet", packet)
|
||||
|
||||
// Reset ping timeout on any packet, incoming data is a good sign of
|
||||
// other side's liveness
|
||||
this.resetPingTimeout(
|
||||
this.server.opts.pingInterval + this.server.opts.pingTimeout
|
||||
)
|
||||
|
||||
switch (packet.type) {
|
||||
case "ping":
|
||||
if (this.transport.protocol !== 3) {
|
||||
this.onError("invalid heartbeat direction")
|
||||
return
|
||||
}
|
||||
// debug("got ping")
|
||||
this.sendPacket("pong")
|
||||
this.emit("heartbeat")
|
||||
break
|
||||
|
||||
case "pong":
|
||||
if (this.transport.protocol === 3) {
|
||||
this.onError("invalid heartbeat direction")
|
||||
return
|
||||
}
|
||||
// debug("got pong")
|
||||
this.schedulePing()
|
||||
this.emit("heartbeat")
|
||||
break
|
||||
|
||||
case "error":
|
||||
this.onClose("parse error")
|
||||
break
|
||||
|
||||
case "message":
|
||||
this.emit("data", packet.data)
|
||||
this.emit("message", packet.data)
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport error.
|
||||
*
|
||||
* @param {Error} error object
|
||||
* @api private
|
||||
*/
|
||||
onError(err: string) {
|
||||
// debug("transport error")
|
||||
this.onClose("transport error", err)
|
||||
}
|
||||
|
||||
/**
|
||||
* Pings client every `this.pingInterval` and expects response
|
||||
* within `this.pingTimeout` or closes connection.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
schedulePing() {
|
||||
clearTimeout(this.pingIntervalTimer)
|
||||
this.pingIntervalTimer = setTimeout(() => {
|
||||
// debug(
|
||||
// "writing ping packet - expecting pong within %sms",
|
||||
// this.server.opts.pingTimeout
|
||||
// )
|
||||
this.sendPacket("ping")
|
||||
this.resetPingTimeout(this.server.opts.pingTimeout)
|
||||
}, this.server.opts.pingInterval)
|
||||
}
|
||||
|
||||
/**
|
||||
* Resets ping timeout.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
resetPingTimeout(timeout: number) {
|
||||
clearTimeout(this.pingTimeoutTimer)
|
||||
this.pingTimeoutTimer = setTimeout(() => {
|
||||
if (this.readyState === "closed") return
|
||||
this.onClose("ping timeout")
|
||||
}, timeout)
|
||||
}
|
||||
|
||||
/**
|
||||
* Attaches handlers for the given transport.
|
||||
*
|
||||
* @param {Transport} transport
|
||||
* @api private
|
||||
*/
|
||||
setTransport(transport: Transport) {
|
||||
console.debug(`engine.io socket ${this.id} set transport ${transport.name}`)
|
||||
const onError = this.onError.bind(this)
|
||||
const onPacket = this.onPacket.bind(this)
|
||||
const flush = this.flush.bind(this)
|
||||
const onClose = this.onClose.bind(this, "transport close")
|
||||
|
||||
this.transport = transport
|
||||
this.transport.once("error", onError)
|
||||
this.transport.on("packet", onPacket)
|
||||
this.transport.on("drain", flush)
|
||||
this.transport.once("close", onClose)
|
||||
// this function will manage packet events (also message callbacks)
|
||||
this.setupSendCallback()
|
||||
|
||||
this.cleanupFn.push(function () {
|
||||
transport.removeListener("error", onError)
|
||||
transport.removeListener("packet", onPacket)
|
||||
transport.removeListener("drain", flush)
|
||||
transport.removeListener("close", onClose)
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* Upgrades socket to the given transport
|
||||
*
|
||||
* @param {Transport} transport
|
||||
* @api private
|
||||
*/
|
||||
maybeUpgrade(transport: Transport) {
|
||||
console.debug(
|
||||
'might upgrade socket transport from "', this.transport.name, '" to "', transport.name, '"'
|
||||
)
|
||||
|
||||
this.upgrading = true
|
||||
|
||||
// set transport upgrade timer
|
||||
this.upgradeTimeoutTimer = setTimeout(() => {
|
||||
console.debug("client did not complete upgrade - closing transport")
|
||||
cleanup()
|
||||
if ("open" === transport.readyState) {
|
||||
transport.close()
|
||||
}
|
||||
}, this.server.opts.upgradeTimeout)
|
||||
|
||||
const onPacket = (packet: { type: string; data: string }) => {
|
||||
if ("ping" === packet.type && "probe" === packet.data) {
|
||||
transport.send([{ type: "pong", data: "probe" }])
|
||||
this.emit("upgrading", transport)
|
||||
clearInterval(this.checkIntervalTimer)
|
||||
this.checkIntervalTimer = setInterval(check, 100)
|
||||
} else if ("upgrade" === packet.type && this.readyState !== "closed") {
|
||||
// debug("got upgrade packet - upgrading")
|
||||
cleanup()
|
||||
this.transport.discard()
|
||||
this.upgraded = true
|
||||
this.clearTransport()
|
||||
this.setTransport(transport)
|
||||
this.emit("upgrade", transport)
|
||||
this.flush()
|
||||
if (this.readyState === "closing") {
|
||||
transport.close(() => {
|
||||
this.onClose("forced close")
|
||||
})
|
||||
}
|
||||
} else {
|
||||
cleanup()
|
||||
transport.close()
|
||||
}
|
||||
}
|
||||
|
||||
// we force a polling cycle to ensure a fast upgrade
|
||||
const check = () => {
|
||||
if ("polling" === this.transport.name && this.transport.writable) {
|
||||
// debug("writing a noop packet to polling for fast upgrade")
|
||||
this.transport.send([{ type: "noop" }])
|
||||
}
|
||||
}
|
||||
|
||||
const cleanup = () => {
|
||||
this.upgrading = false
|
||||
|
||||
clearInterval(this.checkIntervalTimer)
|
||||
this.checkIntervalTimer = null
|
||||
|
||||
clearTimeout(this.upgradeTimeoutTimer)
|
||||
this.upgradeTimeoutTimer = null
|
||||
|
||||
transport.removeListener("packet", onPacket)
|
||||
transport.removeListener("close", onTransportClose)
|
||||
transport.removeListener("error", onError)
|
||||
this.removeListener("close", onClose)
|
||||
}
|
||||
|
||||
const onError = (err: string) => {
|
||||
// debug("client did not complete upgrade - %s", err)
|
||||
cleanup()
|
||||
transport.close()
|
||||
transport = null
|
||||
}
|
||||
|
||||
const onTransportClose = () => {
|
||||
onError("transport closed")
|
||||
}
|
||||
|
||||
const onClose = () => {
|
||||
onError("socket closed")
|
||||
}
|
||||
|
||||
transport.on("packet", onPacket)
|
||||
transport.once("close", onTransportClose)
|
||||
transport.once("error", onError)
|
||||
|
||||
this.once("close", onClose)
|
||||
}
|
||||
|
||||
/**
|
||||
* Clears listeners and timers associated with current transport.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
clearTransport() {
|
||||
let cleanup: () => void
|
||||
|
||||
const toCleanUp = this.cleanupFn.length
|
||||
|
||||
for (let i = 0; i < toCleanUp; i++) {
|
||||
cleanup = this.cleanupFn.shift()
|
||||
cleanup()
|
||||
}
|
||||
|
||||
// silence further transport errors and prevent uncaught exceptions
|
||||
this.transport.on("error", function () {
|
||||
// debug("error triggered by discarded transport")
|
||||
})
|
||||
|
||||
// ensure transport won't stay open
|
||||
this.transport.close()
|
||||
|
||||
clearTimeout(this.pingTimeoutTimer)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport considered closed.
|
||||
* Possible reasons: `ping timeout`, `client error`, `parse error`,
|
||||
* `transport error`, `server close`, `transport close`
|
||||
*/
|
||||
onClose(reason: string, description?: string) {
|
||||
if ("closed" !== this.readyState) {
|
||||
this.readyState = "closed"
|
||||
|
||||
// clear timers
|
||||
clearTimeout(this.pingIntervalTimer)
|
||||
clearTimeout(this.pingTimeoutTimer)
|
||||
|
||||
clearInterval(this.checkIntervalTimer)
|
||||
this.checkIntervalTimer = null
|
||||
clearTimeout(this.upgradeTimeoutTimer)
|
||||
// clean writeBuffer in next tick, so developers can still
|
||||
// grab the writeBuffer on 'close' event
|
||||
process.nextTick(() => {
|
||||
this.writeBuffer = []
|
||||
})
|
||||
this.packetsFn = []
|
||||
this.sentCallbackFn = []
|
||||
this.clearTransport()
|
||||
this.emit("close", reason, description)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Setup and manage send callback
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
setupSendCallback() {
|
||||
// the message was sent successfully, execute the callback
|
||||
const onDrain = () => {
|
||||
if (this.sentCallbackFn.length > 0) {
|
||||
const seqFn = this.sentCallbackFn.splice(0, 1)[0]
|
||||
if ("function" === typeof seqFn) {
|
||||
// debug("executing send callback")
|
||||
seqFn(this.transport)
|
||||
} else if (Array.isArray(seqFn)) {
|
||||
// debug("executing batch send callback")
|
||||
const l = seqFn.length
|
||||
let i = 0
|
||||
for (; i < l; i++) {
|
||||
if ("function" === typeof seqFn[i]) {
|
||||
seqFn[i](this.transport)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.transport.on("drain", onDrain)
|
||||
|
||||
this.cleanupFn.push(() => {
|
||||
this.transport.removeListener("drain", onDrain)
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* Sends a message packet.
|
||||
*
|
||||
* @param {String} message
|
||||
* @param {Object} options
|
||||
* @param {Function} callback
|
||||
* @return {Socket} for chaining
|
||||
* @api public
|
||||
*/
|
||||
send(data: any, options: any, callback: any) {
|
||||
this.sendPacket("message", data, options, callback)
|
||||
return this
|
||||
}
|
||||
|
||||
write(data: any, options: any, callback?: any) {
|
||||
this.sendPacket("message", data, options, callback)
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* Sends a packet.
|
||||
*
|
||||
* @param {String} packet type
|
||||
* @param {String} optional, data
|
||||
* @param {Object} options
|
||||
* @api private
|
||||
*/
|
||||
sendPacket(type: string, data?: string, options?: { compress?: any }, callback?: undefined) {
|
||||
if ("function" === typeof options) {
|
||||
callback = options
|
||||
options = null
|
||||
}
|
||||
|
||||
options = options || {}
|
||||
options.compress = false !== options.compress
|
||||
|
||||
if ("closing" !== this.readyState && "closed" !== this.readyState) {
|
||||
// console.debug('sending packet "%s" (%s)', type, data)
|
||||
|
||||
const packet: any = {
|
||||
type: type,
|
||||
options: options
|
||||
}
|
||||
if (data) packet.data = data
|
||||
|
||||
// exports packetCreate event
|
||||
this.emit("packetCreate", packet)
|
||||
|
||||
this.writeBuffer.push(packet)
|
||||
|
||||
// add send callback to object, if defined
|
||||
if (callback) this.packetsFn.push(callback)
|
||||
|
||||
this.flush()
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Attempts to flush the packets buffer.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
flush() {
|
||||
if (
|
||||
"closed" !== this.readyState &&
|
||||
this.transport.writable &&
|
||||
this.writeBuffer.length
|
||||
) {
|
||||
console.trace("flushing buffer to transport")
|
||||
this.emit("flush", this.writeBuffer)
|
||||
this.server.emit("flush", this, this.writeBuffer)
|
||||
const wbuf = this.writeBuffer
|
||||
this.writeBuffer = []
|
||||
if (!this.transport.supportsFraming) {
|
||||
this.sentCallbackFn.push(this.packetsFn)
|
||||
} else {
|
||||
this.sentCallbackFn.push.apply(this.sentCallbackFn, this.packetsFn)
|
||||
}
|
||||
this.packetsFn = []
|
||||
this.transport.send(wbuf)
|
||||
this.emit("drain")
|
||||
this.server.emit("drain", this)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get available upgrades for this socket.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
getAvailableUpgrades() {
|
||||
const availableUpgrades = []
|
||||
const allUpgrades = this.server.upgrades(this.transport.name)
|
||||
let i = 0
|
||||
const l = allUpgrades.length
|
||||
for (; i < l; ++i) {
|
||||
const upg = allUpgrades[i]
|
||||
if (this.server.opts.transports.indexOf(upg) !== -1) {
|
||||
availableUpgrades.push(upg)
|
||||
}
|
||||
}
|
||||
return availableUpgrades
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes the socket and underlying transport.
|
||||
*
|
||||
* @param {Boolean} optional, discard
|
||||
* @return {Socket} for chaining
|
||||
* @api public
|
||||
*/
|
||||
close(discard?: any) {
|
||||
if ("open" !== this.readyState) return
|
||||
|
||||
this.readyState = "closing"
|
||||
|
||||
if (this.writeBuffer.length) {
|
||||
this.once("drain", this.closeTransport.bind(this, discard))
|
||||
return
|
||||
}
|
||||
|
||||
this.closeTransport(discard)
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes the underlying transport.
|
||||
*
|
||||
* @param {Boolean} discard
|
||||
* @api private
|
||||
*/
|
||||
closeTransport(discard: any) {
|
||||
if (discard) this.transport.discard()
|
||||
this.transport.close(this.onClose.bind(this, "forced close"))
|
||||
}
|
||||
}
|
||||
121
packages/websocket/src/engine.io/transport.ts
Normal file
121
packages/websocket/src/engine.io/transport.ts
Normal file
@@ -0,0 +1,121 @@
|
||||
import { EventEmitter } from 'events'
|
||||
import parser_v4 from "../engine.io-parser"
|
||||
import type { WebSocketClient } from '../server/client'
|
||||
/**
|
||||
* Noop function.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
|
||||
function noop() { }
|
||||
|
||||
export abstract class Transport extends EventEmitter {
|
||||
public sid: string
|
||||
public req /**http.IncomingMessage */
|
||||
public socket: WebSocketClient
|
||||
public writable: boolean
|
||||
public readyState: string
|
||||
public discarded: boolean
|
||||
public protocol: Number
|
||||
public parser: any
|
||||
public perMessageDeflate: any
|
||||
public supportsBinary: boolean = false
|
||||
|
||||
/**
|
||||
* Transport constructor.
|
||||
*
|
||||
* @param {http.IncomingMessage} request
|
||||
* @api public
|
||||
*/
|
||||
constructor(req) {
|
||||
super()
|
||||
this.readyState = "open"
|
||||
this.discarded = false
|
||||
this.protocol = req._query.EIO === "4" ? 4 : 3 // 3rd revision by default
|
||||
this.parser = parser_v4//= this.protocol === 4 ? parser_v4 : parser_v3
|
||||
}
|
||||
|
||||
/**
|
||||
* Flags the transport as discarded.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
discard() {
|
||||
this.discarded = true
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with an incoming HTTP request.
|
||||
*
|
||||
* @param {http.IncomingMessage} request
|
||||
* @api private
|
||||
*/
|
||||
onRequest(req) {
|
||||
console.debug(`engine.io transport ${this.socket.id} setting request`, JSON.stringify(req))
|
||||
this.req = req
|
||||
}
|
||||
|
||||
/**
|
||||
* Closes the transport.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
close(fn?) {
|
||||
if ("closed" === this.readyState || "closing" === this.readyState) return
|
||||
|
||||
this.readyState = "closing"
|
||||
this.doClose(fn || noop)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with a transport error.
|
||||
*
|
||||
* @param {String} message error
|
||||
* @param {Object} error description
|
||||
* @api private
|
||||
*/
|
||||
onError(msg: string, desc?: string) {
|
||||
if (this.listeners("error").length) {
|
||||
const err: any = new Error(msg)
|
||||
err.type = "TransportError"
|
||||
err.description = desc
|
||||
this.emit("error", err)
|
||||
} else {
|
||||
console.debug(`ignored transport error ${msg} (${desc})`)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with parsed out a packets from the data stream.
|
||||
*
|
||||
* @param {Object} packet
|
||||
* @api private
|
||||
*/
|
||||
onPacket(packet) {
|
||||
this.emit("packet", packet)
|
||||
}
|
||||
|
||||
/**
|
||||
* Called with the encoded packet data.
|
||||
*
|
||||
* @param {String} data
|
||||
* @api private
|
||||
*/
|
||||
onData(data) {
|
||||
this.onPacket(this.parser.decodePacket(data))
|
||||
}
|
||||
|
||||
/**
|
||||
* Called upon transport close.
|
||||
*
|
||||
* @api private
|
||||
*/
|
||||
onClose() {
|
||||
this.readyState = "closed"
|
||||
this.emit("close")
|
||||
}
|
||||
abstract get supportsFraming()
|
||||
abstract get name()
|
||||
abstract send(...args: any[])
|
||||
abstract doClose(d: Function)
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user