ngx-minecraft/mcroute.js
2025-06-22 02:37:03 +05:00

277 lines
11 KiB
JavaScript

var where_v = "";
function mcproto(s) {
var bufs=[];
var eating=null;
var eatpr=0;
var cond=null;
var rr=null;
var lasted = false;
async function read_b(count,b) {
var arred=count!=null;
count=count??1;
var out = [];
if(b==null) {
while (count>0) {
if(eating!=null&&eatpr>=eating.length) {
eating = null;
eatpr = 0;
}
if(eating==null) {
if(bufs.length == 0) {
if (lasted) {
throw "ran out of data";
}
await (new Promise((a,r)=>(cond=a,rr=r,[][[]])));
}
eating=bufs.shift();
if(eating==null ) throw "Wtf!"
}
out.push(eating[eatpr++]);count--;
}
} else {
while (count>0) {
b.i=b.i??0;
var cc = b.buf[b.i++];count--;
if(cc==null) {
throw new Error("ran out of data (buf)");
}
out.push(cc);
}
}
return arred?out:out[0];
}
s.on("upstream",function(b,opt){
if (b.length > 0) {
bufs.push(b);
if(cond!=null) cond();
}
lasted = lasted||opt.last;
if (b.length==0 && lasted) {
rr("unexpected end");
}
})
var unreg =_=> s.off("upstream");
async function readVarInt(b) {
var Int = 0;
var power = 0;
var rr;
while(1) {
rr=await read_b(null,b);
Int+=(rr&(0x80-1))<<power;
if(!(rr>>7)) {
break;
}
power+=7;
if(power>32)
throw "varInt too big!";
}
return Int|0;
}
async function readPacket(b) {
var len = await readVarInt(b);
if (len > 2097151) {
throw "packet too big!";
}
var pack = await read_b(len,b);
pack = {buf:pack};
try {
pack.type = await readVarInt(pack);
} catch(e) {
s.error("bad packet! "+JSON.stringify(pack));
throw e;
}
return pack;
}
async function readString(b) {
var len = await readVarInt(b);
var text = await read_b(len,b);
text = (new TextDecoder())
.decode(new Uint8Array(text));
return text;
}
async function readShort(b) {
var p = await read_b(2,b);
return (p[0]<<8)+p[1];
}
return {
read_b, unreg, readVarInt, readPacket, readString, readShort
};
}
function makeVarInt(Int) {
var rr;
var out=[];
while (1) {
rr = Int&(0x80-1);
Int>>=7;
if(Int>0) {
out.push(rr|0x80);
} else {
out.push(rr);
break;
}
}
return out;
}
function makePacket(type,pack) {
var pack1 = [];
var ll=makeVarInt(type);
memcpy(makeVarInt(ll.length+pack.length),pack1)
memcpy(ll,pack1);
memcpy(pack,pack1);
return pack1;
}
function makeString(desc) {
var pack = [];
var str=[];
memcpy((new TextEncoder())
.encode(desc),str);
memcpy(makeVarInt(str.length),pack);
memcpy(str,pack);
return pack;
}
function makeNBTString(desc) {
var pack = [];
var str=[];
memcpy((new TextEncoder())
.encode(desc),str);
memcpy([8,(str.length>>8)&0xff,str.length&0xff],pack);
return pack;
}
function wellcome(desc,s) {
return {"version":{"name":"nginx/"+s.variables.nginx_version,"protocol": 65535},"description":{"text":desc},"players":{"max":1000000,"online":-1},
"favicon": "",};
}
function memcpy(from,to) {
for(var i=0; i<from.length; i++) {
to.push(from[i]);
}
}
var hout;
var stt;
function handler_out() {
return hout;
}
function makeHandler(desc) {
return function(s) {
(async function(){
var pack = [];
memcpy(makeString(JSON.stringify(wellcome(desc,s))),pack);
pack = makePacket(0,pack);
hout = Buffer.from(new Uint8Array(pack)).toString('base64');
s.error(hout);
s.done();
})().catch(e=>{
s.error(e.stack);
s.deny();
return Promise.reject(e);
});
}
}
var err404 = makeHandler("404 Not Found")
var err503 = makeHandler("503 Bad Gateway")
function ponger(s) {
var mm=mcproto(s);
var read_b = mm.read_b;
var unreg = mm.unreg;
var readVarInt = mm.readVarInt;
var readPacket = mm.readPacket;
var readString = mm.readString;
var readShort = mm.readShort;
(async function() {
await readPacket(); // discard hello;
var pack = await readPacket();
if(pack.type != 1) {
throw "ponger wants a ping request!";
}
var data = await read_b(8,pack);
pack = makePacket(1,data);
hout = Buffer.from(new Uint8Array(pack)).toString('base64');
s.done();
})().catch(e=>{
s.error(e.stack);
unreg();
s.deny();
return Promise.reject(e);
});
}
function disconner(s) {
(async function() {
var pack = makePacket(0,makeString(JSON.stringify({text:"404 Not Found\n---------------\nnginx/"+s.variables.nginx_version})));
hout = Buffer.from(new Uint8Array(pack)).toString('base64');
s.done();
})().catch(e=>{
s.error(e.stack);
unreg();
s.deny();
return Promise.reject(e);
});
}
function unb64(s) {
s.on("downstream",function(b){
s.send(Buffer.from(b.toString(),'base64'));
})
}
function preread(s) {
var mm=mcproto(s);
var read_b = mm.read_b;
var unreg = mm.unreg;
var readVarInt = mm.readVarInt;
var readPacket = mm.readPacket;
var readString = mm.readString;
var readShort = mm.readShort;
(async function() {
//await readVarInt(); // we do not care about packet length somehow
// ^ we do actually
var pack = await readPacket();
if(pack.type != 0) {
throw "not minecraft...";
}
await readVarInt(pack); // we do not care what version this is
where_v = (await readString(pack))
.match(/([^\0]*)\0*/)[1]; // Forge Modloader support
var port = (await readShort(pack));
var nstate = (await readVarInt(pack));
if(nstate != 1) {
stt="disconnect";
} else {
pack = await readPacket();
if(pack.type == 1) {
stt="pinger";
} else if (pack.type == 0) {
stt = "status";
}
}
unreg();
s.error("YES minecraft! "+where_v+" from "+s.variables.remote_addr);
s.done();
})().catch(e=>{
s.error(e.stack);
unreg();
s.deny();
return Promise.reject(e);
});
}
function where() {
return where_v;
}
function status_type() {
return stt;
}
export default { preread, where, err404, err503, disconner, ponger, handler_out, status_type, unb64 };