Skip to content

Commit

Permalink
Merge pull request #1442 from TooTallNate/socket-activation
Browse files Browse the repository at this point in the history
Socket activation
  • Loading branch information
marci4 authored Dec 8, 2024
2 parents 3b29042 + e5253dc commit 2b8dd96
Show file tree
Hide file tree
Showing 4 changed files with 161 additions and 2 deletions.
102 changes: 102 additions & 0 deletions src/main/example/SocketActivation.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
/*
* Copyright (c) 2010-2020 Nathan Rajlich
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.util.Collections;
import java.util.concurrent.atomic.AtomicInteger;
import org.java_websocket.WebSocket;
import org.java_websocket.drafts.Draft;
import org.java_websocket.drafts.Draft_6455;
import org.java_websocket.handshake.ClientHandshake;
import org.java_websocket.server.WebSocketServer;

/**
* This is a "smart" chat server which will exit when no more clients are left, in order to demonstrate socket activation
*/
public class SocketActivation extends WebSocketServer {

AtomicInteger clients = new AtomicInteger(0);

public SocketActivation(ServerSocketChannel chan) {
super(chan);
}

@Override
public void onOpen(WebSocket conn, ClientHandshake handshake) {
conn.send("Welcome to the server!"); //This method sends a message to the new client
broadcast("new connection: " + handshake.getResourceDescriptor()); //This method sends a message to all clients connected
if(clients.get() == 0) {
broadcast("You are the first client to join");
}
System.out.println(conn.getRemoteSocketAddress().getAddress().getHostAddress() + " entered the room!");
clients.incrementAndGet();
}

@Override
public void onClose(WebSocket conn, int code, String reason, boolean remote) {
broadcast(conn + " has left the room!");
System.out.println(conn + " has left the room!");
if(clients.decrementAndGet() <= 0) {
System.out.println("No more clients left, exiting");
System.exit(0);
}
}

@Override
public void onMessage(WebSocket conn, String message) {
broadcast(message);
System.out.println(conn + ": " + message);
}

@Override
public void onMessage(WebSocket conn, ByteBuffer message) {
broadcast(message.array());
System.out.println(conn + ": " + message);
}


public static void main(String[] args) throws InterruptedException, IOException {
if(System.inheritedChannel() == null) {
System.err.println("System.inheritedChannel() is null, make sure this program is started with file descriptor zero being a listening socket");
System.exit(1);
}
SocketActivation s = new SocketActivation((ServerSocketChannel)System.inheritedChannel());
s.start();
System.out.println(">>>> SocketActivation started on port: " + s.getPort() + " <<<<");
}

@Override
public void onError(WebSocket conn, Exception ex) {
ex.printStackTrace();
}

@Override
public void onStart() {
System.out.println("Server started!");
}

}
17 changes: 17 additions & 0 deletions src/main/example/jws-activation.service
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
[Unit]
Description=Java-WebSocket systemd activation demo service
After=network.target jws-activation.socket
Requires=jws-activation.socket

[Service]
Type=simple
# Place the command for running SocketActivation.java in file "$HOME"/jws_activation_command:
ExecStart=/bin/sh %h/jws_activation_run
TimeoutStopSec=5
StandardError=journal
StandardOutput=journal
# This is very important - systemd will pass the socket as file descriptor zero, which is what Java expects
StandardInput=socket

[Install]
WantedBy=default.target
9 changes: 9 additions & 0 deletions src/main/example/jws-activation.socket
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
[Unit]
Description=Java-WebSocket systemd activation demo socket
PartOf=jws-activation.service

[Socket]
ListenStream=127.0.0.1:9999

[Install]
WantedBy=sockets.target
35 changes: 33 additions & 2 deletions src/main/java/org/java_websocket/server/WebSocketServer.java
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,30 @@ public WebSocketServer(InetSocketAddress address, int decodercount, List<Draft>
this(address, decodercount, drafts, new HashSet<WebSocket>());
}

// Small internal helper function to get around limitations of Java constructors.
private static InetSocketAddress checkAddressOfExistingChannel(ServerSocketChannel existingChannel) {
assert existingChannel.isOpen();
SocketAddress addr;
try {
addr = existingChannel.getLocalAddress();
} catch (IOException e) {
throw new IllegalArgumentException("Could not get address of channel passed to WebSocketServer, make sure it is bound", e);
}
if (addr == null) {
throw new IllegalArgumentException("Could not get address of channel passed to WebSocketServer, make sure it is bound");
}
return (InetSocketAddress)addr;
}

/**
* @param existingChannel An already open and bound server socket channel, which this server will use.
* For example, it can be System.inheritedChannel() to implement socket activation.
*/
public WebSocketServer(ServerSocketChannel existingChannel) {
this(checkAddressOfExistingChannel(existingChannel));
this.server = existingChannel;
}

/**
* Creates a WebSocketServer that will attempt to bind/listen on the given <var>address</var>, and
* comply with <code>Draft</code> version <var>draft</var>.
Expand Down Expand Up @@ -575,15 +599,22 @@ private void doWrite(SelectionKey key) throws WrappedIOException {
private boolean doSetupSelectorAndServerThread() {
selectorthread.setName("WebSocketSelector-" + selectorthread.getId());
try {
server = ServerSocketChannel.open();
if (server == null) {
server = ServerSocketChannel.open();
// If 'server' is not null, that means WebSocketServer was created from existing channel.
}
server.configureBlocking(false);
ServerSocket socket = server.socket();
int receiveBufferSize = getReceiveBufferSize();
if (receiveBufferSize > 0) {
socket.setReceiveBufferSize(receiveBufferSize);
}
socket.setReuseAddress(isReuseAddr());
socket.bind(address, getMaxPendingConnections());
// Socket may be already bound, if an existing channel was passed to constructor.
// In this case we cannot modify backlog size from pure Java code, so leave it as is.
if (!socket.isBound()) {
socket.bind(address, getMaxPendingConnections());
}
selector = Selector.open();
server.register(selector, server.validOps());
startConnectionLostTimer();
Expand Down

0 comments on commit 2b8dd96

Please sign in to comment.