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