parent
5874e8b47e
commit
b3752fd4ae
27 changed files with 504 additions and 306 deletions
|
@ -1,3 +0,0 @@
|
||||||
key=AAAAAAAAAAA=
|
|
||||||
name="Node Number One"
|
|
||||||
host=example.com
|
|
5
pom.xml
5
pom.xml
|
@ -15,11 +15,6 @@
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
|
||||||
<groupId>org.java-websocket</groupId>
|
|
||||||
<artifactId>Java-WebSocket</artifactId>
|
|
||||||
<version>1.5.7</version>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.json</groupId>
|
<groupId>org.json</groupId>
|
||||||
<artifactId>json</artifactId>
|
<artifactId>json</artifactId>
|
||||||
|
|
|
@ -1,45 +0,0 @@
|
||||||
package eu.m724.autopeerer.client;
|
|
||||||
|
|
||||||
public class ClientPrefs {
|
|
||||||
private static ClientPrefs INSTANCE;
|
|
||||||
|
|
||||||
private final boolean ipv4, restricted4, ipv6, restricted6;
|
|
||||||
|
|
||||||
private ClientPrefs( boolean ipv4, boolean restricted4, boolean ipv6, boolean restricted6) {
|
|
||||||
this.ipv4 = ipv4;
|
|
||||||
this.restricted4 = restricted4;
|
|
||||||
this.ipv6 = ipv6;
|
|
||||||
this.restricted6 = restricted6;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void init(ClientConfiguration configuration) {
|
|
||||||
var ipv4 = configuration.getString("stack.ipv4");
|
|
||||||
var ipv6 = configuration.getString("stack.ipv6");
|
|
||||||
|
|
||||||
INSTANCE = new ClientPrefs(
|
|
||||||
ipv4.equalsIgnoreCase("yes") || ipv4.equalsIgnoreCase("restricted"),
|
|
||||||
ipv4.equalsIgnoreCase("restricted"),
|
|
||||||
ipv6.equalsIgnoreCase("yes") || ipv6.equalsIgnoreCase("restricted"),
|
|
||||||
ipv6.equalsIgnoreCase("restricted")
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* */
|
|
||||||
|
|
||||||
public static boolean ipv4Supported() {
|
|
||||||
return INSTANCE.ipv4;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean ipv6Supported() {
|
|
||||||
return INSTANCE.ipv6;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean ipv4Restricted() {
|
|
||||||
return INSTANCE.restricted4;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean ipv6Restricted() {
|
|
||||||
return INSTANCE.restricted6;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,16 +1,18 @@
|
||||||
package eu.m724.autopeerer.client;
|
package eu.m724.autopeerer.client;
|
||||||
|
|
||||||
import eu.m724.autopeerer.client.bird.BirdLive;
|
import eu.m724.autopeerer.client.bird.BirdLive;
|
||||||
|
import eu.m724.autopeerer.client.connectivity.Connectivity;
|
||||||
import eu.m724.autopeerer.client.wireguard.WireGuardLive;
|
import eu.m724.autopeerer.client.wireguard.WireGuardLive;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
public class Main {
|
public class Main {
|
||||||
public static void main(String[] args) throws InterruptedException {
|
public static void main(String[] args) {
|
||||||
System.out.println("Hello world!");
|
|
||||||
|
|
||||||
var config = new ClientConfiguration();
|
var config = new ClientConfiguration();
|
||||||
try {
|
try {
|
||||||
config.load();
|
config.load();
|
||||||
|
@ -23,16 +25,34 @@ public class Main {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ClientPrefs.init(config);
|
System.out.print("Checking connectivity");
|
||||||
|
Connectivity.check();
|
||||||
|
|
||||||
URI serverUri = URI.create(config.getString("remote"));
|
var list = new ArrayList<String>();
|
||||||
|
|
||||||
|
if (Connectivity.ipv6()) list.add("IPv6");
|
||||||
|
if (Connectivity.ipv4()) list.add("IPv4");
|
||||||
|
|
||||||
|
if (!list.isEmpty()) {
|
||||||
|
System.out.println(": " + String.join(" + ", list));
|
||||||
|
} else {
|
||||||
|
System.err.println(": You have no internet");
|
||||||
|
}
|
||||||
|
|
||||||
|
var remote = config.getString("remote").split(":");
|
||||||
|
var serverAddress = new InetSocketAddress(remote[0], Integer.parseInt(remote[1]));
|
||||||
|
|
||||||
var wireGuardLive = new WireGuardLive(new File(config.getString("wireguard.directory")));
|
var wireGuardLive = new WireGuardLive(new File(config.getString("wireguard.directory")));
|
||||||
var birdLive = new BirdLive(new File(config.getString("bird.directory")));
|
var birdLive = new BirdLive(new File(config.getString("bird.directory")));
|
||||||
|
|
||||||
var packetHandler = new PacketHandler(wireGuardLive, birdLive, config.getString("link-local"));
|
var packetHandler = new PacketHandler(wireGuardLive, birdLive);
|
||||||
var client = new MyWebsocketClient(serverUri, packetHandler);
|
var client = new MyTcpClient(serverAddress, config.getString("key"), packetHandler);
|
||||||
|
|
||||||
|
try {
|
||||||
client.connect();
|
client.connect();
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("Error connecting");
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
64
src/main/java/eu/m724/autopeerer/client/MyTcpClient.java
Normal file
64
src/main/java/eu/m724/autopeerer/client/MyTcpClient.java
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
package eu.m724.autopeerer.client;
|
||||||
|
|
||||||
|
import eu.m724.autopeerer.common.packet.PacketReaderImpl;
|
||||||
|
import eu.m724.autopeerer.common.packet.PacketReaderRunnable;
|
||||||
|
import eu.m724.autopeerer.common.packet.Packets;
|
||||||
|
import eu.m724.autopeerer.common.packet.c2s.LoginPacket;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.Base64;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
|
||||||
|
public class MyTcpClient {
|
||||||
|
private final InetSocketAddress serverAddress;
|
||||||
|
private final byte[] key;
|
||||||
|
private final PacketHandler packetHandler;
|
||||||
|
|
||||||
|
private Socket socket;
|
||||||
|
|
||||||
|
public MyTcpClient(InetSocketAddress serverAddress, String key, PacketHandler packetHandler) {
|
||||||
|
this.serverAddress = serverAddress;
|
||||||
|
this.key = Base64.getDecoder().decode(key);
|
||||||
|
this.packetHandler = packetHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void connect() throws IOException {
|
||||||
|
var connectStart = System.nanoTime();
|
||||||
|
|
||||||
|
this.socket = new Socket(serverAddress.getAddress(), serverAddress.getPort());
|
||||||
|
|
||||||
|
double connectTime = (System.nanoTime() - connectStart) / 1000000.0;
|
||||||
|
System.out.printf("Connected in %.3f ms\n", connectTime);
|
||||||
|
|
||||||
|
packetHandler.outputStream = socket.getOutputStream();
|
||||||
|
Packets.send(new LoginPacket(key), socket.getOutputStream());
|
||||||
|
|
||||||
|
Executors.newSingleThreadExecutor().execute(new PacketReaderRunnable(socket, new PacketReaderImpl() {
|
||||||
|
@Override
|
||||||
|
public void onPacket(ByteBuffer byteBuffer) {
|
||||||
|
try {
|
||||||
|
packetHandler.handle(byteBuffer);
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("Error handling packet");
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onError(Exception exception) {
|
||||||
|
System.err.println("Connection error:");
|
||||||
|
exception.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onClose() {
|
||||||
|
System.out.println("Connection closed");
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
while (socket.isConnected()) { } // TODO
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,58 +0,0 @@
|
||||||
package eu.m724.autopeerer.client;
|
|
||||||
|
|
||||||
import eu.m724.autopeerer.common.packet.Packets;
|
|
||||||
import eu.m724.autopeerer.common.packet.c2s.LoginPacket;
|
|
||||||
import org.java_websocket.client.WebSocketClient;
|
|
||||||
import org.java_websocket.handshake.ServerHandshake;
|
|
||||||
|
|
||||||
import java.net.URI;
|
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
|
|
||||||
public class MyWebsocketClient extends WebSocketClient {
|
|
||||||
private final PacketHandler packetHandler;
|
|
||||||
private long connectStart;
|
|
||||||
|
|
||||||
public MyWebsocketClient(URI serverUri, PacketHandler packetHandler) {
|
|
||||||
super(serverUri);
|
|
||||||
this.packetHandler = packetHandler;
|
|
||||||
packetHandler.sender = this::send;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void connect() {
|
|
||||||
super.connect();
|
|
||||||
connectStart = System.nanoTime();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onOpen(ServerHandshake serverHandshake) {
|
|
||||||
double connectTime = (System.nanoTime() - connectStart) / 1000000.0;
|
|
||||||
System.out.printf("Connected in %.3f ms\n", connectTime);
|
|
||||||
|
|
||||||
Packets.send(new LoginPacket(new byte[8]), this::send);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onMessage(ByteBuffer bytes) {
|
|
||||||
packetHandler.handle(bytes);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onClose(int code, String reason, boolean remote) {
|
|
||||||
if (remote) {
|
|
||||||
System.out.println("Closed by remote: " + code + " " + reason);
|
|
||||||
} else {
|
|
||||||
System.out.println("Closed by client: " + code + " " + reason);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onError(Exception e) {
|
|
||||||
System.err.println("WS error: ");
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
|
|
||||||
// we don't do String
|
|
||||||
@Override
|
|
||||||
public void onMessage(String s) { }
|
|
||||||
}
|
|
|
@ -2,12 +2,14 @@ package eu.m724.autopeerer.client;
|
||||||
|
|
||||||
import eu.m724.autopeerer.client.bird.BirdLive;
|
import eu.m724.autopeerer.client.bird.BirdLive;
|
||||||
import eu.m724.autopeerer.client.bird.BirdSession;
|
import eu.m724.autopeerer.client.bird.BirdSession;
|
||||||
|
import eu.m724.autopeerer.client.connectivity.Connectivity;
|
||||||
import eu.m724.autopeerer.client.wireguard.WireGuardKeys;
|
import eu.m724.autopeerer.client.wireguard.WireGuardKeys;
|
||||||
import eu.m724.autopeerer.client.wireguard.WireGuardLive;
|
import eu.m724.autopeerer.client.wireguard.WireGuardLive;
|
||||||
import eu.m724.autopeerer.client.wireguard.WireGuardSession;
|
import eu.m724.autopeerer.client.wireguard.WireGuardSession;
|
||||||
import eu.m724.autopeerer.common.AddressTools;
|
import eu.m724.autopeerer.common.AddressTools;
|
||||||
import eu.m724.autopeerer.common.packet.Packet;
|
import eu.m724.autopeerer.common.packet.Packet;
|
||||||
import eu.m724.autopeerer.common.packet.Packets;
|
import eu.m724.autopeerer.common.packet.Packets;
|
||||||
|
import eu.m724.autopeerer.common.packet.s2c.LoginResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.s2c.PingRequestPacket;
|
import eu.m724.autopeerer.common.packet.s2c.PingRequestPacket;
|
||||||
import eu.m724.autopeerer.common.packet.s2c.SessionRequestPacket;
|
import eu.m724.autopeerer.common.packet.s2c.SessionRequestPacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
||||||
|
@ -15,6 +17,8 @@ import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
import java.io.BufferedReader;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.net.InetAddress;
|
||||||
import java.net.UnknownHostException;
|
import java.net.UnknownHostException;
|
||||||
import java.nio.BufferUnderflowException;
|
import java.nio.BufferUnderflowException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
@ -22,22 +26,22 @@ import java.nio.file.FileAlreadyExistsException;
|
||||||
import java.util.Base64;
|
import java.util.Base64;
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.concurrent.ThreadLocalRandom;
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
import java.util.function.Consumer;
|
|
||||||
|
|
||||||
public class PacketHandler {
|
public class PacketHandler {
|
||||||
Consumer<ByteBuffer> sender;
|
OutputStream outputStream;
|
||||||
|
private boolean loggedIn;
|
||||||
|
|
||||||
|
private InetAddress linkLocal;
|
||||||
|
|
||||||
private final WireGuardLive wireGuardLive;
|
private final WireGuardLive wireGuardLive;
|
||||||
private final BirdLive birdLive;
|
private final BirdLive birdLive;
|
||||||
private final String serverLinkLocal;
|
|
||||||
|
|
||||||
public PacketHandler(WireGuardLive wireGuardLive, BirdLive birdLive, String serverLinkLocal) {
|
public PacketHandler(WireGuardLive wireGuardLive, BirdLive birdLive) {
|
||||||
this.wireGuardLive = wireGuardLive;
|
this.wireGuardLive = wireGuardLive;
|
||||||
this.birdLive = birdLive;
|
this.birdLive = birdLive;
|
||||||
this.serverLinkLocal = serverLinkLocal;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void handle(ByteBuffer bytes) {
|
void handle(ByteBuffer bytes) throws IOException {
|
||||||
Packet<?> p;
|
Packet<?> p;
|
||||||
try {
|
try {
|
||||||
p = Packets.parseClient(bytes);
|
p = Packets.parseClient(bytes);
|
||||||
|
@ -55,11 +59,19 @@ public class PacketHandler {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (loggedIn) {
|
||||||
if (p instanceof PingRequestPacket packet) {
|
if (p instanceof PingRequestPacket packet) {
|
||||||
handlePingRequest(packet);
|
handlePingRequest(packet);
|
||||||
} else if (p instanceof SessionRequestPacket packet) {
|
} else if (p instanceof SessionRequestPacket packet) {
|
||||||
handleSessionRequest(packet);
|
handleSessionRequest(packet);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
if (p instanceof LoginResponsePacket packet) {
|
||||||
|
this.linkLocal = packet.linkLocal;
|
||||||
|
System.out.println("Logged in as " + packet.nodeId);
|
||||||
|
loggedIn = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handlePingRequest(PingRequestPacket packet) {
|
private void handlePingRequest(PingRequestPacket packet) {
|
||||||
|
@ -87,32 +99,33 @@ public class PacketHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
System.out.printf("Ping request #%d to %s - %s avg %.3f / mdev %.3f ms\n", packet.requestId, packet.target.getHostAddress(), status, average, meanDeviation);
|
||||||
|
Packets.send(new PingResponsePacket(packet.requestId, status, average, meanDeviation), outputStream);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
System.err.println("Error pinging");
|
System.err.println("Error executing ping request");
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
|
|
||||||
System.out.printf("Ping request #%d to %s - %s avg %.3f / mdev %.3f ms\n", packet.requestId, packet.target.getHostAddress(), status, average, meanDeviation);
|
|
||||||
Packets.send(new PingResponsePacket(packet.requestId, status, average, meanDeviation), sender);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleSessionRequest(SessionRequestPacket packet) {
|
private void handleSessionRequest(SessionRequestPacket packet) throws IOException {
|
||||||
// validate endpoint
|
// validate endpoint
|
||||||
var resolved = false;
|
var resolved = false;
|
||||||
try {
|
try {
|
||||||
if (ClientPrefs.ipv6Supported()) {
|
if (Connectivity.ipv6()) {
|
||||||
var res = AddressTools.resolve(packet.endpointHost, true);
|
var res = AddressTools.resolve(packet.endpointHost, true);
|
||||||
resolved = res != null;
|
resolved = res != null;
|
||||||
}
|
}
|
||||||
if (!resolved && ClientPrefs.ipv4Supported()) {
|
if (!resolved && Connectivity.ipv4()) {
|
||||||
var res = AddressTools.resolve(packet.endpointHost, true);
|
var res = AddressTools.resolve(packet.endpointHost, true);
|
||||||
resolved = res != null;
|
resolved = res != null;
|
||||||
}
|
}
|
||||||
} catch (UnknownHostException | AddressTools.MultipleRecordsException ignored) { }
|
} catch (UnknownHostException | AddressTools.MultipleRecordsException ignored) { }
|
||||||
|
|
||||||
if (!resolved) {
|
if (!resolved) {
|
||||||
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_RESOLVE, -1, null), sender);
|
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_RESOLVE, -1, null), outputStream);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,22 +137,43 @@ public class PacketHandler {
|
||||||
port = (int) (packet.asn % 10000);
|
port = (int) (packet.asn % 10000);
|
||||||
}
|
}
|
||||||
|
|
||||||
var wireGuardSession = new WireGuardSession(port, privateKey, serverLinkLocal, packet.linkLocal.getHostAddress(), packet.endpointHost + ":" + packet.endpointPort, packet.publicKey);
|
var wireGuardSession = new WireGuardSession(
|
||||||
var birdSession = new BirdSession(packet.asn, packet.linkLocal.getHostAddress());
|
port,
|
||||||
|
privateKey,
|
||||||
|
linkLocal.getHostAddress(),
|
||||||
|
packet.linkLocal.getHostAddress(),
|
||||||
|
packet.endpointHost + ":" + packet.endpointPort,
|
||||||
|
packet.publicKey
|
||||||
|
);
|
||||||
|
|
||||||
|
var birdSession = new BirdSession(
|
||||||
|
packet.asn,
|
||||||
|
packet.linkLocal.getHostAddress()
|
||||||
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
wireGuardLive.saveSession(packet.asn, wireGuardSession);
|
wireGuardLive.saveSession(packet.asn, wireGuardSession);
|
||||||
birdLive.saveSession(birdSession);
|
birdLive.saveSession(birdSession);
|
||||||
|
|
||||||
System.out.printf("Created session AS%d to %s\n", packet.asn, packet.endpointHost);
|
System.out.printf("Created session AS%d to %s\n", packet.asn, packet.endpointHost);
|
||||||
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.OK, wireGuardSession.listenPort(), publicKey), sender);
|
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.OK, wireGuardSession.listenPort(), publicKey), outputStream);
|
||||||
} catch (FileAlreadyExistsException e) {
|
} catch (FileAlreadyExistsException e) {
|
||||||
System.err.println("Tried to create a session which already exists: AS" + packet.asn);
|
System.err.println("Tried to create a session which already exists: AS" + packet.asn);
|
||||||
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_DUPLICATE, -1, null), sender);
|
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_DUPLICATE, -1, null), outputStream);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_OTHER, -1, null), sender);
|
Packets.send(new SessionResponsePacket(packet.asn, SessionResponsePacket.SessionResult.ERROR_OTHER, -1, null), outputStream);
|
||||||
System.err.println("Failed to save session");
|
|
||||||
throw new RuntimeException(e);
|
var ex = new RuntimeException("Failed to save session", e);
|
||||||
|
|
||||||
|
// clean up
|
||||||
|
try {
|
||||||
|
wireGuardLive.removeSession(packet.asn);
|
||||||
|
birdLive.removeSession(packet.asn);
|
||||||
|
} catch (Exception e1) {
|
||||||
|
ex = new RuntimeException("Failed to clean up after failing to save session", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw ex;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,11 @@ public class BirdLive {
|
||||||
Files.writeString(file.toPath(), session.config(), StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE);
|
Files.writeString(file.toPath(), session.config(), StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void removeSession(long asn) throws IOException {
|
||||||
|
File file = new File(configsPath, "ap_" + asn + ".conf");
|
||||||
|
Files.delete(file.toPath());
|
||||||
|
}
|
||||||
|
|
||||||
public BirdSession getSession(long asn) throws IOException {
|
public BirdSession getSession(long asn) throws IOException {
|
||||||
File file = new File(configsPath, "ap_" + asn + ".conf");
|
File file = new File(configsPath, "ap_" + asn + ".conf");
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
package eu.m724.autopeerer.client.connectivity;
|
||||||
|
|
||||||
|
import java.net.URI;
|
||||||
|
import java.net.http.HttpClient;
|
||||||
|
import java.net.http.HttpRequest;
|
||||||
|
import java.net.http.HttpResponse;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
public class Connectivity {
|
||||||
|
private static boolean ipv4, ipv6;
|
||||||
|
|
||||||
|
public static boolean ipv4() {
|
||||||
|
return ipv4;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean ipv6() {
|
||||||
|
return ipv6;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void check() {
|
||||||
|
var client = HttpClient.newHttpClient();
|
||||||
|
|
||||||
|
var future4 = composeRequest(client, false);
|
||||||
|
var future6 = composeRequest(client, true);
|
||||||
|
|
||||||
|
ipv4 = future4.join();
|
||||||
|
ipv6 = future6.join();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static CompletableFuture<Boolean> composeRequest(HttpClient client, boolean ipv6) {
|
||||||
|
var request = HttpRequest
|
||||||
|
.newBuilder(URI.create("https://ipv%d.google.com/generate_204".formatted(ipv6 ? 6 : 4)))
|
||||||
|
.timeout(Duration.ofSeconds(5))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
return client.sendAsync(request, HttpResponse.BodyHandlers.discarding()).handle((r, ex) -> ex == null);
|
||||||
|
}
|
||||||
|
}
|
|
@ -17,6 +17,11 @@ public class WireGuardLive {
|
||||||
Files.writeString(file.toPath(), session.config(), StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE);
|
Files.writeString(file.toPath(), session.config(), StandardOpenOption.CREATE_NEW, StandardOpenOption.WRITE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void removeSession(long asn) throws IOException {
|
||||||
|
File file = new File(configsPath, "ap_" + asn + ".conf");
|
||||||
|
Files.delete(file.toPath());
|
||||||
|
}
|
||||||
|
|
||||||
public WireGuardSession getSession(long asn) throws IOException {
|
public WireGuardSession getSession(long asn) throws IOException {
|
||||||
File file = new File(configsPath, "ap_" + asn + ".conf");
|
File file = new File(configsPath, "ap_" + asn + ".conf");
|
||||||
|
|
||||||
|
|
|
@ -17,10 +17,10 @@ public class AddressTools {
|
||||||
* @return whether it's IPv6 and link local
|
* @return whether it's IPv6 and link local
|
||||||
*/
|
*/
|
||||||
public static boolean isLinkLocal(InetAddress address) {
|
public static boolean isLinkLocal(InetAddress address) {
|
||||||
if (isIp(address, true)) return false;
|
if (!isIp(address, true)) return false;
|
||||||
var bytes = address.getAddress();
|
var bytes = address.getAddress();
|
||||||
|
|
||||||
return bytes[0] == -2 && bytes[1] >= 0 && bytes[1] <= 63;
|
return bytes[0] == -2 && bytes[1] <= -65;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
14
src/main/java/eu/m724/autopeerer/common/NodeProfile.java
Normal file
14
src/main/java/eu/m724/autopeerer/common/NodeProfile.java
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package eu.m724.autopeerer.common;
|
||||||
|
|
||||||
|
import java.net.Inet6Address;
|
||||||
|
|
||||||
|
public record NodeProfile(
|
||||||
|
String label,
|
||||||
|
String hostname,
|
||||||
|
Inet6Address linkLocal,
|
||||||
|
boolean supportsIpv4,
|
||||||
|
boolean supportsIpv6,
|
||||||
|
boolean restrictedIpv4,
|
||||||
|
boolean restrictedIpv6
|
||||||
|
) {
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
package eu.m724.autopeerer.common.packet;
|
||||||
|
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
|
public interface PacketReaderImpl {
|
||||||
|
void onPacket(ByteBuffer byteBuffer);
|
||||||
|
void onError(Exception exception);
|
||||||
|
void onClose();
|
||||||
|
}
|
|
@ -0,0 +1,38 @@
|
||||||
|
package eu.m724.autopeerer.common.packet;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
|
public class PacketReaderRunnable implements Runnable {
|
||||||
|
private final Socket socket;
|
||||||
|
private final PacketReaderImpl impl;
|
||||||
|
|
||||||
|
public PacketReaderRunnable(Socket socket, PacketReaderImpl impl) {
|
||||||
|
this.socket = socket;
|
||||||
|
this.impl = impl;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
while (socket.isConnected()) {
|
||||||
|
loop();
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
impl.onError(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
impl.onClose();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loop() throws IOException {
|
||||||
|
var is = socket.getInputStream();
|
||||||
|
|
||||||
|
var bs = is.readNBytes(2);
|
||||||
|
var length = bs[0] << 8 | bs[1];
|
||||||
|
|
||||||
|
var buffer = ByteBuffer.wrap(is.readNBytes(length)); // packets are small
|
||||||
|
impl.onPacket(buffer);
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,12 +7,14 @@ import eu.m724.autopeerer.common.packet.s2c.LoginResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
public class Packets {
|
public class Packets {
|
||||||
public static Packet<?> parseClient(ByteBuffer buffer) throws Exception {
|
public static Packet<?> parseClient(ByteBuffer buffer) throws Exception {
|
||||||
byte id = buffer.get();
|
var id = buffer.get();
|
||||||
|
|
||||||
Packet<?> packet = null;
|
Packet<?> packet = null;
|
||||||
|
|
||||||
|
@ -28,7 +30,7 @@ public class Packets {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Packet<?> parseServer(ByteBuffer buffer) throws Exception {
|
public static Packet<?> parseServer(ByteBuffer buffer) throws Exception {
|
||||||
byte id = buffer.get();
|
var id = buffer.get();
|
||||||
|
|
||||||
Packet<?> packet = null;
|
Packet<?> packet = null;
|
||||||
|
|
||||||
|
@ -47,7 +49,10 @@ public class Packets {
|
||||||
ByteBuffer packetBuffer = packet.serialize();
|
ByteBuffer packetBuffer = packet.serialize();
|
||||||
packetBuffer.rewind();
|
packetBuffer.rewind();
|
||||||
|
|
||||||
var bb = ByteBuffer.allocate(1 + packetBuffer.remaining());
|
var packetLength = 1 + packetBuffer.remaining();
|
||||||
|
|
||||||
|
var bb = ByteBuffer.allocate(2 + packetLength);
|
||||||
|
bb.putShort((short) packetLength);
|
||||||
bb.put(packet.getId());
|
bb.put(packet.getId());
|
||||||
bb.put(packetBuffer);
|
bb.put(packetBuffer);
|
||||||
bb.rewind();
|
bb.rewind();
|
||||||
|
@ -59,4 +64,8 @@ public class Packets {
|
||||||
sender.accept(compose(packet));
|
sender.accept(compose(packet));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void send(Packet<?> packet, OutputStream outputStream) throws IOException {
|
||||||
|
outputStream.write(compose(packet).array());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,11 @@
|
||||||
package eu.m724.autopeerer.common.packet.c2s;
|
package eu.m724.autopeerer.common.packet.c2s;
|
||||||
|
|
||||||
|
import eu.m724.autopeerer.common.NodeProfile;
|
||||||
import eu.m724.autopeerer.common.packet.Packet;
|
import eu.m724.autopeerer.common.packet.Packet;
|
||||||
|
|
||||||
|
import java.net.Inet6Address;
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.net.UnknownHostException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
public class LoginPacket implements Packet<LoginPacket> {
|
public class LoginPacket implements Packet<LoginPacket> {
|
||||||
|
@ -21,6 +25,7 @@ public class LoginPacket implements Packet<LoginPacket> {
|
||||||
public ByteBuffer serialize() {
|
public ByteBuffer serialize() {
|
||||||
var buffer = ByteBuffer.allocate(8);
|
var buffer = ByteBuffer.allocate(8);
|
||||||
buffer.put(key);
|
buffer.put(key);
|
||||||
|
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,15 +1,20 @@
|
||||||
package eu.m724.autopeerer.common.packet.s2c;
|
package eu.m724.autopeerer.common.packet.s2c;
|
||||||
|
|
||||||
|
import eu.m724.autopeerer.common.AddressTools;
|
||||||
import eu.m724.autopeerer.common.packet.Packet;
|
import eu.m724.autopeerer.common.packet.Packet;
|
||||||
|
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.net.UnknownHostException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
public class LoginResponsePacket implements Packet<LoginResponsePacket> {
|
public class LoginResponsePacket implements Packet<LoginResponsePacket> {
|
||||||
public final String nodeId;
|
public final String nodeId;
|
||||||
|
public final InetAddress linkLocal;
|
||||||
|
|
||||||
public LoginResponsePacket(String nodeId) {
|
public LoginResponsePacket(String nodeId, InetAddress linkLocal) {
|
||||||
this.nodeId = nodeId;
|
this.nodeId = nodeId;
|
||||||
|
this.linkLocal = linkLocal;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -20,9 +25,13 @@ public class LoginResponsePacket implements Packet<LoginResponsePacket> {
|
||||||
@Override
|
@Override
|
||||||
public ByteBuffer serialize() {
|
public ByteBuffer serialize() {
|
||||||
var nodeIdEncoded = nodeId.getBytes(StandardCharsets.UTF_8);
|
var nodeIdEncoded = nodeId.getBytes(StandardCharsets.UTF_8);
|
||||||
var buffer = ByteBuffer.allocate(1 + nodeIdEncoded.length);
|
var buffer = ByteBuffer.allocate(1 + nodeIdEncoded.length + 16);
|
||||||
|
|
||||||
buffer.put((byte) nodeIdEncoded.length);
|
buffer.put((byte) nodeIdEncoded.length);
|
||||||
buffer.put(nodeIdEncoded);
|
buffer.put(nodeIdEncoded);
|
||||||
|
|
||||||
|
buffer.put(linkLocal.getAddress());
|
||||||
|
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,6 +40,20 @@ public class LoginResponsePacket implements Packet<LoginResponsePacket> {
|
||||||
var nodeIdEncoded = new byte[size];
|
var nodeIdEncoded = new byte[size];
|
||||||
buffer.get(nodeIdEncoded);
|
buffer.get(nodeIdEncoded);
|
||||||
|
|
||||||
return new LoginResponsePacket(new String(nodeIdEncoded, StandardCharsets.UTF_8));
|
var ll = new byte[16];
|
||||||
|
buffer.get(ll);
|
||||||
|
InetAddress linkLocal;
|
||||||
|
try {
|
||||||
|
linkLocal = InetAddress.getByAddress(ll);
|
||||||
|
} catch (UnknownHostException e) {
|
||||||
|
linkLocal = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert AddressTools.isLinkLocal(linkLocal);
|
||||||
|
|
||||||
|
return new LoginResponsePacket(
|
||||||
|
new String(nodeIdEncoded, StandardCharsets.UTF_8),
|
||||||
|
linkLocal
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,67 +4,37 @@ import eu.m724.autopeerer.common.packet.Packet;
|
||||||
import eu.m724.autopeerer.common.packet.Packets;
|
import eu.m724.autopeerer.common.packet.Packets;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.s2c.PingRequestPacket;
|
|
||||||
import eu.m724.autopeerer.common.packet.s2c.SessionRequestPacket;
|
|
||||||
import org.java_websocket.WebSocket;
|
|
||||||
|
|
||||||
import java.net.InetAddress;
|
import java.io.IOException;
|
||||||
import java.nio.ByteBuffer;
|
import java.net.Socket;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.CompletableFuture;
|
|
||||||
import java.util.concurrent.ThreadLocalRandom;
|
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
public class ClientState {
|
public class ClientState {
|
||||||
public boolean authenticated = false;
|
public boolean authenticated = false;
|
||||||
public final int clientId;
|
public final int clientId;
|
||||||
public final WebSocket socket;
|
public final Socket socket;
|
||||||
Node node;
|
Node node;
|
||||||
|
|
||||||
private final Consumer<ByteBuffer> sender;
|
public ClientState(int clientId, Socket socket) {
|
||||||
|
|
||||||
public ClientState(int clientId, WebSocket socket) {
|
|
||||||
this.clientId = clientId;
|
this.clientId = clientId;
|
||||||
this.socket = socket;
|
this.socket = socket;
|
||||||
this.sender = socket::send;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void send(Packet<?> packet) {
|
void send(Packet<?> packet) throws IOException {
|
||||||
Packets.send(packet, sender);
|
Packets.send(packet, socket.getOutputStream());
|
||||||
}
|
}
|
||||||
|
|
||||||
void end() {
|
void end() throws IOException {
|
||||||
socket.close();
|
socket.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
CompletableFuture<PingResponsePacket> ping(InetAddress host) {
|
|
||||||
var future = new CompletableFuture<PingResponsePacket>();
|
|
||||||
|
|
||||||
var id = (short) (ThreadLocalRandom.current().nextInt() & 0xFFFF);
|
|
||||||
|
|
||||||
pingConsumers.put(id, future::complete);
|
|
||||||
send(new PingRequestPacket(id, host));
|
|
||||||
|
|
||||||
return future;
|
|
||||||
}
|
|
||||||
|
|
||||||
CompletableFuture<SessionResponsePacket> session(long asn, InetAddress linkLocal, String publicKey, String endpointHost, int endpointPort) {
|
|
||||||
var future = new CompletableFuture<SessionResponsePacket>();
|
|
||||||
|
|
||||||
sessionConsumers.put(asn, future::complete);
|
|
||||||
send(new SessionRequestPacket(
|
|
||||||
asn, linkLocal, publicKey, endpointHost, endpointPort
|
|
||||||
));
|
|
||||||
|
|
||||||
return future;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Packet functions */
|
/* Packet functions */
|
||||||
|
|
||||||
private final Map<Short, Consumer<PingResponsePacket>> pingConsumers = new HashMap<>();
|
final Map<Short, Consumer<PingResponsePacket>> pingConsumers = new HashMap<>();
|
||||||
private final Map<Long, Consumer<SessionResponsePacket>> sessionConsumers = new HashMap<>();
|
final Map<Long, Consumer<SessionResponsePacket>> sessionConsumers = new HashMap<>();
|
||||||
|
|
||||||
void onPacketReceived(Packet<?> p) {
|
void onPacketReceived(Packet<?> p) {
|
||||||
if (p instanceof PingResponsePacket packet) {
|
if (p instanceof PingResponsePacket packet) {
|
||||||
|
|
|
@ -31,7 +31,7 @@ public class Main {
|
||||||
|
|
||||||
var packetHandler = new PacketHandler(nodes);
|
var packetHandler = new PacketHandler(nodes);
|
||||||
|
|
||||||
var server = new MyWebsocketServer(
|
var server = new MyTcpServer(
|
||||||
new InetSocketAddress(config.getString("socket.address"), config.getInt("socket.port")),
|
new InetSocketAddress(config.getString("socket.address"), config.getInt("socket.port")),
|
||||||
packetHandler
|
packetHandler
|
||||||
);
|
);
|
||||||
|
|
|
@ -53,10 +53,14 @@ public class MyHttpHandler implements HttpHandler {
|
||||||
|
|
||||||
var json = new JSONObject();
|
var json = new JSONObject();
|
||||||
nodes.forEach(node -> {
|
nodes.forEach(node -> {
|
||||||
|
var prof = node.getNodeProfile();
|
||||||
var val = new JSONObject()
|
var val = new JSONObject()
|
||||||
.put("name", node.name())
|
.put("label", node.getNodeProfile().label())
|
||||||
.put("host", node.host())
|
.put("host", node.getNodeProfile().hostname())
|
||||||
.put("online", node.connected());
|
.put("linkLocal", node.getNodeProfile().linkLocal().getHostAddress())
|
||||||
|
.put("ipv4", prof.supportsIpv4() ? (prof.restrictedIpv4() ? "restricted" : true) : false)
|
||||||
|
.put("ipv6", prof.supportsIpv6() ? (prof.restrictedIpv6() ? "restricted" : true) : false)
|
||||||
|
.put("online", node.isConnected());
|
||||||
json.put(node.id(), val);
|
json.put(node.id(), val);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -88,7 +92,7 @@ public class MyHttpHandler implements HttpHandler {
|
||||||
var futures = new HashSet<CompletableFuture<Void>>();
|
var futures = new HashSet<CompletableFuture<Void>>();
|
||||||
|
|
||||||
for (Node node : selectedNodes) {
|
for (Node node : selectedNodes) {
|
||||||
if (!node.connected()) { // node is offline
|
if (!node.isConnected()) { // node is offline
|
||||||
var response = new JSONObject()
|
var response = new JSONObject()
|
||||||
.put("node", node.id())
|
.put("node", node.id())
|
||||||
.put("status", "OFFLINE");
|
.put("status", "OFFLINE");
|
||||||
|
@ -96,7 +100,7 @@ public class MyHttpHandler implements HttpHandler {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
var future = node.getClient().ping(target).handle((result, ex) -> {
|
var future = node.ping(target).handle((result, ex) -> {
|
||||||
try {
|
try {
|
||||||
if (ex != null) {
|
if (ex != null) {
|
||||||
sendResponse(exchange, 500);
|
sendResponse(exchange, 500);
|
||||||
|
@ -160,7 +164,7 @@ public class MyHttpHandler implements HttpHandler {
|
||||||
CompletableFuture<SessionResponsePacket> future = null;
|
CompletableFuture<SessionResponsePacket> future = null;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
future = node.getClient().session(
|
future = node.session(
|
||||||
json.getLong("asn"),
|
json.getLong("asn"),
|
||||||
InetAddress.getByName(json.getString("linkLocal")),
|
InetAddress.getByName(json.getString("linkLocal")),
|
||||||
json.getString("publicKey"),
|
json.getString("publicKey"),
|
||||||
|
|
84
src/main/java/eu/m724/autopeerer/server/MyTcpServer.java
Normal file
84
src/main/java/eu/m724/autopeerer/server/MyTcpServer.java
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
package eu.m724.autopeerer.server;
|
||||||
|
|
||||||
|
import eu.m724.autopeerer.common.packet.PacketReaderImpl;
|
||||||
|
import eu.m724.autopeerer.common.packet.PacketReaderRunnable;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.*;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
|
||||||
|
public class MyTcpServer {
|
||||||
|
private final InetSocketAddress address;
|
||||||
|
private final PacketHandler handler;
|
||||||
|
|
||||||
|
private ServerSocket socket;
|
||||||
|
private int id = 0;
|
||||||
|
|
||||||
|
private final Map<Integer, ClientState> states = new HashMap<>();
|
||||||
|
|
||||||
|
public MyTcpServer(InetSocketAddress address, PacketHandler packetHandler) throws IOException {
|
||||||
|
this.address = address;
|
||||||
|
this.handler = packetHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void start() throws IOException {
|
||||||
|
this.socket = new ServerSocket(address.getPort(), 50, address.getAddress());
|
||||||
|
System.out.printf("TCP server started on %s:%d\n", address.getAddress().getHostAddress(), address.getPort());
|
||||||
|
|
||||||
|
Executors.newSingleThreadExecutor().execute(() -> {
|
||||||
|
try {
|
||||||
|
loop();
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("Error in TCP loop");
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loop() throws IOException {
|
||||||
|
while (!socket.isClosed()) {
|
||||||
|
Socket client = socket.accept();
|
||||||
|
var id = ++this.id;
|
||||||
|
|
||||||
|
var state = new ClientState(id, client);
|
||||||
|
states.put(id, state);
|
||||||
|
System.out.printf("[%d] Connected: %s\n", id, client.getRemoteSocketAddress());
|
||||||
|
|
||||||
|
Executors.newSingleThreadExecutor().execute(
|
||||||
|
new PacketReaderRunnable(client, new PacketReaderImpl() {
|
||||||
|
@Override
|
||||||
|
public void onPacket(ByteBuffer byteBuffer) {
|
||||||
|
try {
|
||||||
|
handler.handle(state, byteBuffer);
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("Error handling packet");
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onError(Exception exception) {
|
||||||
|
// TODO
|
||||||
|
System.err.printf("[%d] Error\n", state.clientId);
|
||||||
|
exception.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onClose() {
|
||||||
|
var state = states.remove(id);
|
||||||
|
state.node.setClient(null);
|
||||||
|
var id = state.authenticated ? state.node.id() : client.getRemoteSocketAddress();
|
||||||
|
System.out.printf("[%d] Disconnected: %s\n", state.clientId, id);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void close() throws IOException {
|
||||||
|
this.socket.close();
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,59 +0,0 @@
|
||||||
package eu.m724.autopeerer.server;
|
|
||||||
|
|
||||||
import org.java_websocket.WebSocket;
|
|
||||||
import org.java_websocket.handshake.ClientHandshake;
|
|
||||||
import org.java_websocket.server.WebSocketServer;
|
|
||||||
|
|
||||||
import java.net.InetSocketAddress;
|
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
public class MyWebsocketServer extends WebSocketServer {
|
|
||||||
private final PacketHandler packetHandler;
|
|
||||||
|
|
||||||
private final Map<WebSocket, ClientState> states = new HashMap<>();
|
|
||||||
|
|
||||||
private int id = 0;
|
|
||||||
|
|
||||||
public MyWebsocketServer(InetSocketAddress address, PacketHandler packetHandler) {
|
|
||||||
super(address);
|
|
||||||
this.setReuseAddr(true);
|
|
||||||
|
|
||||||
this.packetHandler = packetHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onOpen(WebSocket conn, ClientHandshake handshake) {
|
|
||||||
var state = new ClientState(++id, conn);
|
|
||||||
states.put(conn, state);
|
|
||||||
System.out.printf("[%d] Connected: %s\n", id, conn.getRemoteSocketAddress().getHostString());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onClose(WebSocket conn, int code, String reason, boolean remote) {
|
|
||||||
var state = states.remove(conn);
|
|
||||||
state.node.setClient(null);
|
|
||||||
var id = state.authenticated ? state.node.id() : conn.getRemoteSocketAddress().getHostString();
|
|
||||||
System.out.printf("[%d] Disconnected: %s\n", state.clientId, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onMessage(WebSocket conn, ByteBuffer message) {
|
|
||||||
packetHandler.handle(states.get(conn), message);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onError(WebSocket conn, Exception ex) {
|
|
||||||
ex.printStackTrace();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onStart() {
|
|
||||||
System.out.printf("Websocket server started on %s:%d\n", getAddress().getHostString(), getPort());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onMessage(WebSocket conn, String message) {
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,54 +1,80 @@
|
||||||
package eu.m724.autopeerer.server;
|
package eu.m724.autopeerer.server;
|
||||||
|
|
||||||
|
import eu.m724.autopeerer.common.AddressTools;
|
||||||
|
import eu.m724.autopeerer.common.NodeProfile;
|
||||||
import eu.m724.autopeerer.common.packet.Packet;
|
import eu.m724.autopeerer.common.packet.Packet;
|
||||||
import eu.m724.autopeerer.common.packet.Packets;
|
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
||||||
|
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
||||||
|
import eu.m724.autopeerer.common.packet.s2c.PingRequestPacket;
|
||||||
|
import eu.m724.autopeerer.common.packet.s2c.SessionRequestPacket;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.Inet6Address;
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.net.UnknownHostException;
|
||||||
import java.util.Base64;
|
import java.util.Base64;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
|
|
||||||
public final class Node {
|
public final class Node {
|
||||||
private final String id;
|
private final String id;
|
||||||
private final byte[] key;
|
private final byte[] key;
|
||||||
private final String name;
|
|
||||||
private final String host;
|
private final NodeProfile nodeProfile;
|
||||||
|
|
||||||
private ClientState client;
|
private ClientState client;
|
||||||
|
|
||||||
public Node(
|
public Node(
|
||||||
String id,
|
String id,
|
||||||
byte[] key,
|
byte[] key,
|
||||||
String name,
|
NodeProfile nodeProfile
|
||||||
String host
|
|
||||||
) {
|
) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.key = key;
|
this.key = key;
|
||||||
this.name = name;
|
this.nodeProfile = nodeProfile;
|
||||||
this.host = host;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setClient(ClientState client) {
|
|
||||||
this.client = client;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ClientState getClient() {
|
ClientState getClient() {
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean connected() {
|
NodeProfile getNodeProfile() {
|
||||||
|
return nodeProfile;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setClient(ClientState client) {
|
||||||
|
this.client = client;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean isConnected() {
|
||||||
return client != null;
|
return client != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
void send(Packet<?> packet) {
|
void send(Packet<?> packet) throws IOException {
|
||||||
client.send(packet);
|
client.send(packet);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Node fromProperties(String id, Properties properties) {
|
static Node fromProperties(String id, Properties properties) throws UnknownHostException {
|
||||||
|
var ipv4 = properties.getProperty("stack.ipv4");
|
||||||
|
var ipv6 = properties.getProperty("stack.ipv6");
|
||||||
|
|
||||||
|
Inet6Address linkLocal = (Inet6Address) InetAddress.getByName(properties.getProperty("link-local"));
|
||||||
|
if (!AddressTools.isLinkLocal(linkLocal)) throw new UnknownHostException();
|
||||||
|
|
||||||
return new Node(
|
return new Node(
|
||||||
id,
|
id,
|
||||||
Base64.getDecoder().decode(properties.getProperty("key")),
|
Base64.getDecoder().decode(properties.getProperty("key")),
|
||||||
properties.getProperty("name"),
|
new NodeProfile(
|
||||||
properties.getProperty("host")
|
properties.getProperty("label"),
|
||||||
|
properties.getProperty("hostname"),
|
||||||
|
linkLocal,
|
||||||
|
ipv4.equalsIgnoreCase("yes") || ipv4.equalsIgnoreCase("restricted"),
|
||||||
|
ipv4.equalsIgnoreCase("restricted"),
|
||||||
|
ipv6.equalsIgnoreCase("yes") || ipv6.equalsIgnoreCase("restricted"),
|
||||||
|
ipv6.equalsIgnoreCase("restricted")
|
||||||
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,37 +86,48 @@ public final class Node {
|
||||||
return key;
|
return key;
|
||||||
}
|
}
|
||||||
|
|
||||||
public String name() {
|
|
||||||
return name;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String host() {
|
|
||||||
return host;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
if (obj == this) return true;
|
if (obj == this) return true;
|
||||||
if (obj == null || obj.getClass() != this.getClass()) return false;
|
if (obj == null || obj.getClass() != this.getClass()) return false;
|
||||||
var that = (Node) obj;
|
var that = (Node) obj;
|
||||||
return Objects.equals(this.id, that.id) &&
|
return Objects.equals(this.id, that.id) &&
|
||||||
Objects.equals(this.key, that.key) &&
|
Objects.equals(this.key, that.key);
|
||||||
Objects.equals(this.name, that.name) &&
|
|
||||||
Objects.equals(this.host, that.host);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return Objects.hash(id, key, name, host);
|
return Objects.hash(id, key);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "Node[" +
|
return "Node[id=" + id + "]";
|
||||||
"id=" + id + ", " +
|
}
|
||||||
"key=" + key + ", " +
|
|
||||||
"name=" + name + ", " +
|
/* */
|
||||||
"host=" + host + ']';
|
|
||||||
|
|
||||||
|
CompletableFuture<PingResponsePacket> ping(InetAddress host) throws IOException {
|
||||||
|
var future = new CompletableFuture<PingResponsePacket>();
|
||||||
|
|
||||||
|
var id = (short) (ThreadLocalRandom.current().nextInt() & 0xFFFF);
|
||||||
|
|
||||||
|
client.pingConsumers.put(id, future::complete);
|
||||||
|
send(new PingRequestPacket(id, host));
|
||||||
|
|
||||||
|
return future;
|
||||||
|
}
|
||||||
|
|
||||||
|
CompletableFuture<SessionResponsePacket> session(long asn, InetAddress linkLocal, String publicKey, String endpointHost, int endpointPort) throws IOException {
|
||||||
|
var future = new CompletableFuture<SessionResponsePacket>();
|
||||||
|
|
||||||
|
client.sessionConsumers.put(asn, future::complete);
|
||||||
|
send(new SessionRequestPacket(
|
||||||
|
asn, linkLocal, publicKey, endpointHost, endpointPort
|
||||||
|
));
|
||||||
|
|
||||||
|
return future;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,9 @@ import eu.m724.autopeerer.common.packet.Packets;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.LoginPacket;
|
import eu.m724.autopeerer.common.packet.c2s.LoginPacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.PingResponsePacket;
|
||||||
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
import eu.m724.autopeerer.common.packet.c2s.SessionResponsePacket;
|
||||||
|
import eu.m724.autopeerer.common.packet.s2c.LoginResponsePacket;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
import java.nio.BufferUnderflowException;
|
import java.nio.BufferUnderflowException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
@ -17,7 +19,7 @@ public class PacketHandler {
|
||||||
nodes.forEach(n -> this.nodes.put(n, null));
|
nodes.forEach(n -> this.nodes.put(n, null));
|
||||||
}
|
}
|
||||||
|
|
||||||
void handle(ClientState state, ByteBuffer bytes) {
|
void handle(ClientState state, ByteBuffer bytes) throws IOException {
|
||||||
// TODO this is not safe but enough for now
|
// TODO this is not safe but enough for now
|
||||||
Packet<?> p;
|
Packet<?> p;
|
||||||
try {
|
try {
|
||||||
|
@ -60,22 +62,24 @@ public class PacketHandler {
|
||||||
.ifPresent(e -> nodes.remove(e.getKey()));
|
.ifPresent(e -> nodes.remove(e.getKey()));
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleLogin(ClientState state, LoginPacket packet) {
|
private void handleLogin(ClientState state, LoginPacket packet) throws IOException {
|
||||||
var node = nodes.entrySet().stream().filter(e -> Arrays.equals(e.getKey().key(), packet.key)).findFirst().orElse(null);
|
var node = nodes.entrySet().stream().filter(e -> Arrays.equals(e.getKey().key(), packet.key)).findFirst().orElse(null);
|
||||||
if (node != null) {
|
if (node != null) {
|
||||||
if (node.getValue() != null) {
|
if (node.getValue() != null) {
|
||||||
if (node.getValue().socket.isOpen()) { // if there WAS a connection
|
if (node.getValue().socket.isConnected()) { // if there WAS a connection
|
||||||
System.out.printf("[%d] Tried to log in as %s, but is already logged in\n", state.clientId, node.getKey().id());
|
System.out.printf("[%d] Tried to log in as %s, but is already logged in\n", state.clientId, node.getKey().id());
|
||||||
state.end();
|
state.end();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
System.out.printf("[%d] Logged in as %s\n", state.clientId, node.getKey().id());
|
|
||||||
nodes.put(node.getKey(), state);
|
nodes.put(node.getKey(), state);
|
||||||
node.getKey().setClient(state);
|
node.getKey().setClient(state);
|
||||||
state.authenticated = true;
|
state.authenticated = true;
|
||||||
state.node = node.getKey();
|
state.node = node.getKey();
|
||||||
|
|
||||||
|
System.out.printf("[%d] Logged in as %s\n", state.clientId, node.getKey().id());
|
||||||
|
state.send(new LoginResponsePacket(node.getKey().id(), node.getKey().getNodeProfile().linkLocal()));
|
||||||
} else {
|
} else {
|
||||||
System.out.printf("[%d] Tried to log in with invalid key\n", state.clientId);
|
System.out.printf("[%d] Tried to log in with invalid key\n", state.clientId);
|
||||||
state.end();
|
state.end();
|
||||||
|
|
|
@ -7,6 +7,7 @@ import java.io.FileInputStream;
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
@ -21,9 +22,9 @@ public class ServerConfiguration extends Configuration {
|
||||||
|
|
||||||
if (!nd.exists()) {
|
if (!nd.exists()) {
|
||||||
nd.mkdir();
|
nd.mkdir();
|
||||||
var f = new File(nd, "node1.properties");
|
|
||||||
try (var is = getClass().getClassLoader().getResourceAsStream(f.getName())) {
|
try (var is = getClass().getClassLoader().getResourceAsStream("nodes/node1.properties")) {
|
||||||
Files.write(f.toPath(), is.readAllBytes());
|
Files.write(Path.of(nd.getPath(),"node1.properties"), is.readAllBytes());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,8 @@
|
||||||
# The server websocket
|
remote=127.0.0.1:8002
|
||||||
remote=ws://127.0.0.1:8002
|
key=AAAAAAAAAAA=
|
||||||
|
|
||||||
# Where are WG and BIRD configs located
|
# Where are WG and BIRD configs located
|
||||||
wireguard.directory=config/wg
|
wireguard.directory=config/wg
|
||||||
bird.directory=config/bird
|
bird.directory=config/bird
|
||||||
|
|
||||||
# Link local used for peering
|
# Other stuff is configured on the server
|
||||||
link-local=fe80::129:0
|
|
||||||
# Is IPv4 available? Yes, no, restricted
|
|
||||||
stack.ipv4=yes
|
|
||||||
# Is IPv6 available? Yes, no, restricted
|
|
||||||
stack.ipv6=yes
|
|
|
@ -1,3 +1,10 @@
|
||||||
key=AAAAAAAAAAA=
|
key=AAAAAAAAAAA=
|
||||||
name="Node Number One"
|
label="Node Number One"
|
||||||
host=example.com
|
# The hostname used as WireGuard endpoint
|
||||||
|
hostname=example.com
|
||||||
|
# Link local used for peering
|
||||||
|
link-local=fe80::129:0
|
||||||
|
# Is IPv4 available? Yes, no, restricted
|
||||||
|
stack.ipv4=yes
|
||||||
|
# Is IPv6 available? Yes, no, restricted
|
||||||
|
stack.ipv6=yes
|
Loading…
Reference in a new issue