- reorder stuff in admin.jsp

- add my global ip address to admin whitelist
- add kick functionality to admin.jsp
- admin.jsp redirects to itself after processing actions to avoid refreshing re-processing the action
- ConnectedUsers invalidates the user when removing it
This commit is contained in:
Andy Janata 2012-01-30 23:52:45 -08:00
parent 987c01cc5a
commit 83a807f4b8
6 changed files with 78 additions and 25 deletions

View File

@ -2,19 +2,33 @@
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8" %>
<%@ page import="com.google.inject.Injector" %>
<%@ page import="net.socialgamer.cah.Constants.DisconnectReason" %>
<%@ page import="net.socialgamer.cah.Constants.LongPollEvent" %>
<%@ page import="net.socialgamer.cah.Constants.LongPollResponse" %>
<%@ page import="net.socialgamer.cah.Constants.ReturnableData" %>
<%@ page import="net.socialgamer.cah.StartupUtils" %>
<%@ page import="net.socialgamer.cah.data.ConnectedUsers" %>
<%@ page import="net.socialgamer.cah.data.QueuedMessage" %>
<%@ page import="net.socialgamer.cah.data.QueuedMessage.MessageType" %>
<%@ page import="net.socialgamer.cah.data.User" %>
<%@ page import="java.util.Collection" %>
<%@ page import="java.util.Date" %>
<%@ page import="java.util.HashMap" %>
<%@ page import="java.util.Map" %>
<%
String remoteAddr = request.getRemoteAddr();
if (!(remoteAddr.equals("0:0:0:0:0:0:0:1") || remoteAddr.equals("127.0.0.1"))) {
// TODO better access control than hard-coding IP addresses.
if (!(remoteAddr.equals("0:0:0:0:0:0:0:1") || remoteAddr.equals("127.0.0.1") ||
remoteAddr.equals("98.248.33.90"))) {
response.sendError(403, "Access is restricted to known hosts");
return;
}
ServletContext servletContext = pageContext.getServletContext();
Injector injector = (Injector) servletContext.getAttribute(StartupUtils.INJECTOR);
ConnectedUsers connectedUsers = injector.getInstance(ConnectedUsers.class);
// process verbose toggle
String verboseParam = request.getParameter("verbose");
@ -24,11 +38,29 @@ if (verboseParam != null) {
} else {
servletContext.setAttribute(StartupUtils.VERBOSE_DEBUG, Boolean.FALSE);
}
response.sendRedirect("admin.jsp");
return;
}
// process kick
String kickParam = request.getParameter("kick");
if (kickParam != null) {
User user = connectedUsers.getUser(kickParam);
if (user != null) {
Map<ReturnableData, Object> data = new HashMap<ReturnableData, Object>();
data.put(LongPollResponse.EVENT, LongPollEvent.KICKED.toString());
QueuedMessage qm = new QueuedMessage(MessageType.KICKED, data);
user.enqueueMessage(qm);
connectedUsers.removeUser(user, DisconnectReason.KICKED);
}
response.sendRedirect("admin.jsp");
return;
}
%>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
@ -45,20 +77,16 @@ th, td {
</head>
<body>
<%
Injector injector = (Injector) servletContext.getAttribute(StartupUtils.INJECTOR);
%>
<p>
Server up since
<%
Date startedDate = (Date) servletContext.getAttribute(StartupUtils.DATE_NAME);
long uptime = System.currentTimeMillis() - startedDate.getTime();
uptime /= 1000l;
long seconds = uptime % 60l;
long minutes = (uptime / 60l) % 60l;
long hours = (uptime / 60l / 60l) % 24l;
long days = (uptime / 60l / 60l / 24l);
uptime /= 1000L;
long seconds = uptime % 60L;
long minutes = (uptime / 60L) % 60L;
long hours = (uptime / 60L / 60L) % 24L;
long days = (uptime / 60L / 60L / 24L);
out.print(String.format("%s (%d hours, %02d:%02d:%02d)",
startedDate.toString(), days, hours, minutes, seconds));
%>
@ -72,38 +100,37 @@ Injector injector = (Injector) servletContext.getAttribute(StartupUtils.INJECTOR
<tr>
<td>In Use</td>
<td><% out.print((Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory())
/ 1024 / 1024); %></td>
/ 1024L / 1024L); %></td>
</tr>
<tr>
<td>Free</td>
<td><% out.print(Runtime.getRuntime().freeMemory() / 1024 / 1024); %></td>
<td><% out.print(Runtime.getRuntime().freeMemory() / 1024L / 1024L); %></td>
</tr>
<tr>
<td>JVM Allocated</td>
<td><% out.print(Runtime.getRuntime().totalMemory() / 1024 / 1024); %></td>
<td><% out.print(Runtime.getRuntime().totalMemory() / 1024L / 1024L); %></td>
</tr>
<tr>
<td>JVM Max</td>
<td><% out.print(Runtime.getRuntime().maxMemory() / 1024 / 1024); %></td>
<td><% out.print(Runtime.getRuntime().maxMemory() / 1024L / 1024L); %></td>
</tr>
</table>
<%
ConnectedUsers connectedUsers = injector.getInstance(ConnectedUsers.class);
Collection<User> users = connectedUsers.getUsers();
%>
<br/>
<table>
<tr>
<th>Username</th>
<th>Host</th>
<th>Actions</th>
</tr>
<%
Collection<User> users = connectedUsers.getUsers();
for (User u : users) {
// TODO have a ban system. would need to store them somewhere.
%>
<tr>
<td><% out.print(u.getNickname()); %></td>
<td><% out.print(u.getHostName()); %></td>
<td><a href="?kick=<% out.print(u.getNickname()); %>">Kick</a></td>
</tr>
<%
}

View File

@ -193,13 +193,14 @@ cah.$.LongPollEvent = function() {
// Dummy constructor to make Eclipse auto-complete.
};
cah.$.LongPollEvent.prototype.dummyForAutocomplete = undefined;
cah.$.LongPollEvent.KICKED = "kicked";
cah.$.LongPollEvent.GAME_PLAYER_LEAVE = "game_player_leave";
cah.$.LongPollEvent.NEW_PLAYER = "new_player";
cah.$.LongPollEvent.GAME_PLAYER_JOIN = "game_player_join";
cah.$.LongPollEvent.GAME_LIST_REFRESH = "game_list_refresh";
cah.$.LongPollEvent.GAME_ROUND_COMPLETE = "game_round_complete";
cah.$.LongPollEvent.GAME_PLAYER_INFO_CHANGE = "game_player_info_change";
cah.$.LongPollEvent.NOOP = "noop";
cah.$.LongPollEvent.GAME_PLAYER_INFO_CHANGE = "game_player_info_change";
cah.$.LongPollEvent.GAME_BLACK_RESHUFFLE = "game_black_reshuffle";
cah.$.LongPollEvent.GAME_WHITE_RESHUFFLE = "game_white_reshuffle";
cah.$.LongPollEvent.GAME_STATE_CHANGE = "game_state_change";

View File

@ -38,6 +38,15 @@ cah.longpoll.EventHandlers[cah.$.LongPollEvent.NOOP] = function(data) {
// pass
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.KICKED] = function() {
cah.log.status("You have been kicked by the server administrator.");
cah.longpoll.Resume = false;
$("input").attr("disabled", "disabled");
$("#menubar_left").empty();
$("#main").empty();
$("#info_area").empty();
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.CHAT] = function(data) {
// TODO deal with multiple channels eventually
// don't display our own chat

View File

@ -210,6 +210,7 @@ public class Constants {
GAME_STATE_CHANGE("game_state_change"),
GAME_WHITE_RESHUFFLE("game_white_reshuffle"),
HAND_DEAL("hand_deal"),
KICKED("kicked"),
NEW_PLAYER("new_player"),
NOOP("noop"),
PLAYER_LEAVE("player_leave");

View File

@ -12,6 +12,7 @@ import net.socialgamer.cah.Constants.ReturnableData;
import net.socialgamer.cah.data.QueuedMessage.MessageType;
import com.google.inject.Singleton;
import com.sun.istack.internal.Nullable;
/**
@ -47,11 +48,25 @@ public class ConnectedUsers {
public void removeUser(final User user, final DisconnectReason reason) {
synchronized (users) {
users.remove(user.getNickname());
notifyRemoveUser(user, reason);
if (users.containsValue(user)) {
user.noLongerVaild();
users.remove(user.getNickname());
notifyRemoveUser(user, reason);
}
}
}
/**
* Get the User for the specified nickname, or null if no such user exists.
*
* @param nickname
* @return User, or null.
*/
@Nullable
public User getUser(final String nickname) {
return users.get(nickname);
}
private void notifyRemoveUser(final User user, final DisconnectReason reason) {
// We might also have to tell games about this directly, probably with a listener system.
final HashMap<ReturnableData, Object> data = new HashMap<ReturnableData, Object>();

View File

@ -44,7 +44,7 @@ public class QueuedMessage implements Comparable<QueuedMessage> {
* @author ajanata
*/
public enum MessageType {
PLAYER_EVENT(3), GAME_EVENT(3), GAME_PLAYER_EVENT(4), CHAT(5);
KICKED(1), PLAYER_EVENT(3), GAME_EVENT(3), GAME_PLAYER_EVENT(4), CHAT(5);
private final int weight;