use worker threads
getPlayerList will temporailly not work for now because i am too lazy to fix the output thing
This commit is contained in:
parent
342e21a229
commit
b13456c9c8
2 changed files with 103 additions and 63 deletions
86
index.js
86
index.js
|
@ -2,84 +2,48 @@ const { VM } = require('vm2')
|
|||
const { Server } = require('socket.io')
|
||||
const util = require('util')
|
||||
const { stylize } = require('./colors')
|
||||
const randomstring = require('randomstring')
|
||||
const ChatMessage = require('prismarine-chat')('1.20.1')
|
||||
const mc = require('minecraft-protocol')
|
||||
const moment = require('moment-timezone')
|
||||
const crypto = require('crypto')
|
||||
const nbt = require('prismarine-nbt')
|
||||
|
||||
const BRIDGE_PREFIX = 'function:'
|
||||
const { Worker } = require('worker_threads')
|
||||
const path = require('path')
|
||||
|
||||
const io = new Server(3069)
|
||||
|
||||
io.on('connection', (socket) => {
|
||||
let functions
|
||||
let worker
|
||||
|
||||
let proxy
|
||||
function reset () {
|
||||
worker = new Worker(path.join(__dirname, 'vm.js'))
|
||||
|
||||
const handler = {
|
||||
get (target, prop) {
|
||||
if (!target[prop]) throw new Error(`Function "${prop}" not available`)
|
||||
worker.on('message', (msg) => {
|
||||
switch (msg.type) {
|
||||
case 'socketEmit':
|
||||
socket.emit(...msg.data)
|
||||
|
||||
return (...args) => target[prop](...args)
|
||||
break
|
||||
case 'socketOnce':
|
||||
socket.once(...msg.data)
|
||||
|
||||
break
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
reset()
|
||||
|
||||
socket.on('setFunctions', (jsonArray) => {
|
||||
const parsed = JSON.parse(jsonArray)
|
||||
|
||||
functions = {}
|
||||
|
||||
for (const eachFuntion of parsed) {
|
||||
functions[eachFuntion] = (...args) => {
|
||||
socket.emit(BRIDGE_PREFIX + eachFuntion, ...args)
|
||||
|
||||
return new Promise((resolve) => {
|
||||
socket.once(`functionOutput:${eachFuntion}`, (message, parseJSON) => {
|
||||
if (parseJSON) resolve(JSON.parse(message))
|
||||
else resolve(message)
|
||||
worker.postMessage({ type: 'setFunctions', jsonArray })
|
||||
})
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
proxy = new Proxy(functions, handler)
|
||||
|
||||
resetVM()
|
||||
})
|
||||
|
||||
let vm
|
||||
function resetVM () {
|
||||
vm = new VM({
|
||||
timeout: 3000,
|
||||
sandbox: {
|
||||
get bridge () { return proxy },
|
||||
randomstring,
|
||||
ChatMessage,
|
||||
mc,
|
||||
moment,
|
||||
crypto,
|
||||
nbt
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
resetVM()
|
||||
|
||||
socket.on('runCode', (transactionId, code) => {
|
||||
(async () => {
|
||||
try {
|
||||
const output = vm.run(code)
|
||||
worker.postMessage({ type: 'runCode', code })
|
||||
|
||||
socket.emit('codeOutput', transactionId, false, util.inspect(output, { stylize }))
|
||||
} catch (e) {
|
||||
socket.emit('codeOutput', transactionId, true, e.toString())
|
||||
}
|
||||
})()
|
||||
worker.on('message', ({ type, error, output }) => {
|
||||
if (type !== 'codeOutput') return
|
||||
|
||||
socket.emit('codeOutput', transactionId, error, output)
|
||||
})
|
||||
})
|
||||
|
||||
socket.on('reset', resetVM)
|
||||
socket.on('reset', reset)
|
||||
})
|
||||
|
||||
process.on('uncaughtException', (e) => {
|
||||
|
|
76
vm.js
Normal file
76
vm.js
Normal file
|
@ -0,0 +1,76 @@
|
|||
const { parentPort } = require('worker_threads')
|
||||
const { VM } = require('vm2')
|
||||
const randomstring = require('randomstring')
|
||||
const ChatMessage = require('prismarine-chat')('1.20.1')
|
||||
const mc = require('minecraft-protocol')
|
||||
const moment = require('moment-timezone')
|
||||
const crypto = require('crypto')
|
||||
const nbt = require('prismarine-nbt')
|
||||
const util = require('util')
|
||||
const { stylize } = require('./colors')
|
||||
|
||||
const BRIDGE_PREFIX = 'function:'
|
||||
|
||||
let proxy
|
||||
|
||||
let vm
|
||||
|
||||
const handler = {
|
||||
get (target, prop) {
|
||||
if (!target[prop]) throw new Error(`Function "${prop}" not available`)
|
||||
|
||||
return (...args) => target[prop](...args)
|
||||
}
|
||||
}
|
||||
|
||||
let functions
|
||||
|
||||
parentPort.on('message', (msg) => {
|
||||
switch (msg.type) {
|
||||
case 'setFunctions':
|
||||
const parsed = JSON.parse(msg.jsonArray)
|
||||
|
||||
functions = {}
|
||||
|
||||
for (const eachFuntion of parsed) {
|
||||
functions[eachFuntion] = (...args) => {
|
||||
parentPort.postMessage({ type: 'socketEmit', data: [BRIDGE_PREFIX + eachFuntion, ...args] })
|
||||
|
||||
// how do i make thjis work
|
||||
// return new Promise((resolve) => {
|
||||
// parentPort.postMessage({ type: 'socketOnce', data: [`functionOutput:${eachFuntion}`, (message, parseJSON) => {
|
||||
// if (parseJSON) resolve(JSON.parse(message))
|
||||
// else resolve(message)
|
||||
// }]})
|
||||
// })
|
||||
}
|
||||
}
|
||||
|
||||
proxy = new Proxy(functions, handler)
|
||||
|
||||
vm = new VM({
|
||||
timeout: 100,
|
||||
sandbox: {
|
||||
get bridge () { return proxy },
|
||||
randomstring,
|
||||
ChatMessage,
|
||||
mc,
|
||||
moment,
|
||||
crypto,
|
||||
nbt
|
||||
}
|
||||
})
|
||||
|
||||
break
|
||||
case 'runCode':
|
||||
try {
|
||||
const output = vm.run(msg.code)
|
||||
|
||||
parentPort.postMessage({ type: 'codeOutput', output: util.inspect(output, { stylize }), error: false })
|
||||
} catch (e) {
|
||||
parentPort.postMessage({ type: 'codeOutput', output: e.toString(), error: true })
|
||||
}
|
||||
|
||||
break
|
||||
}
|
||||
})
|
Loading…
Reference in a new issue