index.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. const express = require('express')
  2. const app = express()
  3. const port = 9700
  4. const web3operatorAddress = "http://vps.playpoolstudios.com:2015/"
  5. const apiAddress = "https://vps.playpoolstudios.com/metahunt/api/"
  6. app.use(express.json());
  7. app.get('/', (req, res) => {
  8. res.send('Validator is validating')
  9. })
  10. app.listen(port, () => {
  11. console.log(`Mhunt Validator is listening on port ${port}`)
  12. })
  13. app.get('/validateSession', async (req, res) => {
  14. const { tournamentId, address } = req.query;
  15. if (!tournamentId || !address) { res.send("invalid params"); return; }
  16. let tournament = GetTournamentById(tournamentId);
  17. if (tournament == null) {
  18. await CheckForStartedTourneys();
  19. tournament = GetTournamentById(tournamentId);
  20. }
  21. if (tournament == null) {
  22. console.log(`tourney id:${tournamentId} is not available`);
  23. //res.send("This tournament is not either started or valid");
  24. // return;
  25. }
  26. tournament.displayDetails();
  27. const found = tournament.participents.some(participant => participant == address);
  28. if (found) {
  29. return res.send("0");
  30. } else {
  31. return res.send("-1");
  32. }
  33. })
  34. const tournamentTimers = new Map();
  35. app.post('/updateBlock', async (req, res) => {
  36. const jsonData = req.body;
  37. if (!jsonData) {
  38. return res.status(400).send("No JSON data received");
  39. }
  40. const tournamentBlock = new TournamentBlockData(jsonData);
  41. tournamentBlock.minute = timeIndexToMinute(new Date());
  42. if (Object.keys(tournamentBlock.leaderboard).length > 0) {
  43. tournamentBlocks.push(tournamentBlock);
  44. }else{
  45. console.log("Empty leaderboard, probably the initial one. Not pushing");
  46. res.send.status(200);
  47. return;
  48. }
  49. const tournamentData = GetTournamentById(tournamentBlock.tournamentId);
  50. const tournamentEndMinute = timeIndexToMinute(new Date(tournamentData.start_date)) + 10;
  51. if(tournamentBlock.minute >= tournamentEndMinute){
  52. console.log(`this must be the last block ${tournamentBlock.minute}>${tournamentEndMinute}`);
  53. // console.log(`${minuteToTimeIndex(tournamentBlock.minute)}:${minuteToTimeIndex(tournamentEndMinute)}. Time:${(new Date()).toString()}`)
  54. //Final block from this user for this tournament
  55. if (!tournamentTimers.has(tournamentBlock.tournamentId)) {
  56. console.log(`Final block received for tournament ${tournamentBlock.tournamentId}. Starting 5-second timer...`);
  57. // Set a 5-second timer for this tournament
  58. const timer = setTimeout(async () => {
  59. ScanBlocks();
  60. console.log(`Scanned blocks after finishing t:${tournamentBlock.tournamentId}`);
  61. RewardTournament(tournamentBlock.tournamentId);
  62. }, 5000);
  63. // Store the timer in the Map for this tournament
  64. tournamentTimers.set(tournamentBlock.tournamentId, timer);
  65. }else{
  66. }
  67. }
  68. res.sendStatus(200);
  69. });
  70. function timeIndexToMinute(time) {
  71. // Get the total number of milliseconds since the Unix epoch (January 1st, 1970)
  72. const milliseconds = time.getTime();
  73. // Convert milliseconds to minutes (1 minute = 60,000 milliseconds)
  74. const minutes = Math.floor(milliseconds / 60000);
  75. return minutes;
  76. }
  77. function minuteToTimeIndex(minutes) {
  78. // Convert minutes back to milliseconds (1 minute = 60,000 milliseconds)
  79. const milliseconds = minutes * 60000;
  80. // Create a new Date object using the milliseconds since the Unix epoch
  81. const date = new Date(milliseconds);
  82. return date;
  83. }
  84. /* ------------------------- DEV GETS --------------------------------------- */
  85. app.get("/getBlocks", (req,res)=>{
  86. res.setHeader('Content-Type', 'application/json');
  87. res.send(JSON.stringify(tournamentBlocks));
  88. })
  89. app.get("/getTourneys",(req,res)=>{
  90. res.setHeader('Content-Type', 'application/json');
  91. res.send(JSON.stringify(startedTournaments));
  92. })
  93. app.get("/getLeaderboard", (req,res)=>{
  94. res.setHeader('Content-Type', 'application/json');
  95. const {id} = req.query;
  96. if(!id){
  97. res.send(JSON.stringify(confirmedLeaderboards));
  98. }else{
  99. res.send(JSON.stringify(confirmedLeaderboards[id]));
  100. }
  101. })
  102. app.get("/getTournamentEndResults", (req,res)=>{
  103. res.setHeader('Content-Type', 'application/json');
  104. const {id} = req.query;
  105. if(id){
  106. try{
  107. for(const endResult in tournamentEndResults){
  108. if(endResult.id === id){
  109. res.send(JSON.stringify(endResult));
  110. return;
  111. }
  112. }
  113. res.send("0");
  114. }catch{
  115. res.send("-1");
  116. }
  117. }else{
  118. res.send(JSON.stringify(tournamentEndResults));
  119. }
  120. })
  121. app.get("/getTournamentTimers",(req,res)=>{
  122. res.setHeader('Content-Type', 'application/json');
  123. res.send(JSON.stringify(tournamentTimers));
  124. })
  125. /* ------------------------------------------------------------------------------- */
  126. let tournamentsList ;
  127. let startedTournaments;
  128. let tournamentBlocks = [];
  129. let tournamentEndResults=[];
  130. /* ------------------------------------------------------------------------------- */
  131. async function RewardTournament(tournamentId){
  132. const leaderboard = confirmedLeaderboards[tournamentId];
  133. let mostKills =-1;
  134. let topPlayer = "";
  135. for(const playerId in leaderboard){
  136. const playerStat = leaderboard[playerId];
  137. if(playerStat.kills > mostKills){
  138. mostKills =playerStat.kills;
  139. topPlayer = playerId;
  140. }
  141. }
  142. let winnerWallet = "";
  143. for(const block in tournamentBlocks){
  144. if(block.tournamentId == tournamentId && topPlayer == block.owner){
  145. winnerWallet = block.owner;
  146. break;
  147. }
  148. }
  149. try{
  150. const tx = await fetch(web3operatorAddress+`rewardTournament?password=SekretWordHere&tournamentId=${tournamentId}&winnerWallet=${winnerWallet}`);
  151. const endResult = new TournamentEndResult(tournamentId, winnerWallet, tx);
  152. tournamentEndResults.push(endResult);
  153. console.log(`*** Rewarded ${winnerWallet} for tourney : ${tournamentId}`);
  154. }catch{
  155. console.log("*** ERROR REWARDING ***");
  156. }
  157. }
  158. CheckForStartedTourneys();
  159. setInterval(async () => {
  160. CheckForStartedTourneys();
  161. }, 60000)
  162. async function CheckForStartedTourneys() {
  163. try{
  164. const tournamentsResponse = await fetch(apiAddress + "get_tournaments_raw.php");
  165. const tournaments = await tournamentsResponse.json();
  166. tournamentsList = [];
  167. startedTournaments = [];
  168. const now = new Date();
  169. const tenMinutesAgo = new Date(now.getTime() - 10 * 60000); // 10 minutes ago from now
  170. tournaments.forEach(async tournament => {
  171. const tournamentDate = new Date(tournament.start_date); // Converts the string date to a Date object
  172. // Create a new TournamentData instance using the tournament JSON data
  173. const newTournament = new TournamentData(
  174. tournament.id,
  175. tournament.name,
  176. tournament.start_date,
  177. tournament.game_mode,
  178. tournament.reward,
  179. tournament.php_reward,
  180. tournament.is_test,
  181. tournament.ticket_count,
  182. tournament.owner_id
  183. );
  184. if (tournamentDate >= tenMinutesAgo && tournamentDate <= now || true) {
  185. console.log(`Tournament "${tournament.name}" started within the last 10 minutes.`);
  186. try {
  187. const participentsUrl = web3operatorAddress + "getTournamentParticipants?id=" + newTournament.id;
  188. const participentsResponse = await fetch(participentsUrl);
  189. const participentsJson = await participentsResponse.json();
  190. const participents = participentsJson["wallets"].split(',');
  191. participents.forEach(participent => {
  192. newTournament.participents.push(participent);
  193. })
  194. //newTournament.displayDetails();
  195. } catch {
  196. console.log(`tourneyId:${newTournament.id} has no participents. ${JSON.stringify(participents)}`)
  197. }
  198. startedTournaments.push(newTournament);
  199. } else if (tournamentDate > now) {
  200. console.log(`Tournament "${tournament.name}" is yet to come`)
  201. } else {
  202. // console.log(`Tournament "${tournament.name}" is expired`)
  203. }
  204. // newTournament.displayDetails();
  205. // Push the new tournament instance into tournamentsList
  206. tournamentsList.push(newTournament);
  207. });
  208. }catch{
  209. console.log("*** API SERVER IS NOT RESPONDING? ***");
  210. return;
  211. }
  212. }
  213. /* ------------------------------------------------------------------------------- */
  214. startScanBlocksAt30Seconds();
  215. function startScanBlocksAt30Seconds() {
  216. const now = new Date();
  217. const seconds = now.getSeconds();
  218. // Calculate time (in ms) to the next 30th second of the minute
  219. let delay;
  220. if (seconds < 30) {
  221. delay = (30 - seconds) * 1000; // wait till 30th second
  222. } else {
  223. delay = (60 - seconds + 30) * 1000; // wait till next minute's 30th second
  224. }
  225. setTimeout(() => {
  226. // First run at the exact 30th second
  227. ScanBlocks();
  228. // Continue running every 60 seconds after this
  229. setInterval(() => {
  230. ScanBlocks();
  231. }, 60000); // Run every 60 seconds
  232. }, delay);
  233. }
  234. // Define a dictionary to store tournament leaderboards after validation
  235. const confirmedLeaderboards = {};
  236. // Function to scan and analyze blocks
  237. function ScanBlocks() {
  238. const blockValidationCounts = {};
  239. // First, validate each block and count valid ones for each tournament
  240. for (let i = 0; i < tournamentBlocks.length; i++) {
  241. let isValid = true;
  242. // Loop through the started tournaments to validate the block
  243. for (let j = 0; j < startedTournaments.length; j++) {
  244. if (startedTournaments[j].id === tournamentBlocks[i].tournamentId) {
  245. // Found the matching tournament for this block
  246. if (!startedTournaments[j].participants.includes(tournamentBlocks[i].owner)) {
  247. console.log("***Block was sent by non-participant. Allowing this for now.");
  248. isValid = false;
  249. break;
  250. }
  251. // Compare blocks with the same tournamentId and minute number
  252. for (let k = 0; k < tournamentBlocks.length; k++) {
  253. if (
  254. k !== i &&
  255. tournamentBlocks[k].tournamentId === tournamentBlocks[i].tournamentId &&
  256. tournamentBlocks[k].minute === tournamentBlocks[i].minute
  257. ) {
  258. // Found another block with the same tournamentId and minute
  259. if (JSON.stringify(tournamentBlocks[k].leaderboard) !== JSON.stringify(tournamentBlocks[i].leaderboard)) {
  260. console.log(`***Mismatch detected in leaderboard for tournament ${tournamentBlocks[i].tournamentId} at minute ${tournamentBlocks[i].minute}`);
  261. isValid = false;
  262. break;
  263. }
  264. }
  265. }
  266. }
  267. }
  268. if (isValid) {
  269. console.log(`Block ${i + 1} is valid.`);
  270. // Track the valid blocks per tournament
  271. const tournamentId = tournamentBlocks[i].tournamentId;
  272. if (!blockValidationCounts[tournamentId]) {
  273. blockValidationCounts[tournamentId] = { count: 0, total: 0 };
  274. }
  275. blockValidationCounts[tournamentId].count++;
  276. tournamentBlocks[i].validation=1;
  277. } else {
  278. console.log(`Block ${i + 1} is invalid.`);
  279. tournamentBlocks[i].validation=-1;
  280. }
  281. // Track total blocks per tournament for comparison later
  282. const tournamentId = tournamentBlocks[i].tournamentId;
  283. if (!blockValidationCounts[tournamentId]) {
  284. blockValidationCounts[tournamentId] = { count: 0, total: 0 };
  285. }
  286. blockValidationCounts[tournamentId].total++;
  287. }
  288. // After validation, confirm tournaments with more than 50% valid blocks
  289. for (const tournamentId in blockValidationCounts) {
  290. const { count, total } = blockValidationCounts[tournamentId];
  291. if (count > total / 2) {
  292. // More than 50% of blocks are valid, store the leaderboard
  293. let lastValidBlockIndex=-1;
  294. for(let i=0; i<tournamentBlocks.length; i++){
  295. if(tournamentBlocks[i].tournamentId == tournamentId){
  296. if(lastValidBlockIndex>0){
  297. if(tournamentBlocks[lastValidBlockIndex].minute < tournamentBlocks[i].minute){
  298. lastValidBlockIndex=i;
  299. }
  300. }else{
  301. lastValidBlockIndex=i;
  302. }
  303. }
  304. }
  305. const validBlock = tournamentBlocks[lastValidBlockIndex];
  306. if (validBlock) {
  307. if (Object.keys(validBlock.leaderboard).length > 0) {
  308. confirmedLeaderboards[tournamentId] = validBlock.leaderboard;
  309. console.log(`*** Tournament ${tournamentId} confirmed with a valid leaderboard.`);
  310. }else{
  311. console.log(`Block ${validBlock.tournamentId} from ${validBlock.owner} has an empty leaderboard. Ignoring for now`);
  312. }
  313. }else{
  314. console.log(`Could not find a block for t:${tournamentId}`);
  315. }
  316. } else {
  317. console.log(`*** Tournament ${tournamentId} did not meet the 50% valid block requirement.`);
  318. }
  319. }
  320. }
  321. /* ----------------------------------METHODS----------------------------------- */
  322. function GetTournamentById(tournamentId) {
  323. const tournament = startedTournaments.find(tournament => tournament.id == tournamentId);
  324. return tournament;
  325. }
  326. /* ------------------------------------------------------------------------------- */
  327. /* ------------------------------CUSTOM CLASSES------------------------------------ */
  328. class TournamentData {
  329. constructor(id, name, start_date, game_mode, reward, php_reward, is_test, ticket_count, owner_id) {
  330. this.id = id;
  331. this.name = name;
  332. this.start_date = start_date;
  333. this.game_mode = game_mode;
  334. this.reward = reward;
  335. this.php_reward = php_reward;
  336. this.is_test = is_test;
  337. this.ticket_count = ticket_count;
  338. this.owner_id = owner_id;
  339. this.participents = [];
  340. }
  341. // Method to display tournament details
  342. displayDetails() {
  343. console.log(`Tournament ID: ${this.id}`);
  344. console.log(`Name: ${this.name}`);
  345. console.log(`Date: ${this.start_date}`);
  346. console.log(`Game Mode: ${this.game_mode}`);
  347. console.log(`Reward: ${this.reward}`);
  348. console.log(`PHP Reward: ${this.php_reward}`);
  349. console.log(`Test Tournament: ${this.is_test ? "Yes" : "No"}`);
  350. console.log(`Ticket Count: ${this.ticket_count}`);
  351. console.log(`Owner ID: ${this.owner_id}`);
  352. console.log(`Participents: ${this.participents}`);
  353. }
  354. }
  355. // Define the PlayerStat class
  356. class PlayerStat {
  357. constructor(data) {
  358. this.username = data.username || '';
  359. this.xp = data.xp || 0;
  360. this.kills = data.kills || 0;
  361. this.deaths = data.deaths || 0;
  362. this.assists = data.assists || 0;
  363. }
  364. }
  365. // Define the TournamentBlockData class
  366. class TournamentBlockData {
  367. constructor(data) {
  368. console.log(data);
  369. this.tournamentId = data.tournamentId || 0;
  370. this.owner = data.owner || '';
  371. this.owner_username=data.owner_username || '';
  372. this.minute=data.minute;
  373. this.validation =0;
  374. this.leaderboard = {};
  375. // Populate the leaderboard
  376. if (data.leaderboard) {
  377. for (const playerId in data.leaderboard) {
  378. if (data.leaderboard.hasOwnProperty(playerId)) {
  379. this.leaderboard[playerId] = new PlayerStat(data.leaderboard[playerId]);
  380. }
  381. }
  382. }
  383. }
  384. }
  385. class TournamentEndResult{
  386. constructor(id, winner, tx){
  387. this.id=id;
  388. this.winner = winner;
  389. this.tx=tx;
  390. }
  391. }