10 changed files with 193 additions and 161 deletions
@ -0,0 +1,5 @@ |
|||||
|
package app.websocket; |
||||
|
|
||||
|
public interface BaseWebsocketHandler { |
||||
|
String getPath(); |
||||
|
} |
@ -1,113 +0,0 @@ |
|||||
package app.websocket; |
|
||||
|
|
||||
import app.entities.VipGiveMethod; |
|
||||
import app.services.db.VIPService; |
|
||||
import app.utils.SaltedCookie; |
|
||||
import app.utils.SteamIDConverter; |
|
||||
import com.fasterxml.jackson.databind.ObjectMapper; |
|
||||
import lombok.Data; |
|
||||
import lombok.NoArgsConstructor; |
|
||||
import org.slf4j.Logger; |
|
||||
import org.slf4j.LoggerFactory; |
|
||||
import org.springframework.beans.factory.annotation.Autowired; |
|
||||
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.time.Instant; |
|
||||
import java.util.HashSet; |
|
||||
|
|
||||
@Component |
|
||||
@Scope(value = "singleton") |
|
||||
public class IntegrationWebSockerHandler extends TextWebSocketHandler { |
|
||||
private final Logger logger = LoggerFactory.getLogger(getClass()); |
|
||||
private ObjectMapper objectMapper = new ObjectMapper(); |
|
||||
private SaltedCookie saltedCookie; |
|
||||
private VIPService vipService; |
|
||||
private HashSet<String> unique_set; |
|
||||
|
|
||||
@Autowired |
|
||||
public IntegrationWebSockerHandler(SaltedCookie saltedCookie, VIPService vipService) { |
|
||||
super(); |
|
||||
this.saltedCookie = saltedCookie; |
|
||||
this.vipService = vipService; |
|
||||
this.unique_set = new HashSet(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void afterConnectionEstablished(WebSocketSession session) throws Exception { |
|
||||
logger.info("Session {} open", session.getId()); |
|
||||
super.afterConnectionEstablished(session); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void handleTextMessage(WebSocketSession session, TextMessage message) throws IOException { |
|
||||
IntegrationPayload payload = objectMapper.readValue(message.getPayload(), IntegrationPayload.class); |
|
||||
if (payload.test) { |
|
||||
session.sendMessage(new TextMessage("%d".formatted(Instant.now().getEpochSecond()))); |
|
||||
return; |
|
||||
} |
|
||||
if (payload.isVip()) { |
|
||||
if (!saltedCookie.ValidateSecretKey(payload.getSecret_key())) { |
|
||||
logger.warn("Cannot add vip without secret key"); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
if (!payload.vip.isFilled()) { |
|
||||
logger.warn("Can try add VIP but, payload is not full filled!"); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
if (!payload.vip.unique.isEmpty()) { |
|
||||
if (unique_set.contains(payload.vip.unique)) { |
|
||||
logger.warn("VIP uniq payload value already exist! " + payload.vip.unique); |
|
||||
return; |
|
||||
} else unique_set.add(payload.vip.unique); |
|
||||
} |
|
||||
|
|
||||
Integer result = vipService.addVIP( |
|
||||
SteamIDConverter.getSteamID(payload.vip.getSteam()), |
|
||||
payload.vip.getAmount(), |
|
||||
VipGiveMethod.valueOf(payload.vip.service.toUpperCase()), |
|
||||
payload.vip.getExtra()); |
|
||||
logger.info("Success add VIP to {}, result: {}", payload.vip.getSteam(), result); |
|
||||
return; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception { |
|
||||
logger.info("Session {} close", session.getId()); |
|
||||
super.afterConnectionClosed(session, status); |
|
||||
} |
|
||||
|
|
||||
@Data |
|
||||
@NoArgsConstructor |
|
||||
public static class IntegrationPayload { |
|
||||
String secret_key = null; |
|
||||
ExternalVIP vip = null; |
|
||||
Boolean test = false; |
|
||||
|
|
||||
public boolean isVip() { |
|
||||
return vip != null && secret_key != null; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Data |
|
||||
@NoArgsConstructor |
|
||||
public static class ExternalVIP { |
|
||||
String steam; |
|
||||
Integer amount; |
|
||||
String service; |
|
||||
String extra = ""; |
|
||||
String unique = ""; |
|
||||
|
|
||||
public boolean isFilled() { |
|
||||
return steam != null && amount != null && service != null; |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,28 +0,0 @@ |
|||||
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; |
|
||||
private IntegrationWebSockerHandler integrationWebSockerHandler; |
|
||||
|
|
||||
public WebSocketConfig( |
|
||||
WebSocketHandler webSocketHandler, |
|
||||
IntegrationWebSockerHandler integrationWebSockerHandler |
|
||||
) { |
|
||||
this.webSocketHandler = webSocketHandler; |
|
||||
this.integrationWebSockerHandler = integrationWebSockerHandler; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) { |
|
||||
registry.addHandler(webSocketHandler, "/ws"); |
|
||||
registry.addHandler(integrationWebSockerHandler, "/ws_integrations"); |
|
||||
} |
|
||||
} |
|
@ -0,0 +1,30 @@ |
|||||
|
package app.websocket.configuration; |
||||
|
|
||||
|
import app.websocket.BaseWebsocketHandler; |
||||
|
import org.slf4j.Logger; |
||||
|
import org.slf4j.LoggerFactory; |
||||
|
import org.springframework.beans.factory.annotation.Autowired; |
||||
|
import org.springframework.context.annotation.Configuration; |
||||
|
import org.springframework.web.socket.WebSocketHandler; |
||||
|
import org.springframework.web.socket.config.annotation.EnableWebSocket; |
||||
|
import org.springframework.web.socket.config.annotation.WebSocketConfigurer; |
||||
|
import org.springframework.web.socket.config.annotation.WebSocketHandlerRegistry; |
||||
|
|
||||
|
import java.util.List; |
||||
|
|
||||
|
@Configuration |
||||
|
@EnableWebSocket |
||||
|
public class WebSocketConfig implements WebSocketConfigurer { |
||||
|
|
||||
|
@Autowired |
||||
|
private List<BaseWebsocketHandler> handlers; |
||||
|
private final Logger logger = LoggerFactory.getLogger(getClass()); |
||||
|
|
||||
|
@Override |
||||
|
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) { |
||||
|
handlers.forEach(hndl -> { |
||||
|
logger.info("Found websocket bean: {}", hndl.getPath()); |
||||
|
registry.addHandler((WebSocketHandler) hndl, "/ws/" + hndl.getPath()); |
||||
|
}); |
||||
|
} |
||||
|
} |
@ -0,0 +1,115 @@ |
|||||
|
package app.websocket.handlers; |
||||
|
|
||||
|
import app.controllers.other.ExternalVIPController; |
||||
|
import app.entities.VipGiveMethod; |
||||
|
import app.services.db.VIPService; |
||||
|
import app.utils.SaltedCookie; |
||||
|
import app.utils.SteamIDConverter; |
||||
|
import app.websocket.BaseWebsocketHandler; |
||||
|
import com.fasterxml.jackson.core.JsonParseException; |
||||
|
import com.fasterxml.jackson.databind.ObjectMapper; |
||||
|
import lombok.Data; |
||||
|
import lombok.NoArgsConstructor; |
||||
|
import org.slf4j.Logger; |
||||
|
import org.slf4j.LoggerFactory; |
||||
|
import org.springframework.beans.factory.annotation.Autowired; |
||||
|
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.HashSet; |
||||
|
|
||||
|
@Component |
||||
|
public class ExternalVIPHandler extends TextWebSocketHandler implements BaseWebsocketHandler { |
||||
|
private final Logger logger = LoggerFactory.getLogger(getClass()); |
||||
|
private ObjectMapper objectMapper = new ObjectMapper(); |
||||
|
private SaltedCookie saltedCookie; |
||||
|
private VIPService vipService; |
||||
|
private HashSet<String> unique_set; |
||||
|
private ExternalVIPController externalVIPController; |
||||
|
|
||||
|
@Autowired |
||||
|
public ExternalVIPHandler(SaltedCookie saltedCookie, VIPService vipService, ExternalVIPController externalVIPController) { |
||||
|
super(); |
||||
|
this.saltedCookie = saltedCookie; |
||||
|
this.vipService = vipService; |
||||
|
this.externalVIPController = externalVIPController; |
||||
|
this.unique_set = new HashSet(); |
||||
|
} |
||||
|
|
||||
|
@Override |
||||
|
public String getPath() { |
||||
|
return "integration/vip"; |
||||
|
} |
||||
|
|
||||
|
@Override |
||||
|
public void afterConnectionEstablished(WebSocketSession session) throws Exception { |
||||
|
logger.info("Session {} open", session.getId()); |
||||
|
if (!saltedCookie.validateHeaders(session.getHandshakeHeaders())) { |
||||
|
logger.warn("Session {} not acceptable, invalid secret key in header!", session.getId()); |
||||
|
session.close(CloseStatus.NOT_ACCEPTABLE); |
||||
|
return; |
||||
|
} |
||||
|
super.afterConnectionEstablished(session); |
||||
|
session.sendMessage(new TextMessage(objectMapper.writeValueAsString(externalVIPController.getPrices()))); |
||||
|
} |
||||
|
|
||||
|
@Override |
||||
|
public void handleTextMessage(WebSocketSession session, TextMessage message) throws IOException { |
||||
|
if (!saltedCookie.validateHeaders(session.getHandshakeHeaders())) { |
||||
|
logger.warn("Session {} closed, invalid secret key in header!", session.getId()); |
||||
|
session.close(CloseStatus.NOT_ACCEPTABLE); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
ExternalVIP vip; |
||||
|
try { |
||||
|
vip = objectMapper.readValue(message.getPayload(), ExternalVIP.class); |
||||
|
} catch (JsonParseException jpe) { |
||||
|
session.sendMessage(new TextMessage("Invalid payload struct")); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
if (!vip.isFilled()) { |
||||
|
logger.warn("Can try add VIP but, payload is not full filled!"); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
if (!vip.unique.isEmpty()) { |
||||
|
if (unique_set.contains(vip.unique)) { |
||||
|
logger.warn("VIP uniq payload value already exist! " + vip.unique); |
||||
|
return; |
||||
|
} else unique_set.add(vip.unique); |
||||
|
} |
||||
|
|
||||
|
Integer result = vipService.addVIP( |
||||
|
SteamIDConverter.getSteamID(vip.getSteam()), |
||||
|
vip.getAmount(), |
||||
|
VipGiveMethod.valueOf(vip.service.toUpperCase()), |
||||
|
vip.getExtra()); |
||||
|
logger.info("Success add VIP to {}, result: {}", vip.getSteam(), result); |
||||
|
} |
||||
|
|
||||
|
@Override |
||||
|
public void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception { |
||||
|
logger.info("Session {} close", session.getId()); |
||||
|
super.afterConnectionClosed(session, status); |
||||
|
} |
||||
|
|
||||
|
@Data |
||||
|
@NoArgsConstructor |
||||
|
public static class ExternalVIP { |
||||
|
String steam; |
||||
|
Integer amount; |
||||
|
String service; |
||||
|
String extra = ""; |
||||
|
String unique = ""; |
||||
|
|
||||
|
public boolean isFilled() { |
||||
|
return steam != null && amount != null && service != null; |
||||
|
} |
||||
|
} |
||||
|
} |
Loading…
Reference in new issue