5 changed files with 176 additions and 1 deletions
@ -0,0 +1,79 @@ |
|||
package app.controllers.debug; |
|||
|
|||
import org.springframework.web.bind.annotation.GetMapping; |
|||
import org.springframework.web.bind.annotation.RequestMapping; |
|||
import org.springframework.web.bind.annotation.RestController; |
|||
|
|||
@RestController |
|||
@RequestMapping("/debug/ws") |
|||
public class WebSocketPage { |
|||
|
|||
@GetMapping |
|||
public String getPage(){ |
|||
return "<!DOCTYPE html>\n" + |
|||
"<html lang=\"en\">\n" + |
|||
"<head>\n" + |
|||
" <meta charset=\"UTF-8\">\n" + |
|||
" <title>Home page</title>\n" + |
|||
" <link href=\"https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.3.1/semantic.min.css\"\n" + |
|||
" rel=\"stylesheet\">\n" + |
|||
"</head>\n" + |
|||
"<body>\n" + |
|||
"\n" + |
|||
"<div class=\"ui container\">\n" + |
|||
"\n" + |
|||
" <h1>Debug websocket page</h1>\n" + |
|||
"\n" + |
|||
" <div class=\"two column grid\">\n" + |
|||
" <div class=\"row\">\n" + |
|||
" <div class=\"column\">\n" + |
|||
" <label for=\"myMessage\">Message</label>\n" + |
|||
" </div>\n" + |
|||
"\n" + |
|||
" <div class=\"column\">\n" + |
|||
" <div class=\"ui input\">\n" + |
|||
" <input type=\"text\" id=\"myMessage\">\n" + |
|||
" </div>\n" + |
|||
" </div>\n" + |
|||
" </div>\n" + |
|||
"\n" + |
|||
" <div class=\"row\">\n" + |
|||
" <div class=\"column\">\n" + |
|||
" <label for=\"output\">Response from Server</label>\n" + |
|||
" </div>\n" + |
|||
"\n" + |
|||
" <div class=\"column\">\n" + |
|||
" <textarea rows=\"8\" cols=\"50\" id=\"output\" readonly=\"readonly\"></textarea>\n" + |
|||
" </div>\n" + |
|||
" </div>\n" + |
|||
"\n" + |
|||
" <div class=\"row\">\n" + |
|||
" <button class=\"ui button\" onclick=\"send()\">Send</button>\n" + |
|||
" </div>\n" + |
|||
"\n" + |
|||
" </div>\n" + |
|||
"</div>\n" + |
|||
"\n" + |
|||
"\n" + |
|||
"<script>\n" + |
|||
" const socketConn = new WebSocket('ws://localhost:8080/ws');\n" + |
|||
"\n" + |
|||
" function send() {\n" + |
|||
" const clientMsg = document.getElementById('myMessage');\n" + |
|||
"\n" + |
|||
" if (clientMsg.value) {\n" + |
|||
" socketConn.send(clientMsg.value);\n" + |
|||
" }\n" + |
|||
" }\n" + |
|||
"\n" + |
|||
" socketConn.onmessage = (e) => {\n" + |
|||
"\n" + |
|||
" const output = document.getElementById('output');\n" + |
|||
"\n" + |
|||
" output.value += `${e.data}\\n`;\n" + |
|||
" }\n" + |
|||
"</script>\n" + |
|||
"</body>\n" + |
|||
"</html>"; |
|||
} |
|||
} |
@ -0,0 +1,22 @@ |
|||
package app.websocket; |
|||
|
|||
import org.springframework.context.annotation.Configuration; |
|||
import org.springframework.web.socket.config.annotation.EnableWebSocket; |
|||
import org.springframework.web.socket.config.annotation.WebSocketConfigurer; |
|||
import org.springframework.web.socket.config.annotation.WebSocketHandlerRegistry; |
|||
|
|||
@Configuration |
|||
@EnableWebSocket |
|||
public class WebSocketConfig implements WebSocketConfigurer { |
|||
|
|||
private WebSocketHandler webSocketHandler; |
|||
|
|||
public WebSocketConfig(WebSocketHandler webSocketHandler) { |
|||
this.webSocketHandler = webSocketHandler; |
|||
} |
|||
|
|||
@Override |
|||
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) { |
|||
registry.addHandler(webSocketHandler, "/ws"); |
|||
} |
|||
} |
@ -0,0 +1,64 @@ |
|||
package app.websocket; |
|||
|
|||
import app.entities.server.Server; |
|||
import app.updates.BaseUpdater; |
|||
import com.fasterxml.jackson.core.JsonProcessingException; |
|||
import com.fasterxml.jackson.databind.ObjectMapper; |
|||
import org.slf4j.Logger; |
|||
import org.slf4j.LoggerFactory; |
|||
import org.springframework.context.annotation.Scope; |
|||
import org.springframework.stereotype.Component; |
|||
import org.springframework.web.socket.CloseStatus; |
|||
import org.springframework.web.socket.TextMessage; |
|||
import org.springframework.web.socket.WebSocketSession; |
|||
import org.springframework.web.socket.handler.TextWebSocketHandler; |
|||
|
|||
import java.io.IOException; |
|||
import java.util.HashMap; |
|||
import java.util.Map; |
|||
|
|||
@Component |
|||
@Scope(value = "singleton") |
|||
public class WebSocketHandler extends TextWebSocketHandler { |
|||
private final Logger logger = LoggerFactory.getLogger(BaseUpdater.class); |
|||
private final Map<String, WebSocketSession> activeSessions = new HashMap<>(); |
|||
private ObjectMapper objectMapper = new ObjectMapper(); |
|||
|
|||
@Override |
|||
public void afterConnectionEstablished(WebSocketSession session) throws Exception { |
|||
logger.info("Session {} open", session.getId()); |
|||
activeSessions.put(session.getId(), session); |
|||
super.afterConnectionEstablished(session); |
|||
} |
|||
|
|||
@Override |
|||
public void handleTextMessage(WebSocketSession session, TextMessage message) { |
|||
//nothing
|
|||
} |
|||
|
|||
@Override |
|||
public void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception { |
|||
logger.info("Session {} close", session.getId()); |
|||
activeSessions.remove(session.getId()); |
|||
super.afterConnectionClosed(session, status); |
|||
} |
|||
|
|||
private void push2All(String msg) throws IOException { |
|||
for (Map.Entry<String, WebSocketSession> otherSession : activeSessions.entrySet()) { |
|||
otherSession.getValue().sendMessage(new TextMessage(msg)); |
|||
} |
|||
} |
|||
|
|||
public void pushServer(String srv, Server server) { |
|||
try { |
|||
push2All( |
|||
objectMapper.writeValueAsString( |
|||
new HashMap<String, HashMap>() {{ |
|||
put("server", new HashMap<String, Server>(){{ |
|||
put(srv, server);}}); |
|||
}})); |
|||
} catch (IOException err) { |
|||
logger.error("Cannot send message"); |
|||
} |
|||
} |
|||
} |
Loading…
Reference in new issue