In this post, we look at simple integration of Object Deliverer for UDP connectivity. This will be used to handle motion updates over UDP between UE and java server.
This is a free UE marketplace plugin and it can be found here: https://www.unrealengine.com/marketplace/en-US/product/objectdeliverer
data:image/s3,"s3://crabby-images/3656e/3656e40e760f477d470ebe20241884bd296e5405" alt="Object Deliverer plugin in UE"
I was testing this to communicate with my server using UDP comms.
Creating connection component
Many UDP libraries I see in marketplace come as an actor component, which I didn’t want.
This library can be integrated with an actor component if you like, or if you prefer like me, create a separate object entirely to handle the UDP comms.
Create a object blueprint like this:
data:image/s3,"s3://crabby-images/1210c/1210c19abca296b64276e681f1034915dd1daa48" alt="Create a object blueprint"
And inside, we will be configuring the sender and receiver, like this:
data:image/s3,"s3://crabby-images/86156/8615690426b7f5f76fa92a56811abe31fa402cde" alt="Creating a connection component object."
As you can see, I created a basic ‘Object’ class and I will use it to configure UDP communication for sending and receiving data.
I will instantiate it from Game Instance, meaning that it will always be live even as I change maps etc.
Here’s my Game Instance code which will create this object.
data:image/s3,"s3://crabby-images/5bfc5/5bfc580b3833d26be7628375a7b70743ddf66cd0" alt="Game instance which will create the connection component"
configure your address and port as you require, for testing mine was:
- localhost
- port: 9876
Configuring the UDP sender
This will be achieved in three steps:
data:image/s3,"s3://crabby-images/e2846/e2846c65d46559c2a82d4289fcdf56b69a26dbc6" alt="Create object deliverer manager"
data:image/s3,"s3://crabby-images/080e8/080e82ab525684beae520e2737e0132d3941d100" alt="configure deliverybox object sender"
data:image/s3,"s3://crabby-images/31c47/31c47978266d44d2742a3d77f43542fb1addda73" alt="start the server"
- First initialize the delivery manager component.
- Bind on connect/disconnect are not essential
- Next configure the Delivery box sender object.
- Important to create of correct type, I send through String.
- Start the delivery manager, linking to the sender component
- Configure host/port accordingly
- Set correct packet rule to your specs
I then simply have a function to send message from this blueprint:
data:image/s3,"s3://crabby-images/30690/30690f5853827d044b23ce9b65d20537a7dcf249" alt="Send UDP message from my server connector"
For context, this is how I end up sending these messages:
data:image/s3,"s3://crabby-images/b370e/b370e8b72911b4c79110c80e95b206e52fb6afea" alt="Calling send UDP message function"
For reference, I use this library to convert struct to json:
data:image/s3,"s3://crabby-images/bdde4/bdde4ed780bd8f99421c12214c7006956d1dd26b" alt="Struct to JSON library"
Configure UDP receiver
This is configured in a very similar fashion to the sender, it also has 3 steps.
data:image/s3,"s3://crabby-images/47df5/47df5ca2ee83c6201e772944a703313bc944b8e7" alt="Configure the receiver manager"
data:image/s3,"s3://crabby-images/fb308/fb3083c750cf4df42cc39074fac5bd4de45d7dfd" alt="configure the delivery box for receiver"
data:image/s3,"s3://crabby-images/bb7b4/bb7b47a9868602c898cc3433ff4dad33200cb8f0" alt="start the UDP receiver"
- configure the object deliverer receiver
- event bindings here are optional
- configure the delivery box receiver
- Specify the delivery box expected message, mine is String
- Bind the message received here – I will show example of the delegate
- Start the server
- Bind on a correct port – it cannot be the same port that’s used by the sender
- Specify the correct packet rule
Using delegates to pass messages
Inside this server connector object, I configured my UDP receiver and I now want to pass these messages to my components.
I created a connected component blueprint which is just a Actor Component blueprint which will be used as a Parent to all other components I wish to communicate via UDP.
data:image/s3,"s3://crabby-images/ade9a/ade9ac859b73e57d82fa75f8093e6fcd3842edc5" alt="Actor Component which will connect with UDP"
data:image/s3,"s3://crabby-images/98b5a/98b5a6002c744f62b5d2bc28f359ad46207249a3" alt="Check from parent class whether message can be processed"
data:image/s3,"s3://crabby-images/86832/868320199dc5930472bc8448fc6f383a0530b790" alt="Process message from actor component"
So from my other actor component blueprints, such as the example above, I mainly need to override the Process Message function and define the execution.
Obviously you can customize it as per your requirements, this is just how I routed it.
Server sample code
The entire repo can be found here:
The specific code lives inside the UDPServer.java
At the time of writing, I have the following key functions:
Constructor:
public UDPServer() {
Thread serverThread =
new Thread(
() -> {
try {
startServer();
} catch (Exception e) {
log.error(e.getMessage());
}
});
serverThread.start();
}
Server Listener:
private void startServer() {
try {
DatagramSocket socket = new DatagramSocket(UDP_PORT);
byte[] buffer = new byte[1024];
while (true) {
DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
socket.receive(packet);
SocketMessage message = reader.readValue(packet.getData(), SocketMessage.class);
String actorId = getActorId(message);
if (actorId == null) {
log.error("UDP message did not contain actor ID, {}", message);
continue;
}
socketProcessOutgoingService.processUDPMessage(message);
log.info("Message received! {}", message);
}
} catch (Exception e) {
e.printStackTrace();
startServer();
}
}
Server Send message:
public void send(SocketResponse message, InetAddress address) {
try {
byte[] data = writer.writeValueAsBytes(message);
DatagramPacket packet =
new DatagramPacket(
data, data.length, address, 5555);
DatagramSocket socket = new MulticastSocket();
socket.send(packet);
} catch (Exception e) {
e.printStackTrace();
}
}
Check out the repository if you’re interested in how things are wired more in-depth. The code will be changing as its being worked on.