13 changed files with 473 additions and 110 deletions
@ -0,0 +1,36 @@ |
|||
package app.controllers.server; |
|||
|
|||
import app.annotations.enums.AuthMethod; |
|||
import app.annotations.interfaces.CheckWebAccess; |
|||
import app.entities.report.ReportType; |
|||
import app.services.ProfileService; |
|||
import app.services.db.ReportService; |
|||
import jakarta.servlet.http.HttpServletRequest; |
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.http.HttpStatus; |
|||
import org.springframework.http.ResponseEntity; |
|||
import org.springframework.web.bind.annotation.*; |
|||
|
|||
@RestController |
|||
@RequestMapping("api/external/report") |
|||
public class ServerReportController { |
|||
private ReportService reportService; |
|||
private ProfileService profileService; |
|||
|
|||
@Autowired |
|||
public ServerReportController(ReportService reportService, |
|||
ProfileService profileService) { |
|||
this.reportService = reportService; |
|||
this.profileService = profileService; |
|||
} |
|||
|
|||
@PostMapping(value = "/{author_steam64}/{reported_steam64}") |
|||
@CheckWebAccess(auth_method = AuthMethod.SECRET_KEY) |
|||
public ResponseEntity sendReport(HttpServletRequest request, |
|||
@PathVariable(required = true) String author_steam64, |
|||
@PathVariable(required = false) String reported_steam64, |
|||
@RequestBody String text) { |
|||
return ResponseEntity.status(HttpStatus.OK).body( |
|||
reportService.createReport(ReportType.IN_GAME, profileService.GetProfile(author_steam64, "permition,steam_data"), profileService.GetProfile(reported_steam64, "permition"), text)); |
|||
} |
|||
} |
@ -0,0 +1,134 @@ |
|||
package app.entities.report; |
|||
|
|||
import app.entities.other.SteamID; |
|||
import app.utils.SteamIDConverter; |
|||
|
|||
import java.sql.ResultSet; |
|||
import java.sql.SQLException; |
|||
|
|||
public class Report { |
|||
private final Integer id; |
|||
|
|||
//author
|
|||
private final String a_nickname; |
|||
private final String a_steam2; |
|||
private final Integer a_permition; |
|||
private final Integer a_kills; |
|||
private final Integer a_deads; |
|||
private final Integer a_seconds; |
|||
|
|||
//reported
|
|||
private final String r_nickname; |
|||
private final String r_steam2; |
|||
private final Integer r_permition; |
|||
private final Integer r_kills; |
|||
private final Integer r_deads; |
|||
private final Integer r_seconds; |
|||
|
|||
//based
|
|||
private final String reasons; |
|||
private final long utime; |
|||
private final String srv; |
|||
private final Integer online; |
|||
private final Integer type; |
|||
|
|||
public Report(ResultSet rs) throws SQLException { |
|||
id = rs.getInt("id"); |
|||
//
|
|||
a_nickname = rs.getString("a_nickname"); |
|||
a_steam2 = rs.getString("a_steam2"); |
|||
a_permition = rs.getInt("a_permition"); |
|||
a_kills = rs.getInt("a_kills"); |
|||
a_deads = rs.getInt("a_deads"); |
|||
a_seconds = rs.getInt("a_seconds"); |
|||
//
|
|||
r_nickname = rs.getString("r_nickname"); |
|||
r_steam2 = rs.getString("r_steam2"); |
|||
r_permition = rs.getInt("r_permition"); |
|||
r_kills = rs.getInt("r_kills"); |
|||
r_deads = rs.getInt("r_deads"); |
|||
r_seconds = rs.getInt("r_seconds"); |
|||
//based
|
|||
reasons = rs.getString("reasons"); |
|||
utime = rs.getLong("utime"); |
|||
srv = rs.getString("srv"); |
|||
online = rs.getInt("online"); |
|||
type = rs.getInt("type"); |
|||
} |
|||
|
|||
public Integer getId() { |
|||
return id; |
|||
} |
|||
|
|||
public String getA_nickname() { |
|||
return a_nickname; |
|||
} |
|||
|
|||
public ReportPermition getA_permition() { |
|||
return ReportPermition.values()[a_permition]; |
|||
} |
|||
|
|||
public Integer getA_kills() { |
|||
return a_kills; |
|||
} |
|||
|
|||
public Integer getA_deads() { |
|||
return a_deads; |
|||
} |
|||
|
|||
public Integer getA_seconds() { |
|||
return a_seconds; |
|||
} |
|||
|
|||
public String getR_nickname() { |
|||
return r_nickname; |
|||
} |
|||
|
|||
public ReportPermition getR_permition() { |
|||
return ReportPermition.values()[r_permition]; |
|||
} |
|||
|
|||
public Integer getR_kills() { |
|||
return r_kills; |
|||
} |
|||
|
|||
public Integer getR_deads() { |
|||
return r_deads; |
|||
} |
|||
|
|||
public Integer getR_seconds() { |
|||
return r_seconds; |
|||
} |
|||
|
|||
public String getReasons() { |
|||
return reasons; |
|||
} |
|||
|
|||
public long getUtime() { |
|||
return utime; |
|||
} |
|||
|
|||
public String getSrv() { |
|||
return srv; |
|||
} |
|||
|
|||
public Integer getOnline() { |
|||
return online; |
|||
} |
|||
|
|||
public ReportType getType() { |
|||
return ReportType.values()[type]; |
|||
} |
|||
|
|||
public SteamID getA_steam() { |
|||
return SteamIDConverter.getSteamID(a_steam2); |
|||
} |
|||
|
|||
public SteamID getR_steam() { |
|||
return SteamIDConverter.getSteamID(r_steam2); |
|||
} |
|||
|
|||
public boolean selfReport() { |
|||
return r_nickname == null && r_steam2 == null; |
|||
} |
|||
} |
@ -0,0 +1,25 @@ |
|||
package app.entities.report; |
|||
|
|||
public class ReportCount { |
|||
private final long created; |
|||
private final long accepted; |
|||
private final long selfcount; |
|||
|
|||
public ReportCount(long created, long accepted, long selfcount) { |
|||
this.created = created; |
|||
this.accepted = accepted; |
|||
this.selfcount = selfcount; |
|||
} |
|||
|
|||
public long getCreated() { |
|||
return created; |
|||
} |
|||
|
|||
public long getAccepted() { |
|||
return accepted; |
|||
} |
|||
|
|||
public long getSelfcount() { |
|||
return selfcount; |
|||
} |
|||
} |
@ -0,0 +1,19 @@ |
|||
package app.entities.report; |
|||
|
|||
public enum ReportPermition { |
|||
DONT_HAVE, |
|||
ROOT, |
|||
ADMIN, |
|||
MODERATOR, |
|||
FREE, |
|||
VIP, |
|||
UNKNOWN; |
|||
|
|||
public String getPrefixToDiscord() { |
|||
if (this.equals(DONT_HAVE)) return ""; |
|||
if (this.equals(UNKNOWN)) return ""; |
|||
return new StringBuilder(" | ") |
|||
.append(toString().toUpperCase()) |
|||
.toString(); |
|||
} |
|||
} |
@ -0,0 +1,5 @@ |
|||
package app.entities.report; |
|||
|
|||
public enum ReportType { |
|||
IN_GAME, WEBSITE |
|||
} |
@ -1,95 +0,0 @@ |
|||
package app.services; |
|||
|
|||
import app.entities.PlayerProfile; |
|||
import app.entities.Stats; |
|||
import app.entities.db.Permition; |
|||
import app.entities.server.players.RCONPlayer; |
|||
import com.fasterxml.jackson.databind.ObjectMapper; |
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.beans.factory.annotation.Value; |
|||
import org.springframework.stereotype.Service; |
|||
import org.springframework.web.client.RestTemplate; |
|||
|
|||
import java.time.Instant; |
|||
import java.util.ArrayList; |
|||
import java.util.HashMap; |
|||
import java.util.List; |
|||
import java.util.Map; |
|||
|
|||
@Service |
|||
public class ReportService { |
|||
|
|||
@Value("${backend.report.webhook_url}") |
|||
String webhook_url; |
|||
@Value("${backend.report.kd}") |
|||
long kd; |
|||
HashMap<Long, Long> userKD; |
|||
Stats stats; |
|||
StatsService statsService; |
|||
String tf2_icon = "https://steamcdn-a.akamaihd.net/steamcommunity/public/images/apps/440/e3f595a92552da3d664ad00277fad2107345f743.jpg"; |
|||
RestTemplate restTemplate; |
|||
|
|||
@Autowired |
|||
public ReportService(Stats stats, |
|||
StatsService statsService) { |
|||
this.stats = stats; |
|||
this.statsService = statsService; |
|||
userKD = new HashMap<>(); |
|||
restTemplate = new RestTemplate(); |
|||
} |
|||
|
|||
public long createReport(PlayerProfile current_user, PlayerProfile reported_user, String text) { |
|||
if (reported_user.getPlay_on() == null) return 0L; |
|||
if (userKD.containsKey(current_user.getSteamids().steam64) && |
|||
(Instant.now().getEpochSecond() - userKD.get(current_user.getSteamids().steam64)) < kd) { |
|||
return (Instant.now().getEpochSecond() - userKD.get(current_user.getSteamids().steam64)) - kd; |
|||
} |
|||
|
|||
String author = "%s | %s%s | WEBSITE".formatted( |
|||
current_user.getSteam_data().getNickname(), |
|||
current_user.getSteamids().steam2, |
|||
Permition2Prefix(current_user.getPermition()) |
|||
); |
|||
|
|||
RCONPlayer reported_player = statsService.searchPlayer(reported_user.getPlay_on()); |
|||
|
|||
String reported = "%s | %s%s | %d/? %s".formatted( |
|||
reported_player.getName(), |
|||
reported_user.getSteamids().steam2, |
|||
Permition2Prefix(reported_user.getPermition()), |
|||
reported_player.getScore(), |
|||
reported_player.getDuration() |
|||
); |
|||
|
|||
//Create Embed
|
|||
HashMap<String, Object> embed = new HashMap<>(); |
|||
embed.put("author", Map.of("name", author, "url", current_user.getSteamids().community_url)); |
|||
embed.put("title", reported); |
|||
embed.put("url", reported_user.getSteamids().community_url); |
|||
embed.put("description", text); |
|||
embed.put("color", 1488); |
|||
embed.put("footer", Map.of( |
|||
"text", "%s • %d/%d".formatted( |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getName(), |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getPlayer_count(), |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getMax_players()), |
|||
"icon_url", tf2_icon |
|||
)); |
|||
|
|||
HashMap<String, Object> payload = new HashMap<>(); |
|||
payload.put("embeds", List.of(embed)); |
|||
restTemplate.postForEntity(webhook_url, (Object) payload, String.class); |
|||
userKD.merge(current_user.getSteamids().steam64, Instant.now().getEpochSecond(), (x,y) -> y); |
|||
return userKD.get(current_user.getSteamids().steam64); |
|||
} |
|||
|
|||
public String Permition2Prefix(Permition permition) { |
|||
if (permition == null) return ""; |
|||
if (permition.getFlags().contains("z")) return " | [ROOT]"; |
|||
if (permition.getFlags().contains("b")) return " | [ADMIN]"; |
|||
if (permition.getFlags().contains("d")) return " | [MODERATOR]"; |
|||
if (permition.getFlags().contains("t")) return " | [FREE]"; |
|||
if (permition.getFlags().contains("a")) return " | [VIP]"; |
|||
return ""; |
|||
} |
|||
} |
@ -0,0 +1,189 @@ |
|||
package app.services.db; |
|||
|
|||
import app.entities.PlayerProfile; |
|||
import app.entities.Stats; |
|||
import app.entities.db.Permition; |
|||
import app.entities.other.SteamID; |
|||
import app.entities.report.Report; |
|||
import app.entities.report.ReportPermition; |
|||
import app.entities.report.ReportType; |
|||
import app.entities.server.players.RCONPlayer; |
|||
import app.entities.server.request.PlayerOnServer; |
|||
import app.services.ProfileService; |
|||
import app.services.StatsService; |
|||
import com.fasterxml.jackson.databind.ObjectMapper; |
|||
import org.springframework.beans.factory.annotation.Autowired; |
|||
import org.springframework.beans.factory.annotation.Qualifier; |
|||
import org.springframework.beans.factory.annotation.Value; |
|||
import org.springframework.jdbc.core.JdbcTemplate; |
|||
import org.springframework.stereotype.Service; |
|||
import org.springframework.web.client.RestTemplate; |
|||
|
|||
import java.sql.Timestamp; |
|||
import java.time.Instant; |
|||
import java.util.ArrayList; |
|||
import java.util.HashMap; |
|||
import java.util.List; |
|||
import java.util.Map; |
|||
|
|||
@Service |
|||
public class ReportService { |
|||
|
|||
@Value("${backend.report.webhook_url}") |
|||
String webhook_url; |
|||
@Value("${backend.report.kd}") |
|||
long kd; |
|||
HashMap<Long, Long> userKD; |
|||
Stats stats; |
|||
StatsService statsService; |
|||
String tf2_icon = "https://steamcdn-a.akamaihd.net/steamcommunity/public/images/apps/440/e3f595a92552da3d664ad00277fad2107345f743.jpg"; |
|||
RestTemplate restTemplate; |
|||
|
|||
@Autowired |
|||
@Qualifier("jt_rw") |
|||
private JdbcTemplate jdbcTemplate_rw; |
|||
|
|||
@Autowired |
|||
@Qualifier("jt_ro") |
|||
private JdbcTemplate jdbcTemplate_ro; |
|||
|
|||
@Autowired |
|||
public ReportService(Stats stats, |
|||
StatsService statsService) { |
|||
this.stats = stats; |
|||
this.statsService = statsService; |
|||
userKD = new HashMap<>(); |
|||
restTemplate = new RestTemplate(); |
|||
} |
|||
|
|||
private void addReport(PlayerProfile current_user, PlayerProfile reported_user, String text, ReportType reportType) { |
|||
PlayerOnServer current_player = (PlayerOnServer) statsService.searchPlayer(current_user); |
|||
PlayerOnServer reported_player = (PlayerOnServer) statsService.searchPlayer(reported_user); |
|||
|
|||
jdbcTemplate_rw.update("INSERT INTO user_reports (a_nickname, a_steam2, a_permition, a_kills, a_deads, a_seconds, " + |
|||
"r_nickname, r_steam2, r_permition, r_kills, r_deads, r_seconds, " + |
|||
"reasons, utime, srv, online, type) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)", |
|||
current_player == null ? current_user.getSteam_data().getNickname() : current_player.getName(), current_user.getSteamids().steam2, Permition2Prefix(current_user.getPermition()).ordinal(), |
|||
current_player == null?null:current_player.getScore(), current_player == null?null:current_player.getDeads(),current_player == null?null:current_player.getDuration_seconds(), |
|||
|
|||
reported_player==null?null:reported_player.getName(), |
|||
reported_player==null?null:reported_user.getSteamids().steam2, |
|||
reported_player==null?null:Permition2Prefix(reported_user.getPermition()).ordinal(), |
|||
reported_player==null?null:reported_player.getScore(), |
|||
reported_player==null?null:reported_player.getDeads(), |
|||
reported_player==null?null:reported_player.getDuration_seconds(), |
|||
|
|||
text, Instant.now().getEpochSecond(), reported_user.getPlay_on().getServer_id(), |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getPlayer_count(), |
|||
reportType.ordinal()); |
|||
} |
|||
|
|||
public long getCreatedReportsCount(SteamID steamID) { |
|||
return jdbcTemplate_ro.query("SELECT COUNT(*) as c FROM user_reports WHERE a_steam2 LIKE ?", new Object[]{steamID.steam2}, (rs,n) -> rs.getLong("c")).get(0); |
|||
} |
|||
|
|||
public long getReportsCount(SteamID steamID) { |
|||
return jdbcTemplate_ro.query("SELECT COUNT(*) as c FROM user_reports WHERE r_steam2 LIKE ?", new Object[]{steamID.steam2}, (rs,n) -> rs.getLong("c")).get(0); |
|||
} |
|||
|
|||
public long getCreatedSelfReportsCount(SteamID steamID) { |
|||
return jdbcTemplate_ro.query("SELECT COUNT(*) as c FROM user_reports WHERE a_steam2 LIKE ? AND r_steam2 IS NULL", new Object[]{steamID.steam2}, (rs,n) -> rs.getLong("c")).get(0); |
|||
} |
|||
|
|||
public List<Report> getCreatedReports(SteamID steamID) { |
|||
return jdbcTemplate_ro.query("SELECT * FROM user_reports WHERE a_steam2 LIKE ?", new Object[]{steamID.steam2}, (rs,n) -> new Report(rs)); |
|||
} |
|||
|
|||
public List<Report> getReports(SteamID steamID) { |
|||
return jdbcTemplate_ro.query("SELECT * FROM user_reports WHERE r_steam2 LIKE ?", new Object[]{steamID.steam2}, (rs,n) -> new Report(rs)); |
|||
} |
|||
|
|||
public long createReport(ReportType reportType, PlayerProfile current_user, PlayerProfile reported_user, String text) { |
|||
switch (reportType) { |
|||
case WEBSITE -> { |
|||
if (reported_user.getPlay_on() == null) return 0L; |
|||
if (userKD.containsKey(current_user.getSteamids().steam64) && |
|||
(Instant.now().getEpochSecond() - userKD.get(current_user.getSteamids().steam64)) < kd) { |
|||
return (Instant.now().getEpochSecond() - userKD.get(current_user.getSteamids().steam64)) - kd; |
|||
} |
|||
addReport(current_user, reported_user, text, reportType); |
|||
long res = createReportToDiscord(current_user, reported_user, text); |
|||
if (res == 0) return 0; |
|||
|
|||
userKD.merge(current_user.getSteamids().steam64, res, (x,y) -> y); |
|||
return res; |
|||
} |
|||
case IN_GAME -> { |
|||
addReport(current_user, reported_user, text, reportType); |
|||
return createReportToDiscord(current_user, reported_user, text); |
|||
} |
|||
default -> { |
|||
return -1; |
|||
} |
|||
} |
|||
} |
|||
|
|||
private long createReportToDiscord(PlayerProfile current_user, PlayerProfile reported_user, String text) { |
|||
PlayerOnServer current_player = (PlayerOnServer) statsService.searchPlayer(current_user); |
|||
String author = "%s | %s%s | %s".formatted( |
|||
current_player == null ? current_user.getSteam_data().getNickname() : current_player.getName(), |
|||
current_user.getSteamids().steam2, |
|||
Permition2PrefixToDiscord(current_user.getPermition()), |
|||
current_player == null?"WEBSITE":"%d/%d %s".formatted( |
|||
current_player.getScore(), current_player.getDeads(), current_player.getDuration()) |
|||
); |
|||
|
|||
String reported = ""; |
|||
if (reported_user != null) { |
|||
PlayerOnServer reported_player = (PlayerOnServer) statsService.searchPlayer(reported_user); |
|||
if (reported_player == null) return 0L; |
|||
|
|||
reported = "%s | %s%s | %d/%d %s".formatted( |
|||
reported_player.getName(), |
|||
reported_user.getSteamids().steam2, |
|||
Permition2PrefixToDiscord(reported_user.getPermition()), |
|||
reported_player.getScore(), |
|||
reported_player.getDeads(), |
|||
reported_player.getDuration() |
|||
); |
|||
} |
|||
|
|||
//Create Embed
|
|||
HashMap<String, Object> embed = new HashMap<>(); |
|||
embed.put("author", Map.of("name", author, "url", current_user.getSteamids().community_url)); |
|||
|
|||
if (reported_user != null) |
|||
embed.put("title", reported); |
|||
|
|||
embed.put("url", reported_user.getSteamids().community_url); |
|||
embed.put("description", text); |
|||
embed.put("color", 16581375); // 255 * 255 * 255 | black
|
|||
embed.put("footer", Map.of( |
|||
"text", "%s • %d/%d ".formatted( |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getName(), |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getPlayer_count(), |
|||
stats.getServers().get(reported_user.getPlay_on().getServer_id()).getMax_players()), |
|||
"icon_url", tf2_icon |
|||
)); |
|||
embed.put("timestamp", Instant.now().toString()); |
|||
|
|||
HashMap<String, Object> payload = new HashMap<>(); |
|||
payload.put("embeds", List.of(embed)); |
|||
restTemplate.postForEntity(webhook_url, (Object) payload, String.class); |
|||
return Instant.now().getEpochSecond(); |
|||
} |
|||
|
|||
private String Permition2PrefixToDiscord(Permition permition) { |
|||
return Permition2Prefix(permition).getPrefixToDiscord(); |
|||
} |
|||
|
|||
private ReportPermition Permition2Prefix(Permition permition) { |
|||
if (permition == null) return ReportPermition.DONT_HAVE; |
|||
if (permition.getFlags().contains("z")) return ReportPermition.ROOT; |
|||
if (permition.getFlags().contains("b")) return ReportPermition.ADMIN; |
|||
if (permition.getFlags().contains("d")) return ReportPermition.MODERATOR; |
|||
if (permition.getFlags().contains("t")) return ReportPermition.FREE; |
|||
if (permition.getFlags().contains("a")) return ReportPermition.VIP; |
|||
return ReportPermition.UNKNOWN; |
|||
} |
|||
} |
Loading…
Reference in new issue