javastun

Simple STUN client in java


I have found several java STUN implementations

Java and Which Stun libraries i should use?

There is

See also: STUN, TURN, ICE library for Java

But it is jars with many classes. I wish to find something simple in form of single method or at least single small class. Like following python code.

https://github.com/jtriley/pystun/blob/develop/stun/init.py

Reasonable answer why STUN in Java is so huge is also acceptable.


Solution

  • Reasonable answer why STUN in Java is so huge is also acceptable.

    It's a reasonable question. 99% of what STUN is just a simple echo/response protocol for a client to self-discover the IP and port mapping as a result of NAT between it and the public internet. Having built a STUN library in C++, I have some insight.

    Let's think about what is required of a STUN library:

    So combining all this into a library that anyone can use for all the different purposes of STUN including mapped address discovery, NAT classification, and ICE negotiation, it starts to get big quick.

    You could easily just roll some socket code that hardcodes the bytes of a binding request and then some hacked up parsing to parse the response. That might meet your own needs, but a well established open source library would never be written this way.

    JSTUN is a good start. I've shared some interop and bug fixing code with the original author. He doesn't actively maintain it, but it's a good implementation of RFC 3489. I even hacked it up once to run on Android.

    To generate a STUN binding request in JSTUN.

    MessageHeader sendMH = new MessageHeader(MessageHeader.MessageHeaderType.BindingRequest);
    sendMH.generateTransactionID();
    
    // add an empty ChangeRequest attribute. Not required by the standard, but JSTUN server requires it
    ChangeRequest changeRequest = new ChangeRequest();
    sendMH.addMessageAttribute(changeRequest);
    
    byte[] data = sendMH.getBytes();
    
    // not shown - sending the message
    

    Then to parse the response back:

    byte [] receivedData = new byte[500];
    
    // not shown - socket code that receives the messages into receivedData
    receiveMH.parseAttributes(receivedData);
    MappedAddress ma = (MappedAddress) receiveMH.getMessageAttribute(MessageAttribute.MessageAttributeType.MappedAddress);    
    

    Then combine the above with some socket code. The best example of combining the above with socket code can be found in the DiscoveryTest.java source file. You really just need the code in the test1() method of this class.