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))<>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{ 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 };