16 changed files with 388 additions and 188 deletions
@ -1,5 +1,7 @@ |
|||
services: |
|||
a2s_backend: |
|||
build: ./ |
|||
extra_hosts: |
|||
- "tf2.pblr-nyk.pro:192.168.3.3" |
|||
ports: |
|||
- 8082:8082 |
|||
- 8085:8082 |
@ -0,0 +1,74 @@ |
|||
package app.entities.a2s.external; |
|||
|
|||
import app.entities.a2s.requests.A2SRequest; |
|||
import app.entities.a2s.requests.RCONRequest; |
|||
import app.entities.server.players.RCONPlayer; |
|||
import app.utils.SteamIDConverter; |
|||
import com.fasterxml.jackson.annotation.JsonIgnore; |
|||
import jakarta.annotation.PostConstruct; |
|||
import org.springframework.beans.factory.annotation.Value; |
|||
import org.springframework.web.client.RestTemplate; |
|||
|
|||
import java.util.HashMap; |
|||
import java.util.List; |
|||
import java.util.Map; |
|||
|
|||
public abstract class ExternalValveClient { |
|||
@JsonIgnore |
|||
RestTemplate restTemplate; |
|||
@JsonIgnore |
|||
boolean enabled = true; |
|||
@JsonIgnore |
|||
public String gateway = System.getenv("A2S_BACKEND_URL"); |
|||
|
|||
public ExternalValveClient(){ |
|||
restTemplate = new RestTemplate(); |
|||
CheckApi(); |
|||
} |
|||
|
|||
public void CheckApi(){ |
|||
System.out.printf("Check status: %s/api/ping\n", gateway); |
|||
try { |
|||
enabled = restTemplate.getForEntity("%s/api/ping".formatted(gateway), HashMap.class).getBody().containsKey("pong"); |
|||
} catch (Exception err) { |
|||
System.out.print("A2S Backend not respond\n"); |
|||
} |
|||
} |
|||
@JsonIgnore |
|||
public String ExecuteRCON(RCONRequest request){ |
|||
if(!enabled) { |
|||
System.out.printf("External client not enabled, cannot get rcon on %s\n", gateway); |
|||
return "not enabled"; |
|||
} |
|||
try { |
|||
return restTemplate.postForEntity("%s/api/rcon".formatted(gateway), request, String.class).getBody(); |
|||
} catch (Exception err) { |
|||
return "backend error"; |
|||
} |
|||
} |
|||
@JsonIgnore |
|||
public HashMap GetA2SInfo(A2SRequest request){ |
|||
if(!enabled) { |
|||
System.out.printf("External client not enabled, cannot get a2s on %s\n", gateway); |
|||
return null; |
|||
} |
|||
try { |
|||
return restTemplate.postForEntity("%s/api/a2s/info".formatted(gateway), request, HashMap.class).getBody(); |
|||
} catch (Exception err) { |
|||
return null; |
|||
} |
|||
} |
|||
|
|||
public List<RCONPlayer> GetRCONPlayers(RCONRequest request){ |
|||
if(!enabled) { |
|||
System.out.printf("External client not enabled, cannot get rcon players on %s\n", gateway); |
|||
return null; |
|||
} |
|||
try { |
|||
return restTemplate.postForEntity("%s/api/players".formatted(gateway), request, List.class).getBody(); |
|||
} catch (Exception err) { |
|||
err.printStackTrace(); |
|||
return List.of(); |
|||
} |
|||
} |
|||
} |
@ -0,0 +1,162 @@ |
|||
package app.entities.a2s.internal; |
|||
|
|||
import app.entities.a2s.requests.RCONRequest; |
|||
import com.fasterxml.jackson.annotation.JsonIgnore; |
|||
import com.ibasco.agql.core.enums.RateLimitType; |
|||
import com.ibasco.agql.core.util.FailsafeOptions; |
|||
import com.ibasco.agql.core.util.GeneralOptions; |
|||
import com.ibasco.agql.protocols.valve.source.query.SourceQueryClient; |
|||
import com.ibasco.agql.protocols.valve.source.query.SourceQueryOptions; |
|||
import com.ibasco.agql.protocols.valve.source.query.rcon.SourceRconClient; |
|||
import com.ibasco.agql.protocols.valve.source.query.rcon.SourceRconOptions; |
|||
import com.ibasco.agql.protocols.valve.source.query.rcon.message.SourceRconAuthResponse; |
|||
|
|||
import java.util.concurrent.ExecutorService; |
|||
import java.util.concurrent.Executors; |
|||
|
|||
public abstract class InternalValveClient { |
|||
@JsonIgnore |
|||
private ExecutorService executorServices_query; |
|||
@JsonIgnore |
|||
private ExecutorService executorServices_rcon; |
|||
@JsonIgnore |
|||
private SourceRconClient sourceRconClient; |
|||
@JsonIgnore |
|||
private SourceQueryClient sourceQueryClient; |
|||
|
|||
@JsonIgnore |
|||
public SourceRconClient GetSourceRconClient() { |
|||
if (executorServices_rcon == null) executorServices_rcon = Executors.newCachedThreadPool(); |
|||
if (sourceRconClient == null) { |
|||
SourceRconOptions options = SourceRconOptions.builder() |
|||
.option(FailsafeOptions.FAILSAFE_RATELIMIT_TYPE, RateLimitType.SMOOTH) |
|||
.option(GeneralOptions.THREAD_EXECUTOR_SERVICE, executorServices_rcon) |
|||
.build(); |
|||
sourceRconClient = new SourceRconClient(options); |
|||
} |
|||
return sourceRconClient; |
|||
} |
|||
|
|||
|
|||
@JsonIgnore |
|||
public SourceQueryClient GetSourceQueryClient() { |
|||
if (executorServices_query == null) executorServices_query = Executors.newCachedThreadPool(); |
|||
if (sourceQueryClient == null) { |
|||
SourceQueryOptions options = SourceQueryOptions.builder() |
|||
.option(FailsafeOptions.FAILSAFE_RATELIMIT_TYPE, RateLimitType.SMOOTH) |
|||
.option(GeneralOptions.THREAD_EXECUTOR_SERVICE, executorServices_query) |
|||
.build(); |
|||
sourceQueryClient = new SourceQueryClient(options); |
|||
} |
|||
return sourceQueryClient; |
|||
} |
|||
|
|||
@JsonIgnore |
|||
public String ExecuteRCON(RCONRequest request) { |
|||
try (SourceRconClient rconClient = GetSourceRconClient()) { |
|||
SourceRconAuthResponse response = rconClient.authenticate(request.getInetAddress(), request.getPassword().getBytes()).join(); |
|||
if (!response.isAuthenticated()) { |
|||
if (!rconClient.getExecutor().isShutdown()) rconClient.getExecutor().shutdown(); |
|||
return null; |
|||
} |
|||
return rconClient.execute(request.getInetAddress(), request.getCommand()) |
|||
.thenApplyAsync(out -> { |
|||
rconClient.cleanup(true); |
|||
return out.getResult(); |
|||
}) |
|||
.join(); |
|||
} catch (Exception err) { |
|||
return ""; |
|||
} |
|||
} |
|||
|
|||
/* |
|||
* public void UpdateStatusFromA2S(SourceQueryInfoResponse response) { |
|||
SetDownStatus(); |
|||
if (response == null) return; |
|||
|
|||
setMax_players(response.getResult().getMaxPlayers()); |
|||
setPlayer_count(response.getResult().getNumOfPlayers()); |
|||
setMap(response.getResult().getMapName()); |
|||
setStatus(true); |
|||
}*/ |
|||
/* |
|||
* public void RefreshServerA2SData(String server_name) { |
|||
//try (SourceQueryClient sourceQueryClient = context.getBean(SourceQueryClient.class)) {
|
|||
try (SourceQueryClient sourceQueryClient = getServers().get(server_name).GetSourceQueryClient()) { |
|||
sourceQueryClient.getInfo(getServers().get(server_name).getInetAddress()).whenComplete((info, error) -> { |
|||
if (!sourceQueryClient.getExecutor().isShutdown()) sourceQueryClient.getExecutor().shutdown(); |
|||
if (error != null) { |
|||
getServers().get(server_name).SetDownStatus(); |
|||
return; |
|||
} |
|||
getServers().get(server_name).UpdateStatusFromA2S(info); |
|||
}).join(); |
|||
} catch (CompletionException | IOException err) { |
|||
} |
|||
|
|||
if (!getServers().get(server_name).isStatus() || getServers().get(server_name).getPlayer_count() < 1) { |
|||
return; |
|||
} |
|||
////////////////////////////////////////////////////////////////////////
|
|||
//If player count > 0 make base player request
|
|||
////////////////////////////////////////////////////////////////////////
|
|||
//try (SourceQueryClient sourceQueryClient = context.getBean(SourceQueryClient.class)) {
|
|||
try (SourceQueryClient sourceQueryClient = getServers().get(server_name).GetSourceQueryClient()) { |
|||
sourceQueryClient.getPlayers(getServers().get(server_name).getInetAddress()).whenComplete((players, error) -> { |
|||
if (!sourceQueryClient.getExecutor().isShutdown()) sourceQueryClient.getExecutor().shutdown(); |
|||
if (error != null) return; |
|||
getServers().get(server_name).UpdatePlayersFromA2S(players); |
|||
}).join(); |
|||
} catch (CompletionException | IOException err) {} |
|||
///////////////////////////////////////////////////////////////////////
|
|||
//Extend current players of rcon result
|
|||
//////////////////////////////////////////////////////////////////////
|
|||
try { |
|||
String response = getServers().get(server_name).ExecuteRCON("status"); |
|||
getServers().get(server_name).UpdatePlayersFromRCON(response); |
|||
} catch (RconException | CompletionException err) { |
|||
return; |
|||
} |
|||
}*/ |
|||
/* |
|||
* public void UpdatePlayersFromA2S(SourceQueryPlayerResponse response) { |
|||
a2s_players.clear(); |
|||
if (response != null) { |
|||
response.getResult().stream().map(app.entities.server.players.SourcePlayer::new).forEach(a2s_players::add); |
|||
} |
|||
} |
|||
|
|||
public void UpdatePlayersFromRCON(String response) { |
|||
players.clear(); |
|||
int start_index = response.indexOf("# userid"); |
|||
if (start_index == -1) return; |
|||
List<String> players_list = Arrays.stream(response.substring(start_index, response.length()).split("\n")).toList(); |
|||
boolean skip_table_header = true; |
|||
for(String player_text: players_list) { |
|||
if (skip_table_header || player_text.length() < 1) { |
|||
skip_table_header = false; |
|||
continue; |
|||
} |
|||
/////////////////////////////////////////////////////
|
|||
List<String> player_line = Arrays.stream(player_text.split("\\s+")).toList(); |
|||
RCONPlayer player; |
|||
try { |
|||
player = new RCONPlayer(player_line); |
|||
} catch (Exception parse_err) { |
|||
System.out.println("Cannot parse: " + player_line); |
|||
continue; |
|||
} |
|||
|
|||
for (SourcePlayer sourcePlayer: a2s_players) { |
|||
if (sourcePlayer.getName().equals(player.getName())) { |
|||
player.setScore(sourcePlayer.getScore()); |
|||
a2s_players.remove(sourcePlayer); |
|||
players.add(player); |
|||
break; |
|||
} |
|||
} |
|||
} |
|||
a2s_players.clear(); |
|||
}*/ |
|||
} |
@ -0,0 +1,17 @@ |
|||
package app.entities.a2s.requests; |
|||
|
|||
import com.fasterxml.jackson.databind.annotation.JsonSerialize; |
|||
import lombok.AllArgsConstructor; |
|||
import lombok.Data; |
|||
|
|||
@JsonSerialize |
|||
@Data |
|||
public class A2SRequest { |
|||
String server; |
|||
|
|||
public A2SRequest(){} |
|||
|
|||
public A2SRequest(String server){ |
|||
this.server = server; |
|||
} |
|||
} |
@ -0,0 +1,23 @@ |
|||
package app.entities.a2s.requests; |
|||
|
|||
import com.fasterxml.jackson.annotation.JsonIgnore; |
|||
import com.fasterxml.jackson.databind.annotation.JsonSerialize; |
|||
import lombok.AllArgsConstructor; |
|||
import lombok.Getter; |
|||
|
|||
import java.net.InetSocketAddress; |
|||
|
|||
@JsonSerialize |
|||
@AllArgsConstructor |
|||
@Getter |
|||
public class RCONRequest extends A2SRequest{ |
|||
String server; |
|||
String password; |
|||
String command; |
|||
|
|||
@JsonIgnore |
|||
public InetSocketAddress getInetAddress() { |
|||
String[] splitted_address = server.split(":", 2); |
|||
return new InetSocketAddress(splitted_address[0], Integer.parseInt(splitted_address[1])); |
|||
} |
|||
} |
@ -1,51 +0,0 @@ |
|||
package app.entities.server; |
|||
|
|||
import com.fasterxml.jackson.annotation.JsonIgnore; |
|||
import com.ibasco.agql.core.enums.RateLimitType; |
|||
import com.ibasco.agql.core.util.FailsafeOptions; |
|||
import com.ibasco.agql.core.util.GeneralOptions; |
|||
import com.ibasco.agql.protocols.valve.source.query.SourceQueryClient; |
|||
import com.ibasco.agql.protocols.valve.source.query.SourceQueryOptions; |
|||
import com.ibasco.agql.protocols.valve.source.query.rcon.SourceRconClient; |
|||
import com.ibasco.agql.protocols.valve.source.query.rcon.SourceRconOptions; |
|||
|
|||
import java.util.concurrent.ExecutorService; |
|||
import java.util.concurrent.Executors; |
|||
|
|||
public abstract class BaseServer { |
|||
@JsonIgnore |
|||
private ExecutorService executorServices_query; |
|||
@JsonIgnore |
|||
private ExecutorService executorServices_rcon; |
|||
@JsonIgnore |
|||
private SourceRconClient sourceRconClient; |
|||
@JsonIgnore |
|||
private SourceQueryClient sourceQueryClient; |
|||
|
|||
@JsonIgnore |
|||
public SourceRconClient GetSourceRconClient() { |
|||
if (executorServices_rcon == null) executorServices_rcon = Executors.newCachedThreadPool(); |
|||
if (sourceRconClient == null) { |
|||
SourceRconOptions options = SourceRconOptions.builder() |
|||
.option(FailsafeOptions.FAILSAFE_RATELIMIT_TYPE, RateLimitType.SMOOTH) |
|||
.option(GeneralOptions.THREAD_EXECUTOR_SERVICE, executorServices_rcon) |
|||
.build(); |
|||
sourceRconClient = new SourceRconClient(options); |
|||
} |
|||
return sourceRconClient; |
|||
} |
|||
|
|||
|
|||
@JsonIgnore |
|||
public SourceQueryClient GetSourceQueryClient() { |
|||
if (executorServices_query == null) executorServices_query = Executors.newCachedThreadPool(); |
|||
if (sourceQueryClient == null) { |
|||
SourceQueryOptions options = SourceQueryOptions.builder() |
|||
.option(FailsafeOptions.FAILSAFE_RATELIMIT_TYPE, RateLimitType.SMOOTH) |
|||
.option(GeneralOptions.THREAD_EXECUTOR_SERVICE, executorServices_query) |
|||
.build(); |
|||
sourceQueryClient = new SourceQueryClient(options); |
|||
} |
|||
return sourceQueryClient; |
|||
} |
|||
} |
@ -0,0 +1,29 @@ |
|||
package app.servers; |
|||
|
|||
import app.entities.a2s.external.ExternalValveClient; |
|||
import app.entities.server.Server; |
|||
import org.junit.Test; |
|||
|
|||
public class TestExternalA2S { |
|||
Server server = new Server(); |
|||
|
|||
public TestExternalA2S(){ |
|||
server.gateway = System.getenv("A2S_BACKEND_URL"); |
|||
server.setAddress(System.getenv("TEST_SERVER")); |
|||
} |
|||
|
|||
@Test |
|||
public void CheckA2SInfo(){ |
|||
server.UpdateStatusFromA2S(); |
|||
System.out.printf("%s\n", server); |
|||
} |
|||
|
|||
@Test |
|||
public void CheckPlayers(){ |
|||
server.setRcon_password(System.getenv("TEST_SERVER_PASSWORD")); |
|||
server.UpdatePlayers(); |
|||
System.out.print(server.getPlayers()); |
|||
} |
|||
|
|||
|
|||
} |
Loading…
Reference in new issue