Files
Module-HTTPD/src/main/java/dev/plex/request/AbstractServlet.java
T
Telesphoreo 94cb2a98c4 Replace IP-based auth with XenForo OAuth2
Also, resolves the long standing issues #2 and #3
2026-05-17 19:06:39 -04:00

222 lines
7.8 KiB
Java

package dev.plex.request;
import com.google.common.collect.Lists;
import dev.plex.HTTPDModule;
import dev.plex.authentication.AuthenticatedUser;
import dev.plex.authentication.AuthenticationManager;
import dev.plex.authentication.OAuth2Provider;
import dev.plex.logging.Log;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.nio.charset.StandardCharsets;
import java.text.CharacterIterator;
import java.text.StringCharacterIterator;
import java.util.List;
import java.util.Objects;
import lombok.Data;
import org.eclipse.jetty.ee10.servlet.ServletHolder;
public class AbstractServlet extends HttpServlet
{
private final List<Mapping> GET_MAPPINGS = Lists.newArrayList();
public AbstractServlet()
{
for (Method declaredMethod : this.getClass().getDeclaredMethods())
{
declaredMethod.setAccessible(true);
if (declaredMethod.isAnnotationPresent(GetMapping.class))
{
GetMapping getMapping = declaredMethod.getAnnotation(GetMapping.class);
Mapping mapping = new Mapping(declaredMethod, getMapping);
if (declaredMethod.isAnnotationPresent(MappingHeaders.class))
{
mapping.setHeaders(declaredMethod.getAnnotation(MappingHeaders.class));
}
GET_MAPPINGS.add(mapping);
ServletHolder holder = new ServletHolder(this);
String endpoint = getMapping.endpoint();
String pattern = endpoint.endsWith("/") ? endpoint + "*" : endpoint;
HTTPDModule.context.addServlet(holder, pattern);
}
}
}
@Override
public void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException
{
String ipAddress = req.getRemoteAddr();
if (ipAddress.equals("127.0.0.1"))
{
ipAddress = req.getHeader("X-FORWARDED-FOR");
}
String requestPath = getRequestPath(req);
Log.log(ipAddress + " visited endpoint " + requestPath);
/*Enumeration<String> headerz = req.getHeaderNames();
while (headerz.hasMoreElements()) {
String header = headerz.nextElement();
PlexLog.debug("Header: {0} Value {1}", header, req.getHeader(header));
}*/
GET_MAPPINGS.stream().filter(mapping -> endpointMatchesRequest(mapping.getMapping().endpoint(), requestPath)).forEach(mapping ->
{
resp.setCharacterEncoding("UTF-8");
if (mapping.headers != null)
{
for (String headers : mapping.headers.headers())
{
String[] parts = headers.split(";", 2);
if (parts.length == 2)
{
resp.addHeader(parts[0], parts[1]);
}
}
}
if (resp.getContentType() == null)
{
resp.setContentType("text/html; charset=UTF-8");
}
resp.setStatus(HttpServletResponse.SC_OK);
try
{
Object object = mapping.method.invoke(this, req, resp);
if (object != null)
{
resp.getWriter().println(object.toString());
}
}
catch (IOException | IllegalAccessException | InvocationTargetException e)
{
e.printStackTrace();
}
});
}
private static boolean endpointMatchesRequest(String endpoint, String requestPath)
{
String normalizedEndpoint = normalizeEndpoint(endpoint);
if (normalizedEndpoint.equals("/"))
{
return requestPath.equals("/");
}
String endpointPrefix = normalizedEndpoint + "/";
return requestPath.equalsIgnoreCase(normalizedEndpoint) || requestPath.regionMatches(true, 0, endpointPrefix, 0, endpointPrefix.length());
}
private static String normalizeEndpoint(String endpoint)
{
if (endpoint.equals("//"))
{
return "/";
}
if (endpoint.length() > 1 && endpoint.endsWith("/"))
{
return endpoint.substring(0, endpoint.length() - 1);
}
return endpoint;
}
private static String getRequestPath(HttpServletRequest req)
{
String requestPath = req.getRequestURI();
String contextPath = req.getContextPath();
if (contextPath != null && !contextPath.isEmpty() && !contextPath.equals("/") && requestPath.startsWith(contextPath))
{
requestPath = requestPath.substring(contextPath.length());
}
return requestPath.isEmpty() ? "/" : requestPath;
}
public static AuthenticatedUser currentUser(HttpServletRequest request)
{
AuthenticationManager manager = HTTPDModule.getAuthenticationManager();
if (manager == null) return null;
OAuth2Provider provider = manager.provider();
if (provider == null) return null;
return provider.lookup(request);
}
public static AuthenticatedUser currentStaff(HttpServletRequest request)
{
AuthenticatedUser user = currentUser(request);
return (user != null && user.staff()) ? user : null;
}
public static String signInPrompt(String action)
{
return "You must <a class=\"text-primary underline\" href=\"/oauth2/login\">sign in</a> as staff " + action + ".";
}
public static String readFile(InputStream filename)
{
String base = HTTPDModule.template;
String page = readFileReal(filename);
String[] info = page.split("\n", 3);
base = base.replace("${TITLE}", info[0]);
base = base.replace("${ACTIVE_" + info[1] + "}", "active");
base = base.replace("${ACTIVE_HOME}", "");
base = base.replace("${ACTIVE_PLAYERS}", "");
base = base.replace("${ACTIVE_INDEFBANS}", "");
base = base.replace("${ACTIVE_COMMANDS}", "");
base = base.replace("${ACTIVE_PUNISHMENTS}", "");
base = base.replace("${ACTIVE_SCHEMATICS}", "");
base = base.replace("${CONTENT}", info[2]);
return base;
}
public static String readFileReal(InputStream filename)
{
StringBuilder contentBuilder = new StringBuilder();
try
{
BufferedReader in = new BufferedReader(new InputStreamReader(Objects.requireNonNull(filename), StandardCharsets.UTF_8));
String str;
while ((str = in.readLine()) != null)
{
contentBuilder.append(str).append("\n");
}
in.close();
}
catch (IOException ignored)
{
}
return contentBuilder.toString();
}
// Code from https://programming.guide/java/formatting-byte-size-to-human-readable-format.html
public static String formattedSize(long bytes)
{
long absB = bytes == Long.MIN_VALUE ? Long.MAX_VALUE : Math.abs(bytes);
if (absB < 1024)
{
return bytes + " B";
}
long value = absB;
CharacterIterator ci = new StringCharacterIterator("KMGTPE");
for (int i = 40; i >= 0 && absB > 0xfffccccccccccccL >> i; i -= 10)
{
value >>= 10;
ci.next();
}
value *= Long.signum(bytes);
return String.format("%.1f %ciB", value / 1024.0, ci.current());
}
@Data
public static class Mapping
{
private final Method method;
private final GetMapping mapping;
private MappingHeaders headers;
}
}