Control panel started x2
This commit is contained in:
parent
883457522d
commit
d425583a7e
2
.gitignore
vendored
2
.gitignore
vendored
@ -27,6 +27,8 @@ build/
|
||||
### VS Code ###
|
||||
.vscode/
|
||||
|
||||
last.log
|
||||
|
||||
### Mac OS ###
|
||||
.DS_Store
|
||||
/cache/
|
||||
|
@ -277,4 +277,8 @@ public class GameServer {
|
||||
this.deleteSelf(playerId, this.players.get(playerId).getLocationId());
|
||||
this.players.remove(playerId);
|
||||
}
|
||||
|
||||
public int getPlayersCount() {
|
||||
return this.players.size();
|
||||
}
|
||||
}
|
@ -16,6 +16,7 @@ import com.alterdekim.game.repository.BodyPartRepository;
|
||||
import com.alterdekim.game.repository.SmileRepository;
|
||||
import com.alterdekim.game.service.*;
|
||||
import com.alterdekim.game.storage.StorageProperties;
|
||||
import lombok.Getter;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.javatuples.Pair;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
@ -38,6 +39,9 @@ public class StartUpListener {
|
||||
|
||||
private static final String ADMIN_USERNAME = "admin";
|
||||
|
||||
@Getter
|
||||
private long startTime;
|
||||
|
||||
@Autowired
|
||||
private ServerConfig config;
|
||||
|
||||
@ -139,6 +143,7 @@ public class StartUpListener {
|
||||
|
||||
@EventListener
|
||||
public void onApplicationEvent(ContextRefreshedEvent event) {
|
||||
this.startTime = System.currentTimeMillis();
|
||||
|
||||
// todo: compile other swf's
|
||||
if( userService.findByUsername(ADMIN_USERNAME) != null ) {
|
||||
|
@ -0,0 +1,72 @@
|
||||
package com.alterdekim.game.controller;
|
||||
|
||||
import com.alterdekim.game.component.GameServer;
|
||||
import com.alterdekim.game.component.StartUpListener;
|
||||
import com.alterdekim.game.service.UserService;
|
||||
import com.alterdekim.game.utils.DateUtils;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.core.env.Environment;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.RandomAccessFile;
|
||||
import java.nio.file.Path;
|
||||
import java.nio.file.Paths;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.time.Duration;
|
||||
import java.time.temporal.ChronoUnit;
|
||||
import java.util.Date;
|
||||
|
||||
@Slf4j
|
||||
@RestController
|
||||
@RequestMapping("/api")
|
||||
public class ApiController {
|
||||
@Autowired
|
||||
private Environment env;
|
||||
|
||||
@Autowired
|
||||
private StartUpListener startUpListener;
|
||||
|
||||
@Autowired
|
||||
private GameServer gameServer;
|
||||
|
||||
@Autowired
|
||||
private UserService userService;
|
||||
|
||||
@GetMapping("/get_dashboard_logs")
|
||||
public ResponseEntity<byte[]> getLogs() {
|
||||
try {
|
||||
Path logFile = Paths.get(env.getProperty("logging.file.name"));
|
||||
RandomAccessFile file = new RandomAccessFile(logFile.toFile(), "r");
|
||||
int n = 2048;
|
||||
file.seek(logFile.toFile().length() - n);
|
||||
byte[] b = new byte[n];
|
||||
file.read(b, 0, n);
|
||||
return ResponseEntity.ok(b);
|
||||
} catch (IOException e) {
|
||||
log.error("getLogs error: {}", e.getMessage());
|
||||
return ResponseEntity.badRequest().build();
|
||||
}
|
||||
}
|
||||
|
||||
@GetMapping("/get_uptime")
|
||||
public ResponseEntity<String> getUpTime() {
|
||||
return ResponseEntity.ok(
|
||||
DateUtils.prettyPrint(Duration.of(System.currentTimeMillis() - startUpListener.getStartTime(), ChronoUnit.MILLIS))
|
||||
);
|
||||
}
|
||||
|
||||
@GetMapping("/get_players_online")
|
||||
public ResponseEntity<Integer> getPlayersOnline() {
|
||||
return ResponseEntity.ok(gameServer.getPlayersCount());
|
||||
}
|
||||
|
||||
@GetMapping("/get_users_count")
|
||||
public ResponseEntity<Long> getUsersCount() {
|
||||
return ResponseEntity.ok(userService.getUsersCount());
|
||||
}
|
||||
}
|
@ -34,6 +34,7 @@ public class FileServerController {
|
||||
case "css" -> MediaType.parseMediaType("text/css");
|
||||
case "_js" -> MediaType.parseMediaType("text/javascript");
|
||||
case "img" -> MediaType.parseMediaType("image/jpeg");
|
||||
case "svg" -> MediaType.parseMediaType("image/svg+xml");
|
||||
default -> MediaType.TEXT_PLAIN;
|
||||
}).body(Files.readAllBytes(path));
|
||||
} catch (Exception e) {
|
||||
|
@ -14,10 +14,17 @@ import org.springframework.ui.Model;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.PostMapping;
|
||||
import org.springframework.web.bind.annotation.RequestBody;
|
||||
import org.springframework.web.bind.annotation.RequestParam;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.io.RandomAccessFile;
|
||||
import java.net.URLDecoder;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.nio.file.Path;
|
||||
import java.nio.file.Paths;
|
||||
|
||||
import static com.alterdekim.game.controller.FileServerController.URL_PATH;
|
||||
|
||||
@Slf4j
|
||||
@Controller
|
||||
@ -43,7 +50,23 @@ public class StaticController {
|
||||
}
|
||||
|
||||
@GetMapping("/panel")
|
||||
public String panel() {
|
||||
public String panel(Model model, @RequestParam(value = "section", defaultValue = "Dashboard") PanelSection section) {
|
||||
model.addAttribute("section", section.name());
|
||||
model.addAttribute("script", "/" + URL_PATH + "/_js/" + section.name().toLowerCase() + ".js");
|
||||
return "panel";
|
||||
}
|
||||
|
||||
public enum PanelSection {
|
||||
Dashboard,
|
||||
Locations,
|
||||
Games,
|
||||
Preloaders,
|
||||
Users,
|
||||
Banlist,
|
||||
Cache,
|
||||
Goods,
|
||||
Items,
|
||||
Actions,
|
||||
Support
|
||||
}
|
||||
}
|
||||
|
@ -2,10 +2,14 @@ package com.alterdekim.game.repository;
|
||||
|
||||
import com.alterdekim.game.entity.User;
|
||||
import org.springframework.data.jpa.repository.JpaRepository;
|
||||
import org.springframework.data.jpa.repository.Query;
|
||||
import org.springframework.stereotype.Repository;
|
||||
|
||||
@Repository
|
||||
public interface UserRepository extends JpaRepository<User, Long> {
|
||||
User findByUsername(String username);
|
||||
|
||||
@Query(value = "SELECT COUNT(u) FROM User u")
|
||||
Long findUsersCount();
|
||||
}
|
||||
|
||||
|
@ -35,7 +35,8 @@ public class SpringSecurity {
|
||||
authorize
|
||||
.requestMatchers("/").permitAll()
|
||||
.requestMatchers("/login").permitAll()
|
||||
.requestMatchers("/panel").hasAnyAuthority("ROLE_USER") // todo: make several roles and separate them
|
||||
.requestMatchers("/panel").permitAll() // todo: make several roles and separate them
|
||||
.requestMatchers("/api/**").permitAll() // todo: make several roles and separate them
|
||||
.requestMatchers("/main").hasAnyAuthority("ROLE_USER")
|
||||
.requestMatchers("/"+ FileServerController.URL_PATH +"/**").permitAll()
|
||||
.requestMatchers("/ConstructorACHandler.ashx").permitAll()
|
||||
|
@ -261,4 +261,8 @@ public class UserService {
|
||||
public void removeUser(long userId) {
|
||||
this.userRepository.deleteById(userId);
|
||||
}
|
||||
|
||||
public Long getUsersCount() {
|
||||
return this.userRepository.findUsersCount();
|
||||
}
|
||||
}
|
||||
|
12
src/main/java/com/alterdekim/game/utils/DateUtils.java
Normal file
12
src/main/java/com/alterdekim/game/utils/DateUtils.java
Normal file
@ -0,0 +1,12 @@
|
||||
package com.alterdekim.game.utils;
|
||||
|
||||
import java.time.Duration;
|
||||
|
||||
public class DateUtils {
|
||||
public static String prettyPrint(Duration duration) {
|
||||
return duration.toString()
|
||||
.substring(2)
|
||||
.replaceAll("(\\d[HMS])(?!$)", "$1 ")
|
||||
.toLowerCase();
|
||||
}
|
||||
}
|
@ -1,40 +1,78 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en" xmlns:th="http://www.w3.org/1999/xhtml">
|
||||
<head>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<meta charset="UTF-8">
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<meta name="color-scheme" content="light dark">
|
||||
<link rel="stylesheet" href="/file/css/pico.min.css">
|
||||
<script src="/file/_js/jquery-3.7.1.min.js"></script>
|
||||
<title>WhimsyWorld</title>
|
||||
<link rel="stylesheet" type="text/css" href="/file/css/mvp.css"/>
|
||||
<style>
|
||||
aside > nav > ul > li {
|
||||
font-size: 18px;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<header>
|
||||
<nav>
|
||||
<h1><a href="/panel">WhimsyWorld</a></h1>
|
||||
<ul>
|
||||
<li>Dashboard</li>
|
||||
<li><a href="#">Users</a></li>
|
||||
<li>
|
||||
<a href="#">Learn more</a>
|
||||
<ul>
|
||||
<li><a href="#">About us</a></li>
|
||||
<li><a href="#">Contact us</a></li>
|
||||
<li><a href="#">Get help</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</nav>
|
||||
<article>
|
||||
<aside>
|
||||
<center>System <b>warning</b> message</center>
|
||||
</aside>
|
||||
</article>
|
||||
<br><br>
|
||||
<h1>Where god <mark>thrive</mark> with <code>code</code>.</h1>
|
||||
<p>We are world's <b>#1</b> idea building company 💡 coding all night long 👨🏻💻</p>
|
||||
<br>
|
||||
<p><a href="#"><i>Get Quote</i></a><a href="#"><b>Get Started →</b></a></p>
|
||||
<div class="container">
|
||||
<nav>
|
||||
<ul>
|
||||
<li><strong>WhimsyWorld</strong></li>
|
||||
</ul>
|
||||
<ul>
|
||||
<li><a href="https://gitea.awain.net"><img src="/file/svg/github.svg"/></a></li>
|
||||
<li><a href="/logout"><button class="secondary">Logout</button></a></li>
|
||||
</ul>
|
||||
</nav>
|
||||
</div>
|
||||
</header>
|
||||
<main>
|
||||
<main class="container grid" style="grid-template-columns: 1fr 6fr;">
|
||||
<aside>
|
||||
<nav>
|
||||
<ul>
|
||||
<li><a href="/panel">Dashboard</a></li>
|
||||
<li><a href="/panel?section=Locations">Locations</a></li>
|
||||
<li><a href="/panel?section=Games">Games</a></li>
|
||||
<li><a href="/panel?section=Preloaders">Preloaders</a></li>
|
||||
<li><a href="/panel?section=Users">Users</a></li>
|
||||
<li><a href="/panel?section=Banlist">Banlist</a></li>
|
||||
<li><a href="/panel?section=Cache">Cache</a></li>
|
||||
<li><a href="/panel?section=Goods">Goods</a></li>
|
||||
<li><a href="/panel?section=Items">Items list</a></li>
|
||||
<li><a href="/panel?section=Actions">Actions</a></li>
|
||||
<li><a href="/panel?section=Support">Support</a></li>
|
||||
</ul>
|
||||
</nav>
|
||||
</aside>
|
||||
<div class="container-fluid">
|
||||
<th:block th:switch="${section}">
|
||||
<th:block th:case="Preloaders">
|
||||
<h2>Preloaders</h2>
|
||||
</th:block>
|
||||
<th:block th:case="*">
|
||||
<h2>Dashboard</h2>
|
||||
<div class="grid">
|
||||
<div>
|
||||
<h4>Uptime</h4>
|
||||
<p id="uptime"></p>
|
||||
</div>
|
||||
<div>
|
||||
<h4>Online</h4>
|
||||
<p id="online"></p>
|
||||
</div>
|
||||
<div>
|
||||
<h4>Users count</h4>
|
||||
<p id="users_db"></p>
|
||||
</div>
|
||||
</div>
|
||||
<h3>Logs</h3>
|
||||
<article id="logs_div" aria-busy="true" style="overflow-y: scroll; max-height:750px;">
|
||||
</article>
|
||||
</th:block>
|
||||
</th:block>
|
||||
</div>
|
||||
</main>
|
||||
<script th:src="${script}"></script>
|
||||
</body>
|
||||
</html>
|
Loading…
x
Reference in New Issue
Block a user