mirror of
https://github.com/webbukkit/dynmap.git
synced 2024-12-25 18:17:37 +01:00
More work on HttpServer.
This commit is contained in:
parent
bf0edea7e2
commit
226cc5f86c
80
src/main/java/org/dynmap/web/FileHandler.java
Normal file
80
src/main/java/org/dynmap/web/FileHandler.java
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
package org.dynmap.web;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public abstract class FileHandler implements HttpHandler {
|
||||||
|
private byte[] readBuffer = new byte[40960];
|
||||||
|
|
||||||
|
private static Map<String, String> mimes = new HashMap<String, String>();
|
||||||
|
static {
|
||||||
|
mimes.put(".html", "text/html");
|
||||||
|
mimes.put(".htm", "text/html");
|
||||||
|
mimes.put(".js", "text/javascript");
|
||||||
|
mimes.put(".png", "image/png");
|
||||||
|
mimes.put(".css", "text/css");
|
||||||
|
mimes.put(".txt", "text/plain");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final String getMimeTypeFromExtension(String extension) {
|
||||||
|
String m = mimes.get(extension);
|
||||||
|
if (m != null)
|
||||||
|
return m;
|
||||||
|
return "application/octet-steam";
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract InputStream getFileInput(String path);
|
||||||
|
|
||||||
|
protected String getExtension(String path) {
|
||||||
|
int dotindex = path.lastIndexOf('.');
|
||||||
|
if (dotindex > 0)
|
||||||
|
return path.substring(dotindex);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final String formatPath(String path) {
|
||||||
|
int qmark = path.indexOf('?');
|
||||||
|
if (qmark >= 0)
|
||||||
|
path = path.substring(0, qmark);
|
||||||
|
|
||||||
|
if (path.startsWith("/") || path.startsWith("."))
|
||||||
|
return null;
|
||||||
|
if (path.length() == 0)
|
||||||
|
path = getDefaultFilename(path);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected String getDefaultFilename(String path) {
|
||||||
|
return path + "index.html";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
||||||
|
path = formatPath(path);
|
||||||
|
InputStream fileInput = getFileInput(path);
|
||||||
|
if (fileInput == null) {
|
||||||
|
response.statusCode = 404;
|
||||||
|
response.statusMessage = "Not found";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String extension = getExtension(path);
|
||||||
|
|
||||||
|
response.fields.put("Content-Type", getMimeTypeFromExtension(extension));
|
||||||
|
response.fields.put("Connection", "close");
|
||||||
|
OutputStream out = response.getBody();
|
||||||
|
try {
|
||||||
|
int readBytes;
|
||||||
|
while ((readBytes = fileInput.read(readBuffer)) > 0) {
|
||||||
|
out.write(readBuffer, 0, readBytes);
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
fileInput.close();
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
fileInput.close();
|
||||||
|
}
|
||||||
|
}
|
@ -6,6 +6,7 @@ import java.util.HashMap;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
public class HttpResponse {
|
public class HttpResponse {
|
||||||
|
public String version = "1.0";
|
||||||
public int statusCode = 200;
|
public int statusCode = 200;
|
||||||
public String statusMessage = "OK";
|
public String statusMessage = "OK";
|
||||||
public Map<String, String> fields = new HashMap<String, String>();
|
public Map<String, String> fields = new HashMap<String, String>();
|
||||||
|
@ -5,6 +5,7 @@ import java.io.BufferedOutputStream;
|
|||||||
import java.io.BufferedReader;
|
import java.io.BufferedReader;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
@ -54,16 +55,10 @@ public class WebServerRequest extends Thread {
|
|||||||
this.playerList = playerList;
|
this.playerList = playerList;
|
||||||
this.configuration = configuration;
|
this.configuration = configuration;
|
||||||
|
|
||||||
handlers.put("/", new HttpHandler() {
|
handlers.put("/", new FilesystemHandler(mgr.webDirectory));
|
||||||
@Override
|
handlers.put("/tiles/", new FilesystemHandler(mgr.webDirectory));
|
||||||
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
handlers.put("/up/", new ClientUpdateHandler());
|
||||||
response.fields.put("Content-Type", "text/plain");
|
handlers.put("/up/configuration", new ClientConfigurationHandler());
|
||||||
BufferedOutputStream s = new BufferedOutputStream(response.getBody());
|
|
||||||
s.write("Hallo".getBytes());
|
|
||||||
s.flush();
|
|
||||||
s.close();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
handlers.put("/test/", new HttpHandler() {
|
handlers.put("/test/", new HttpHandler() {
|
||||||
@Override
|
@Override
|
||||||
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
||||||
@ -76,26 +71,6 @@ public class WebServerRequest extends Thread {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void writeHttpHeader(BufferedOutputStream out, int statusCode, String statusText) throws IOException {
|
|
||||||
out.write("HTTP/1.0 ".getBytes());
|
|
||||||
out.write(Integer.toString(statusCode).getBytes());
|
|
||||||
out.write((" " + statusText + "\r\n").getBytes());
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void writeHeaderField(BufferedOutputStream out, String name, String value) throws IOException {
|
|
||||||
out.write(name.getBytes());
|
|
||||||
out.write((int) ':');
|
|
||||||
out.write((int) ' ');
|
|
||||||
out.write(value.getBytes());
|
|
||||||
out.write(13);
|
|
||||||
out.write(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void writeEndOfHeaders(BufferedOutputStream out) throws IOException {
|
|
||||||
out.write(13);
|
|
||||||
out.write(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Pattern requestHeaderLine = Pattern.compile("^(\\S+)\\s+(\\S+)\\s+HTTP/(.+)$");
|
private static Pattern requestHeaderLine = Pattern.compile("^(\\S+)\\s+(\\S+)\\s+HTTP/(.+)$");
|
||||||
private static Pattern requestHeaderField = Pattern.compile("^([^:]+):\\s*(.+)$");
|
private static Pattern requestHeaderField = Pattern.compile("^([^:]+):\\s*(.+)$");
|
||||||
private static boolean readRequestHeader(InputStream in, HttpRequest request) throws IOException {
|
private static boolean readRequestHeader(InputStream in, HttpRequest request) throws IOException {
|
||||||
@ -127,6 +102,9 @@ public class WebServerRequest extends Thread {
|
|||||||
public static void writeResponseHeader(OutputStream out, HttpResponse response) throws IOException {
|
public static void writeResponseHeader(OutputStream out, HttpResponse response) throws IOException {
|
||||||
BufferedOutputStream o = new BufferedOutputStream(out);
|
BufferedOutputStream o = new BufferedOutputStream(out);
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("HTTP/");
|
||||||
|
sb.append(response.version);
|
||||||
|
sb.append(" ");
|
||||||
sb.append(response.statusCode);
|
sb.append(response.statusCode);
|
||||||
sb.append(" ");
|
sb.append(" ");
|
||||||
sb.append(response.statusMessage);
|
sb.append(response.statusMessage);
|
||||||
@ -169,8 +147,16 @@ public class WebServerRequest extends Thread {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (response != null) {
|
if (response != null) {
|
||||||
|
if (response.fields.get("Content-Length") == null) {
|
||||||
|
response.fields.put("Content-Length", "0");
|
||||||
|
OutputStream out = response.getBody();
|
||||||
|
if (out != null) {
|
||||||
|
out.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
String connection = response.fields.get("Connection");
|
String connection = response.fields.get("Connection");
|
||||||
if (connection != null && connection.equals("close")) {
|
if (connection == null || connection.equals("close")) {
|
||||||
socket.close();
|
socket.close();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -179,16 +165,6 @@ public class WebServerRequest extends Thread {
|
|||||||
socket.close();
|
socket.close();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
/*debugger.debug("request: " + path);
|
|
||||||
if (path.equals("/up/configuration")) {
|
|
||||||
handleConfiguration(out);
|
|
||||||
} else if (path.startsWith("/up/")) {
|
|
||||||
handleUp(out, path.substring(3));
|
|
||||||
} else if (path.startsWith("/tiles/")) {
|
|
||||||
handleMapToDirectory(out, path.substring(6), mgr.tileDirectory);
|
|
||||||
} else if (path.startsWith("/")) {
|
|
||||||
handleMapToDirectory(out, path, mgr.webDirectory);
|
|
||||||
}*/
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
try { socket.close(); } catch(IOException ex) { }
|
try { socket.close(); } catch(IOException ex) { }
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
@ -208,12 +184,11 @@ public class WebServerRequest extends Thread {
|
|||||||
} else if (o instanceof Integer || o instanceof Long || o instanceof Float || o instanceof Double) {
|
} else if (o instanceof Integer || o instanceof Long || o instanceof Float || o instanceof Double) {
|
||||||
return o.toString();
|
return o.toString();
|
||||||
} else if (o instanceof LinkedHashMap<?, ?>) {
|
} else if (o instanceof LinkedHashMap<?, ?>) {
|
||||||
@SuppressWarnings("unchecked")
|
LinkedHashMap<?, ?> m = (LinkedHashMap<?, ?>) o;
|
||||||
LinkedHashMap<String, Object> m = (LinkedHashMap<String, Object>) o;
|
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
sb.append("{");
|
sb.append("{");
|
||||||
boolean first = true;
|
boolean first = true;
|
||||||
for (String key : m.keySet()) {
|
for (Object key : m.keySet()) {
|
||||||
if (first)
|
if (first)
|
||||||
first = false;
|
first = false;
|
||||||
else
|
else
|
||||||
@ -226,8 +201,7 @@ public class WebServerRequest extends Thread {
|
|||||||
sb.append("}");
|
sb.append("}");
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
} else if (o instanceof ArrayList<?>) {
|
} else if (o instanceof ArrayList<?>) {
|
||||||
@SuppressWarnings("unchecked")
|
ArrayList<?> l = (ArrayList<?>) o;
|
||||||
ArrayList<Object> l = (ArrayList<Object>) o;
|
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
int count = 0;
|
int count = 0;
|
||||||
for (int i = 0; i < l.size(); i++) {
|
for (int i = 0; i < l.size(); i++) {
|
||||||
@ -241,146 +215,104 @@ public class WebServerRequest extends Thread {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void handleConfiguration(BufferedOutputStream out) throws IOException {
|
|
||||||
|
|
||||||
String s = stringifyJson(configuration.getProperty("web"));
|
public class ClientConfigurationHandler implements HttpHandler {
|
||||||
|
@Override
|
||||||
|
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
||||||
|
String s = stringifyJson(configuration.getProperty("web"));
|
||||||
|
|
||||||
byte[] bytes = s.getBytes();
|
byte[] bytes = s.getBytes();
|
||||||
String dateStr = new Date().toString();
|
String dateStr = new Date().toString();
|
||||||
writeHttpHeader(out, 200, "OK");
|
|
||||||
writeHeaderField(out, "Date", dateStr);
|
response.fields.put("Date", dateStr);
|
||||||
writeHeaderField(out, "Content-Type", "text/plain");
|
response.fields.put("Content-Type", "text/plain");
|
||||||
writeHeaderField(out, "Expires", "Thu, 01 Dec 1994 16:00:00 GMT");
|
response.fields.put("Expires", "Thu, 01 Dec 1994 16:00:00 GMT");
|
||||||
writeHeaderField(out, "Last-modified", dateStr);
|
response.fields.put("Last-modified", dateStr);
|
||||||
writeHeaderField(out, "Content-Length", Integer.toString(bytes.length));
|
response.fields.put("Content-Length", Integer.toString(bytes.length));
|
||||||
writeEndOfHeaders(out);
|
BufferedOutputStream out = new BufferedOutputStream(response.getBody());
|
||||||
out.write(bytes);
|
out.write(s.getBytes());
|
||||||
|
out.flush();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void handleUp(BufferedOutputStream out, String path) throws IOException {
|
public class ClientUpdateHandler implements HttpHandler {
|
||||||
int current = (int) (System.currentTimeMillis() / 1000);
|
@Override
|
||||||
long cutoff = 0;
|
public void handle(String path, HttpRequest request, HttpResponse response) throws IOException {
|
||||||
|
int current = (int) (System.currentTimeMillis() / 1000);
|
||||||
|
long cutoff = 0;
|
||||||
|
|
||||||
if (path.charAt(0) == '/') {
|
if (path.charAt(0) == '/') {
|
||||||
try {
|
try {
|
||||||
cutoff = ((long) Integer.parseInt(path.substring(1))) * 1000;
|
cutoff = ((long) Integer.parseInt(path.substring(1))) * 1000;
|
||||||
} catch (NumberFormatException e) {
|
} catch (NumberFormatException e) {
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
long relativeTime = world.getTime() % 24000;
|
long relativeTime = world.getTime() % 24000;
|
||||||
sb.append(current + " " + relativeTime + "\n");
|
sb.append(current + " " + relativeTime + "\n");
|
||||||
|
|
||||||
Player[] players = playerList.getVisiblePlayers();
|
Player[] players = playerList.getVisiblePlayers();
|
||||||
for (Player player : players) {
|
for (Player player : players) {
|
||||||
sb.append("player " + player.getName() + " " + player.getLocation().getX() + " " + player.getLocation().getY() + " " + player.getLocation().getZ() + "\n");
|
sb.append("player " + player.getName() + " " + player.getLocation().getX() + " " + player.getLocation().getY() + " " + player.getLocation().getZ() + "\n");
|
||||||
}
|
|
||||||
|
|
||||||
TileUpdate[] tileUpdates = mgr.staleQueue.getTileUpdates(cutoff);
|
|
||||||
for (TileUpdate tu : tileUpdates) {
|
|
||||||
sb.append("tile " + tu.tile.getName() + "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
ChatQueue.ChatMessage[] messages = mgr.chatQueue.getChatMessages(cutoff);
|
|
||||||
for (ChatQueue.ChatMessage cu : messages) {
|
|
||||||
sb.append("chat " + cu.playerName + " " + cu.message + "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
debugger.debug("Sending " + players.length + " players, " + tileUpdates.length + " tile-updates, and " + messages.length + " chats. " + path + ";" + cutoff);
|
|
||||||
|
|
||||||
byte[] bytes = sb.toString().getBytes();
|
|
||||||
|
|
||||||
String dateStr = new Date().toString();
|
|
||||||
writeHttpHeader(out, 200, "OK");
|
|
||||||
writeHeaderField(out, "Date", dateStr);
|
|
||||||
writeHeaderField(out, "Content-Type", "text/plain");
|
|
||||||
writeHeaderField(out, "Expires", "Thu, 01 Dec 1994 16:00:00 GMT");
|
|
||||||
writeHeaderField(out, "Last-modified", dateStr);
|
|
||||||
writeHeaderField(out, "Content-Length", Integer.toString(bytes.length));
|
|
||||||
writeEndOfHeaders(out);
|
|
||||||
out.write(bytes);
|
|
||||||
}
|
|
||||||
|
|
||||||
private byte[] readBuffer = new byte[40960];
|
|
||||||
|
|
||||||
public void writeFile(BufferedOutputStream out, String path, InputStream fileInput) throws IOException {
|
|
||||||
int dotindex = path.lastIndexOf('.');
|
|
||||||
String extension = null;
|
|
||||||
if (dotindex > 0)
|
|
||||||
extension = path.substring(dotindex);
|
|
||||||
|
|
||||||
writeHttpHeader(out, 200, "OK");
|
|
||||||
writeHeaderField(out, "Content-Type", getMimeTypeFromExtension(extension));
|
|
||||||
writeHeaderField(out, "Connection", "close");
|
|
||||||
writeEndOfHeaders(out);
|
|
||||||
try {
|
|
||||||
int readBytes;
|
|
||||||
while ((readBytes = fileInput.read(readBuffer)) > 0) {
|
|
||||||
out.write(readBuffer, 0, readBytes);
|
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
|
||||||
fileInput.close();
|
TileUpdate[] tileUpdates = mgr.staleQueue.getTileUpdates(cutoff);
|
||||||
throw e;
|
for (TileUpdate tu : tileUpdates) {
|
||||||
|
sb.append("tile " + tu.tile.getName() + "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
ChatQueue.ChatMessage[] messages = mgr.chatQueue.getChatMessages(cutoff);
|
||||||
|
for (ChatQueue.ChatMessage cu : messages) {
|
||||||
|
sb.append("chat " + cu.playerName + " " + cu.message + "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
debugger.debug("Sending " + players.length + " players, " + tileUpdates.length + " tile-updates, and " + messages.length + " chats. " + path + ";" + cutoff);
|
||||||
|
|
||||||
|
byte[] bytes = sb.toString().getBytes();
|
||||||
|
|
||||||
|
String dateStr = new Date().toString();
|
||||||
|
response.fields.put("Date", dateStr);
|
||||||
|
response.fields.put("Content-Type", "text/plain");
|
||||||
|
response.fields.put("Expires", "Thu, 01 Dec 1994 16:00:00 GMT");
|
||||||
|
response.fields.put("Last-modified", dateStr);
|
||||||
|
response.fields.put("Content-Length", Integer.toString(bytes.length));
|
||||||
|
BufferedOutputStream out = new BufferedOutputStream(response.getBody());
|
||||||
|
out.write(bytes);
|
||||||
|
out.flush();
|
||||||
}
|
}
|
||||||
fileInput.close();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getFilePath(String path) {
|
public class JarFileHandler extends FileHandler {
|
||||||
int qmark = path.indexOf('?');
|
private String root;
|
||||||
if (qmark >= 0)
|
public JarFileHandler(String root) {
|
||||||
path = path.substring(0, qmark);
|
if (root.endsWith("/")) root = root.substring(0, root.length()-1);
|
||||||
path = path.substring(1);
|
this.root = root;
|
||||||
|
}
|
||||||
|
@Override
|
||||||
|
protected InputStream getFileInput(String path) {
|
||||||
|
return this.getClass().getResourceAsStream(root + "/" + path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (path.startsWith("/") || path.startsWith("."))
|
public class FilesystemHandler extends FileHandler {
|
||||||
|
private File root;
|
||||||
|
public FilesystemHandler(File root) {
|
||||||
|
if (!root.isDirectory())
|
||||||
|
throw new IllegalArgumentException();
|
||||||
|
this.root = root;
|
||||||
|
}
|
||||||
|
@Override
|
||||||
|
protected InputStream getFileInput(String path) {
|
||||||
|
File file = new File(root, path);
|
||||||
|
if (file.getAbsolutePath().startsWith(root.getAbsolutePath()) && file.isFile()) {
|
||||||
|
try {
|
||||||
|
return new FileInputStream(file);
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
if (path.length() == 0)
|
|
||||||
path = "index.html";
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void handleMapToJar(BufferedOutputStream out, String path) throws IOException {
|
|
||||||
path = getFilePath(path);
|
|
||||||
if (path != null) {
|
|
||||||
InputStream s = this.getClass().getResourceAsStream("/web/" + path);
|
|
||||||
if (s != null) {
|
|
||||||
writeFile(out, path, s);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
writeHttpHeader(out, 404, "Not found");
|
|
||||||
writeEndOfHeaders(out);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void handleMapToDirectory(BufferedOutputStream out, String path, File directory) throws IOException {
|
|
||||||
path = getFilePath(path);
|
|
||||||
if (path != null) {
|
|
||||||
File tileFile = new File(directory, path);
|
|
||||||
|
|
||||||
if (tileFile.getAbsolutePath().startsWith(directory.getAbsolutePath()) && tileFile.isFile()) {
|
|
||||||
FileInputStream s = new FileInputStream(tileFile);
|
|
||||||
writeFile(out, path, s);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
writeHttpHeader(out, 404, "Not found");
|
|
||||||
writeEndOfHeaders(out);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Map<String, String> mimes = new HashMap<String, String>();
|
|
||||||
static {
|
|
||||||
mimes.put(".html", "text/html");
|
|
||||||
mimes.put(".htm", "text/html");
|
|
||||||
mimes.put(".js", "text/javascript");
|
|
||||||
mimes.put(".png", "image/png");
|
|
||||||
mimes.put(".css", "text/css");
|
|
||||||
mimes.put(".txt", "text/plain");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getMimeTypeFromExtension(String extension) {
|
|
||||||
String m = mimes.get(extension);
|
|
||||||
if (m != null)
|
|
||||||
return m;
|
|
||||||
return "application/octet-steam";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user