/* * Dungeons Guide - The most intelligent Hypixel Skyblock Dungeons Mod * Copyright (C) 2021 cyoung06 * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ package kr.syeyoung.dungeonsguide.stomp; import lombok.Getter; import org.java_websocket.client.WebSocketClient; import org.java_websocket.handshake.ServerHandshake; import org.java_websocket.server.DefaultSSLWebSocketServerFactory; import sun.security.ssl.SSLSocketFactoryImpl; import javax.net.ssl.*; import java.io.ByteArrayInputStream; import java.io.IOException; import java.net.Socket; import java.net.URI; import java.security.KeyManagementException; import java.security.KeyStore; import java.security.KeyStoreException; import java.security.NoSuchAlgorithmException; import java.security.cert.CertificateException; import java.security.cert.CertificateFactory; import java.security.cert.X509Certificate; import java.util.HashMap; import java.util.Map; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; public class StompClient extends WebSocketClient implements StompInterface { public StompClient(URI serverUri, final String token, CloseListener closeListener) throws Exception { super(serverUri); this.closeListener = closeListener; addHeader("Authorization", token); System.out.println("connecting websocket"); if (!connectBlocking()) { throw new RuntimeException("Can't connect to ws"); } System.out.println("connected, stomp handshake"); while(this.stompClientStatus == StompClientStatus.CONNECTING); System.out.println("fully connected"); } private final CloseListener closeListener; @Getter private volatile StompClientStatus stompClientStatus = StompClientStatus.CONNECTING; @Getter private StompPayload errorPayload; private ScheduledFuture heartbeat = null; private static final ScheduledExecutorService ex = Executors.newScheduledThreadPool(1); @Override public void onOpen(ServerHandshake handshakedata) { send(new StompPayload().method(StompHeader.CONNECT) .header("accept-version","1.2") .header("heart-beat", "30000,30000") .header("host",uri.getHost()).getBuilt() ); } @Override public void onMessage(String message) { try { StompPayload payload = StompPayload.parse(message); if (payload.method() == StompHeader.CONNECTED) { stompClientStatus = StompClientStatus.CONNECTED; String heartbeat = payload.headers().get("heart-beat"); if (heartbeat != null) { int sx = Integer.parseInt(heartbeat.split(",")[0]); int sy = Integer.parseInt(heartbeat.split(",")[1]); if (sy == 0) return; int heartbeatMS = Integer.max(30000, sy); this.heartbeat = ex.scheduleAtFixedRate(() -> { send("\n"); }, heartbeatMS-1000, heartbeatMS-1000, TimeUnit.MILLISECONDS); } } else if (payload.method() == StompHeader.ERROR) { errorPayload = payload; stompClientStatus = StompClientStatus.ERROR; this.close(); } else if (payload.method() == StompHeader.MESSAGE) { // mesage StompSubscription stompSubscription = stompSubscriptionMap.get(Integer.parseInt(payload.headers().get("subscription"))); try { stompSubscription.getStompMessageHandler().handle(this, payload); if (stompSubscription.getAckMode() != StompSubscription.AckMode.AUTO) { send(new StompPayload().method(StompHeader.ACK) .header("id",payload.headers().get("ack")).getBuilt() ); } } catch (Exception e) { e.printStackTrace(); if (stompSubscription.getAckMode() != StompSubscription.AckMode.AUTO) { send(new StompPayload().method(StompHeader.NACK) .header("id",payload.headers().get("ack")).getBuilt() ); } } } else if (payload.method() == StompHeader.RECEIPT) { String receipt_id = payload.headers().get("receipt-id"); StompPayload payload1 = receiptMap.remove(Integer.parseInt(receipt_id)); if (payload1.method() == StompHeader.DISCONNECT) { stompClientStatus = StompClientStatus.DISCONNECTED; close(); } } } catch (Exception e) { e.printStackTrace(); } } @Override public void onClose(int code, String reason, boolean remote) { if (heartbeat != null) heartbeat.cancel(true); closeListener.onClose(code, reason, remote); } @Override public void onError(Exception ex) { ex.printStackTrace(); } private final Map stompSubscriptionMap = new HashMap(); private final Map receiptMap = new HashMap(); private int idIncrement = 0; @Override public void send(StompPayload payload) { if (stompClientStatus != StompClientStatus.CONNECTED) throw new IllegalStateException("not connected"); payload.method(StompHeader.SEND); if (payload.headers().get("receipt") != null) receiptMap.put(Integer.parseInt(payload.headers().get("receipt")), payload); send(payload.getBuilt()); } @Override public void subscribe(StompSubscription stompSubscription) { if (stompClientStatus != StompClientStatus.CONNECTED) throw new IllegalStateException("not connected"); stompSubscription.setId(++idIncrement); send(new StompPayload().method(StompHeader.SUBSCRIBE) .header("id",String.valueOf(stompSubscription.getId())) .header("destination", stompSubscription.getDestination()) .header("ack", stompSubscription.getAckMode().getValue()).getBuilt() ); stompSubscriptionMap.put(stompSubscription.getId(), stompSubscription); } @Override public void unsubscribe(StompSubscription stompSubscription) { if (stompClientStatus != StompClientStatus.CONNECTED) throw new IllegalStateException("not connected"); send(new StompPayload().method(StompHeader.UNSUBSCRIBE) .header("id",String.valueOf(stompSubscription.getId())).getBuilt() ); stompSubscriptionMap.remove(stompSubscription.getId()); } @Override public void disconnect() { if (stompClientStatus != StompClientStatus.CONNECTED) throw new IllegalStateException("not connected"); StompPayload stompPayload; stompClientStatus =StompClientStatus.DISCONNECTING; send((stompPayload = new StompPayload().method(StompHeader.DISCONNECT) .header("receipt", String.valueOf(++idIncrement))) .getBuilt() ); receiptMap.put(idIncrement, stompPayload); } }