News:

Who uses forums anymore?

Main Menu

[Java] Buffer Problems (Again)

Started by Lance, July 17, 2008, 11:02:53 PM

Previous topic - Next topic

0 Members and 1 Guest are viewing this topic.

Lance

I have JavaOP2 receive packets (non-battle.net ones). I am currently modifying the core to meet my needs.

This works:

byte []packet = new byte[127];
                   input.read(packet);

but I have to specify the size which is completely inaccurate and it throws off my data by adding lots of 0x00's. I need a solution to this problem ;(

The traditional

int len1 = (input.read() & 0x000000FF);
                   int len2 = (input.read() & 0x000000FF) << 8;

int length = len1 | len2;

byte []packet = new byte[length];

for(int i = 0; i < packet.length; i++)
                       packet[i] = (byte)input.read();

results in my incoming packets not processing for some reason.

Ron said something about looping through each byte but I am still pretty new to java :p

Thanks :)
Quote from: Joe
[23:55:31] [william@enterprise ~/Documents/Programming/javaop2]$ svn commit -m 'Tried to fix StayConnected. Again.'
Committed revision 63.
StayConnected strikes back!

Camel

You should really be using a packet buffer for both incoming and outgoing packets. I already posted links to a detailed example of how I do it in my bot, so go check that out.

<Camel> i said what what
<Blaze> in the butt
<Camel> you want to do it in my butt?
<Blaze> in my butt
<Camel> let's do it in the butt
<Blaze> Okay!

Lance

Link to the links please? I searched through BNUBot support forums and your recent posts ;;
Quote from: Joe
[23:55:31] [william@enterprise ~/Documents/Programming/javaop2]$ svn commit -m 'Tried to fix StayConnected. Again.'
Committed revision 63.
StayConnected strikes back!

Camel

It's one of the top two or three threads in this forum, and it isn't this one.

<Camel> i said what what
<Blaze> in the butt
<Camel> you want to do it in my butt?
<Blaze> in my butt
<Camel> let's do it in the butt
<Blaze> Okay!

Lance

Quote from: Camel!
You can't use a Writer for binary data. Writers are to Streams as Strings are to byte arrays. You need to use byte arrays for binary data; if you Strings, you can be guaranteed that your code will work inconsistently in different environments.

Even if you use a Stream, Print* is still inappropriate for sockets (but not because it won't work). You should use a packet buffer to construct your packets and convert them to a byte[], and then write() that directly to the socket's OutputStream, thus making the write operation atomic, fault-resistant, and thread-safe.
That's it, right? :p

I have the Buffer class from JavaOP2 available but I have no idea how to use it for input streams :p
Quote from: Joe
[23:55:31] [william@enterprise ~/Documents/Programming/javaop2]$ svn commit -m 'Tried to fix StayConnected. Again.'
Committed revision 63.
StayConnected strikes back!

Camel

#5
Here's how I do it; I'm not familiar with the JavaOp structure, but I assume it's similar since this is a pretty standard approach to the problem.

First, create a decorated class for input/output that can read/write the data types you're using. These are the ones I wrote: BNetInputStream, and BNetOutputStream.

Then, create packet buffers that operate on these streams: BNCSPacket, BNCSPacketReader. You can think of the former as BNCSPacketWriter; I simply never got around to refactoring it.

Finally, tie it all together with a socket: BNCSConnection.

<Camel> i said what what
<Blaze> in the butt
<Camel> you want to do it in my butt?
<Blaze> in my butt
<Camel> let's do it in the butt
<Blaze> Okay!

Lance

That's alot of extra code ;;
There isn't a shorter way to do this?
Quote from: Joe
[23:55:31] [william@enterprise ~/Documents/Programming/javaop2]$ svn commit -m 'Tried to fix StayConnected. Again.'
Committed revision 63.
StayConnected strikes back!

Camel

#7
The whole point of those classes is so that you don't have to implement those methods inline, thus decreasing the size of your codebase overall by increasing reuse. Those BNet*Stream classes are extremely reusable; they're not bnet-specific, they're really just packet buffers. The BNCSPacket* classes are specifically built for BNCS, but can be easily modified for other binary protocols.

If you're referring to the BNCSConnection class, that's a complete implementation of BNCS, which necessitates hugeness. You will not need to use anything from there; I only provided it for reference as to how I use the other four classes. Reading and writing packets is extremely simple once you've built the boilerplate code; here's an excerpt from BNCSConnection:

p = new BNCSPacket(BNCSPacketId.SID_AUTH_INFO);
p.writeDWord(0); // Protocol ID (0)
p.writeDWord(PlatformIDs.PLATFORM_IX86); // Platform ID (IX86)
p.writeDWord(productID.getDword()); // Product ID
p.writeDWord(verByte); // Version byte
p.writeDWord(prodLang); // Product language
p.writeDWord(0); // Local IP
p.writeDWord(tzBias); // TZ bias
p.writeDWord(0x409); // Locale ID
p.writeDWord(0x409); // Language ID
p.writeNTString(loc.getISO3Country()); // Country abreviation
p.writeNTString(loc.getDisplayCountry()); // Country
p.SendPacket(bncsOutputStream);


Another example, this one reads SID_PING and replies with the same DWORD:

case SID_PING: {
BNCSPacket p = new BNCSPacket(BNCSPacketId.SID_PING);
p.writeDWord(is.readDWord());
p.SendPacket(bncsOutputStream);
break;
}

<Camel> i said what what
<Blaze> in the butt
<Camel> you want to do it in my butt?
<Blaze> in my butt
<Camel> let's do it in the butt
<Blaze> Okay!

Lance

I just want to get the raw byte data  :'(. Nothing more nothing less.
This isn't for BNet packets so, would a modified BNCSConnection excerpt still work properly?
Quote from: Joe
[23:55:31] [william@enterprise ~/Documents/Programming/javaop2]$ svn commit -m 'Tried to fix StayConnected. Again.'
Committed revision 63.
StayConnected strikes back!

Camel

The packet buffers exist because you can't determine how long the packet is going to be until you've finished writing it, yet the length word is at the beginning of the packet.

The decorated I/O streams are for reading/writing various types. The base InputStream and OutputStream provide methods for reading/writing a single byte already.

<Camel> i said what what
<Blaze> in the butt
<Camel> you want to do it in my butt?
<Blaze> in my butt
<Camel> let's do it in the butt
<Blaze> Okay!