From 96d7ce46cc6398a8f1bde93dab0db8814b6aeb4c Mon Sep 17 00:00:00 2001 From: gsd Date: Sun, 5 Nov 2023 19:01:05 +0300 Subject: [PATCH] threaded profile and new tops --- .../controllers/user/KillFeedController.java | 16 ++ .../java/app/services/ProfileService.java | 155 ++++++++++++------ .../java/app/services/db/KillfeedService.java | 56 ++++++- 3 files changed, 165 insertions(+), 62 deletions(-) diff --git a/src/main/java/app/controllers/user/KillFeedController.java b/src/main/java/app/controllers/user/KillFeedController.java index d79181a..ef2ce27 100644 --- a/src/main/java/app/controllers/user/KillFeedController.java +++ b/src/main/java/app/controllers/user/KillFeedController.java @@ -37,6 +37,14 @@ public class KillFeedController { return new ResponseEntity(killfeedService.getPopulateWeapons(steamID, srv, offset, limit), HttpStatus.OK); } + @GetMapping("/top") + @CheckWebAccess(auth_method = AuthMethod.STEAM64) + @WaitAfterNext(order = "top") + public ResponseEntity getTopKills(HttpServletRequest request, + @RequestParam(required = false) String srv) { + return new ResponseEntity(killfeedService.getTopKills(srv), HttpStatus.OK); + } + @GetMapping @CheckWebAccess(auth_method = AuthMethod.STEAM64) @WaitAfterNext(order = "kills") @@ -81,4 +89,12 @@ public class KillFeedController { if (limit > 20) return new ResponseEntity(HttpStatus.NOT_ACCEPTABLE); return new ResponseEntity(killfeedService.getDeads(steamID, srv, offset, limit), HttpStatus.OK); } + + @DeleteMapping("/top") + @CheckWebAccess(auth_method = AuthMethod.STEAM64) + @WaitAfterNext(order = "top") + public ResponseEntity getTopDeads(HttpServletRequest request, + @RequestParam(required = false) String srv) { + return new ResponseEntity(killfeedService.getTopDeads(srv), HttpStatus.OK); + } } diff --git a/src/main/java/app/services/ProfileService.java b/src/main/java/app/services/ProfileService.java index d76588d..5012ca1 100644 --- a/src/main/java/app/services/ProfileService.java +++ b/src/main/java/app/services/ProfileService.java @@ -14,7 +14,15 @@ import org.springframework.stereotype.Service; import java.time.Instant; import java.util.HashMap; +import java.util.HashSet; import java.util.List; +import java.util.Set; +import java.util.concurrent.Callable; +import java.util.concurrent.Executor; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.function.Consumer; +import java.util.function.Supplier; @Service public class ProfileService { @@ -61,97 +69,136 @@ public class ProfileService { } public PlayerProfile GetProfile(String steam64, List requests) { + final int requests_pool = 16; + ExecutorService executor = Executors.newFixedThreadPool(requests_pool); + Set> callables = new HashSet<>(requests_pool); + PlayerProfile profile = new PlayerProfile(); profile.setResponse_time(new HashMap<>()); SteamID steamID = SteamIDConverter.getSteamID(steam64); profile.setSteamids(steamID); - Long start_time, end_time; + Long start_time_total = Instant.now().toEpochMilli(); - start_time = Instant.now().toEpochMilli(); - profile.setPlay_on(statsService.searchPlayer(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("play_on", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setPlay_on(statsService.searchPlayer(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("play_on", (double) end_time / 1000); + return null; + }); if(requests.contains("steam_data")){ - start_time = Instant.now().toEpochMilli(); - profile.setSteam_data(steamWebApi.getSteamData(steamID.steam64)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("steam_data", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setSteam_data(steamWebApi.getSteamData(steamID.steam64)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("steam_data", (double) end_time / 1000); + return null; + }); } if(requests.contains("lastplay")){ - start_time = Instant.now().toEpochMilli(); - profile.setLastplay(usertimeService.getPlayerLastplay(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("lastplay", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setLastplay(usertimeService.getPlayerLastplay(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("lastplay", (double) end_time / 1000); + return null; + }); } if(requests.contains("usertime")){ - start_time = Instant.now().toEpochMilli(); - profile.setGametime(usertimeService.getPlayerUsertime(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("usertime", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setGametime(usertimeService.getPlayerUsertime(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("usertime", (double) end_time / 1000); + return null; + }); } //if(requests.contains("permition")){ - start_time = Instant.now().toEpochMilli(); - profile.setPermition(permitionService.getPermition(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("permition", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setPermition(permitionService.getPermition(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("permition", (double) end_time / 1000); + return null; + }); //} //if(requests.contains("ban")){ - start_time = Instant.now().toEpochMilli(); - profile.setBan(banService.getBan(steamID)); - /*if(profile.getBan() != null) { - profile.getBan().setAdmin_info( - permitionService.getAdminInfo(SteamIDConverter.getSteamID(profile.getBan().getBanned_by_id())) - ); - }*/ - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("ban", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setBan(banService.getBan(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("ban", (double) end_time / 1000); + return null; + }); //} if(requests.contains("attached_discord")){ - start_time = Instant.now().toEpochMilli(); - profile.setAttached_discords(detectService.getAttachedDiscordAccountPerSteam(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("attached_discord", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setAttached_discords(detectService.getAttachedDiscordAccountPerSteam(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("attached_discord", (double) end_time / 1000); + return null; + }); } if(requests.contains("donates")){ - start_time = Instant.now().toEpochMilli(); - profile.setDonates(donateService.getDonateStatistic(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("donates", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setDonates(donateService.getDonateStatistic(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("donates", (double) end_time / 1000); + return null; + }); } if(requests.contains("ban_list")){ - start_time = Instant.now().toEpochMilli(); - profile.setBan_list(banService.getBans(steamID)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("ban_list", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setBan_list(banService.getBans(steamID)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("ban_list", (double) end_time / 1000); + return null; + }); } //if(requests.contains("killfeed")){ - start_time = Instant.now().toEpochMilli(); - profile.setKillfeed((new Killfeed()) - .setKills(killfeedService.getKills(steamID, null)) - .setDeads(killfeedService.getDeads(steamID, null)) - .setAssists(killfeedService.getAssists(steamID, null)) - .setSuicides(killfeedService.getSuicides(steamID, null))); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("killfeed", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setKillfeed((new Killfeed()) + .setKills(killfeedService.getKills(steamID, null)) + .setDeads(killfeedService.getDeads(steamID, null)) + .setAssists(killfeedService.getAssists(steamID, null)) + .setSuicides(killfeedService.getSuicides(steamID, null))); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("killfeed", (double) end_time / 1000); + return null; + }); //} //if(requests.contains("messages")){ - start_time = Instant.now().toEpochMilli(); - profile.setMessages(messageService.getMessageCount(steamID, null)); - end_time = Instant.now().toEpochMilli() - start_time; - profile.getResponse_time().put("messages", Double.valueOf(end_time) / 1000); + callables.add(() -> { + long start_time = Instant.now().toEpochMilli(); + profile.setMessages(messageService.getMessageCount(steamID, null)); + long end_time = Instant.now().toEpochMilli() - start_time; + profile.getResponse_time().put("messages", (double) end_time / 1000); + return null; + }); //} + try { + executor.invokeAll(callables); + } catch (InterruptedException ie) {} + finally { + profile.getResponse_time().put("total", (double) (Instant.now().toEpochMilli() - start_time_total) / 1000); + executor.shutdown(); + } return profile; } diff --git a/src/main/java/app/services/db/KillfeedService.java b/src/main/java/app/services/db/KillfeedService.java index 1fa64fe..bc5a3c9 100644 --- a/src/main/java/app/services/db/KillfeedService.java +++ b/src/main/java/app/services/db/KillfeedService.java @@ -37,6 +37,13 @@ public class KillfeedService { .getSingleResult(); } + public List getTopKills(String server_id) { + List result = entityManager.createNativeQuery("SELECT `attacker_id`, count(*) as c, `server_id` FROM `user_killfeed` WHERE `victim_id` != `attacker_id` AND `attacker_id` != 0 AND `server_id` LIKE ?1 GROUP BY `attacker_id` ORDER BY `c` DESC LIMIT 10") + .setParameter(1, server_id==null||server_id.isEmpty()?'%':server_id) + .getResultStream().map(obj -> new TopInFeed(obj)).toList(); + return result.stream().peek(kif -> kif.setNicknames(nicknameService)).toList(); + } + public Long getDeads(SteamID steamID, String server_id) { return (Long) entityManager.createNativeQuery("SELECT COUNT(*) FROM `user_killfeed` WHERE `victim_id` = ?1 AND `attacker_id` != `victim_id` AND `server_id` like ?2") .setParameter(1, steamID.account_id) @@ -44,6 +51,13 @@ public class KillfeedService { .getSingleResult(); } + public List getTopDeads(String server_id) { + List result = entityManager.createNativeQuery("SELECT `attacker_id`, count(*) as c, `server_id` FROM `user_killfeed` WHERE `victim_id` != `attacker_id` AND `victim_id` != 0 AND `server_id` LIKE ?1 GROUP BY `victim_id` ORDER BY `c` DESC LIMIT 10") + .setParameter(1, server_id==null||server_id.isEmpty()?'%':server_id) + .getResultStream().map(obj -> new TopInFeed(obj)).toList(); + return result.stream().peek(kif -> kif.setNicknames(nicknameService)).toList(); + } + public Long getSuicides(SteamID steamID, String server_id) { return (Long) entityManager.createNativeQuery("SELECT COUNT(*) FROM `user_killfeed` WHERE `victim_id` = ?1 AND `attacker_id` = `victim_id` AND `server_id` like ?2") .setParameter(1, steamID.account_id) @@ -59,7 +73,7 @@ public class KillfeedService { } public Map getPopulateWeapons(SteamID steamID, String server_id, int offset, int limit) { - List result = entityManager.createNativeQuery("SELECT COUNT(u.`weapon_index`) as c, i.`name`, u.`server_id` FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`attacker_id` = ?1 AND `attacker_id` != `victim_id` AND u.server_id like ?2 GROUP BY u.`weapon_index` DESC ORDER BY `c` DESC LIMIT ?3 OFFSET ?4") + List result = entityManager.createNativeQuery("SELECT COUNT(u.`weapon_index`) as c, i.`name`, u.`server_id`, u.`weapon_classname` FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`attacker_id` = ?1 AND `attacker_id` != `victim_id` AND u.server_id like ?2 GROUP BY u.`weapon_index` DESC ORDER BY `c` DESC LIMIT ?3 OFFSET ?4") .setParameter(1, steamID.account_id) .setParameter(2, server_id==null||server_id.isEmpty()?'%':server_id) .setParameter(3, limit) @@ -73,7 +87,7 @@ public class KillfeedService { } public Map getKills(SteamID steamID, String server_id, int offset, int limit) { - List result = entityManager.createNativeQuery("SELECT -1, u.victim_id, u.assister_id, u.utime, i.name, u.server_id FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`attacker_id` = ?1 AND u.`attacker_id` != u.`victim_id` AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") + List result = entityManager.createNativeQuery("SELECT -1, u.victim_id, u.assister_id, u.utime, i.name, u.server_id, u.weapon_classname FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`attacker_id` = ?1 AND u.`attacker_id` != u.`victim_id` AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") .setParameter(1, steamID.account_id) .setParameter(2, server_id==null||server_id.isEmpty()?'%':server_id) .setParameter(3, limit) @@ -87,7 +101,7 @@ public class KillfeedService { } public Map getDeads(SteamID steamID, String server_id, int offset, int limit) { - List result = entityManager.createNativeQuery("SELECT u.attacker_id, -1, u.assister_id, u.utime, i.name, u.server_id FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`victim_id` = ?1 AND u.`attacker_id` != u.`victim_id` AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") + List result = entityManager.createNativeQuery("SELECT u.attacker_id, -1, u.assister_id, u.utime, i.name, u.server_id, u.weapon_classname FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`victim_id` = ?1 AND u.`attacker_id` != u.`victim_id` AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") .setParameter(1, steamID.account_id) .setParameter(2, server_id==null||server_id.isEmpty()?'%':server_id) .setParameter(3, limit) @@ -101,7 +115,7 @@ public class KillfeedService { } public Map getAssists(SteamID steamID, String server_id, int offset, int limit) { - List result = entityManager.createNativeQuery("SELECT u.attacker_id, u.victim_id, -1, u.utime, i.name, u.server_id FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`assister_id` = ?1 AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") + List result = entityManager.createNativeQuery("SELECT u.attacker_id, u.victim_id, -1, u.utime, i.name, u.server_id, u.weapon_classname FROM `user_killfeed` as u INNER JOIN `tf2idb`.`tf2idb_item` as i ON u.`weapon_index` = i.`id` WHERE u.`assister_id` = ?1 AND u.`server_id` like ?2 ORDER BY u.`id` DESC LIMIT ?3 OFFSET ?4") .setParameter(1, steamID.account_id) .setParameter(2, server_id==null||server_id.isEmpty()?'%':server_id) .setParameter(3, limit) @@ -130,6 +144,23 @@ public class KillfeedService { KILL, DEAD, ASSIST } + @Data + class TopInFeed { + int uid; + String name; + long count; + String server_id; + TopInFeed(Object obj) { + this.uid = (int) ((Object[]) obj) [0]; + this.count = (long) ((Object[]) obj) [1]; + this.server_id = (String) ((Object[]) obj) [2]; + } + + public void setNicknames(NicknameService nicknameService) { + this.name = uid <= 0 || server_id.isEmpty() || server_id == null ?"":nicknameService.grabNickname(uid, server_id); + } + } + @Data class KillsInFeed { int attacker_id; @@ -139,7 +170,8 @@ public class KillfeedService { int assister_id; String assister_name; long utime; - String weapon_name; + private String weapon_name; + private String weapon_classname; String server_id; KillsInFeed(Object obj) { this.attacker_id = (int) ((Object[]) obj)[0]; @@ -148,12 +180,20 @@ public class KillfeedService { this.utime = (long) ((Object[]) obj)[3]; this.weapon_name = (String) ((Object[]) obj)[4]; this.server_id = (String) ((Object[]) obj)[5]; + this.weapon_classname = (String) ((Object[]) obj)[6]; } public void setNicknames(NicknameService nicknameService) { - this.attacker_name = attacker_id == -1?"":nicknameService.grabNickname(attacker_id, server_id); - this.victim_name = victim_id == -1?"":nicknameService.grabNickname(victim_id, server_id); - this.assister_name = assister_id == -1?"":nicknameService.grabNickname(assister_id, server_id); + this.attacker_name = attacker_id == -1 || server_id.isEmpty() || server_id == null?"":nicknameService.grabNickname(attacker_id, server_id); + this.victim_name = victim_id == -1 || server_id.isEmpty() || server_id == null?"":nicknameService.grabNickname(victim_id, server_id); + this.assister_name = assister_id == -1 || server_id.isEmpty() || server_id == null?"":nicknameService.grabNickname(assister_id, server_id); + } + + public String getWeapon_name(){ + //obj_sentrygun + if (this.weapon_classname.contains("obj_sentrygun")) + return "Турель " + (this.weapon_classname.replace("obj_sentrygun", "").isEmpty()?"1":this.weapon_classname.replace("obj_sentrygun", "")) + " уровня"; + return this.weapon_name; } } }