Mirror von
https://github.com/GeyserMC/Geyser.git
synchronisiert 2024-12-26 16:12:46 +01:00
Merge branch 'compression' into dev
Dieser Commit ist enthalten in:
Commit
9656caf4cb
@ -0,0 +1,52 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019-2021 GeyserMC. http://geysermc.org
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* @author GeyserMC
|
||||||
|
* @link https://github.com/GeyserMC/Geyser
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.geysermc.geyser.platform.bungeecord;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelOutboundHandlerAdapter;
|
||||||
|
import io.netty.channel.ChannelPromise;
|
||||||
|
import net.md_5.bungee.protocol.packet.LoginSuccess;
|
||||||
|
import net.md_5.bungee.protocol.packet.SetCompression;
|
||||||
|
|
||||||
|
public class GeyserBungeeCompressionDisabler extends ChannelOutboundHandlerAdapter {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
|
||||||
|
if (!(msg instanceof SetCompression)) {
|
||||||
|
if (msg instanceof LoginSuccess) {
|
||||||
|
// We're past the point that compression can be enabled
|
||||||
|
if (ctx.pipeline().get("compress") != null) {
|
||||||
|
ctx.pipeline().remove("compress");
|
||||||
|
}
|
||||||
|
if (ctx.pipeline().get("decompress") != null) {
|
||||||
|
ctx.pipeline().remove("decompress");
|
||||||
|
}
|
||||||
|
ctx.pipeline().remove(this);
|
||||||
|
}
|
||||||
|
super.write(ctx, msg, promise);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -140,6 +140,11 @@ public class GeyserBungeeInjector extends GeyserInjector implements Listener {
|
|||||||
channelInitializer = PipelineUtils.SERVER_CHILD;
|
channelInitializer = PipelineUtils.SERVER_CHILD;
|
||||||
}
|
}
|
||||||
initChannel.invoke(channelInitializer, ch);
|
initChannel.invoke(channelInitializer, ch);
|
||||||
|
|
||||||
|
if (bootstrap.getGeyserConfig().isDisableCompression()) {
|
||||||
|
ch.pipeline().addAfter(PipelineUtils.PACKET_ENCODER, "geyser-compression-disabler",
|
||||||
|
new GeyserBungeeCompressionDisabler());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.childAttr(listener, listenerInfo)
|
.childAttr(listener, listenerInfo)
|
||||||
@ -163,7 +168,7 @@ public class GeyserBungeeInjector extends GeyserInjector implements Listener {
|
|||||||
// If native compression is enabled, then this line is tripped up if a heap buffer is sent over in such a situation
|
// If native compression is enabled, then this line is tripped up if a heap buffer is sent over in such a situation
|
||||||
// as a new direct buffer is not created with that patch (HeapByteBufs throw an UnsupportedOperationException here):
|
// as a new direct buffer is not created with that patch (HeapByteBufs throw an UnsupportedOperationException here):
|
||||||
// https://github.com/SpigotMC/BungeeCord/blob/a283aaf724d4c9a815540cd32f3aafaa72df9e05/native/src/main/java/net/md_5/bungee/jni/zlib/NativeZlib.java#L43
|
// https://github.com/SpigotMC/BungeeCord/blob/a283aaf724d4c9a815540cd32f3aafaa72df9e05/native/src/main/java/net/md_5/bungee/jni/zlib/NativeZlib.java#L43
|
||||||
// This issue could be mitigated down the line by preventing Bungee from setting compression
|
// If disable compression is enabled, this can probably be disabled now, but BungeeCord (not Waterfall) complains
|
||||||
LocalSession.createDirectByteBufAllocator();
|
LocalSession.createDirectByteBufAllocator();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,114 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019-2021 GeyserMC. http://geysermc.org
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* @author GeyserMC
|
||||||
|
* @link https://github.com/GeyserMC/Geyser
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.geysermc.geyser.platform.spigot;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelOutboundHandlerAdapter;
|
||||||
|
import io.netty.channel.ChannelPromise;
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
|
import org.geysermc.geyser.GeyserImpl;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Disables the compression packet (and the compression handlers from being added to the pipeline) for Geyser clients
|
||||||
|
* that won't be receiving the data over the network.
|
||||||
|
*
|
||||||
|
* As of 1.8 - 1.17.1, compression is enabled in the Netty pipeline by adding a listener after a packet is written.
|
||||||
|
* If we simply "cancel" or don't forward the packet, then the listener is never called.
|
||||||
|
*/
|
||||||
|
public class GeyserSpigotCompressionDisabler extends ChannelOutboundHandlerAdapter {
|
||||||
|
static final boolean ENABLED;
|
||||||
|
|
||||||
|
private static final Class<?> COMPRESSION_PACKET_CLASS;
|
||||||
|
private static final Class<?> LOGIN_SUCCESS_PACKET_CLASS;
|
||||||
|
private static final boolean PROTOCOL_SUPPORT_INSTALLED;
|
||||||
|
|
||||||
|
static {
|
||||||
|
PROTOCOL_SUPPORT_INSTALLED = Bukkit.getPluginManager().getPlugin("ProtocolSupport") != null;
|
||||||
|
|
||||||
|
Class<?> compressionPacketClass = null;
|
||||||
|
Class<?> loginSuccessPacketClass = null;
|
||||||
|
boolean enabled = false;
|
||||||
|
try {
|
||||||
|
compressionPacketClass = findCompressionPacket();
|
||||||
|
loginSuccessPacketClass = findLoginSuccessPacket();
|
||||||
|
enabled = true;
|
||||||
|
} catch (Exception e) {
|
||||||
|
GeyserImpl.getInstance().getLogger().error("Could not initialize compression disabler!", e);
|
||||||
|
}
|
||||||
|
COMPRESSION_PACKET_CLASS = compressionPacketClass;
|
||||||
|
LOGIN_SUCCESS_PACKET_CLASS = loginSuccessPacketClass;
|
||||||
|
ENABLED = enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GeyserSpigotCompressionDisabler() {
|
||||||
|
if (!ENABLED) {
|
||||||
|
throw new RuntimeException("Geyser compression disabler cannot be initialized in its current state!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
|
||||||
|
Class<?> msgClass = msg.getClass();
|
||||||
|
// Don't let any compression packet get through
|
||||||
|
if (!COMPRESSION_PACKET_CLASS.isAssignableFrom(msgClass)) {
|
||||||
|
if (LOGIN_SUCCESS_PACKET_CLASS.isAssignableFrom(msgClass)) {
|
||||||
|
if (PROTOCOL_SUPPORT_INSTALLED) {
|
||||||
|
// ProtocolSupport must send the compression packet, so let's remove what it did before it does damage
|
||||||
|
if (ctx.pipeline().get("compress") != null) {
|
||||||
|
ctx.pipeline().remove("compress");
|
||||||
|
}
|
||||||
|
if (ctx.pipeline().get("decompress") != null) {
|
||||||
|
ctx.pipeline().remove("decompress");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// We're past the point that a compression packet can be sent, so we can safely yeet ourselves away
|
||||||
|
ctx.channel().pipeline().remove(this);
|
||||||
|
}
|
||||||
|
super.write(ctx, msg, promise);
|
||||||
|
} else if (PROTOCOL_SUPPORT_INSTALLED) {
|
||||||
|
// We must indicate it "succeeded" or ProtocolSupport will time us out
|
||||||
|
promise.setSuccess();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Class<?> findCompressionPacket() throws ClassNotFoundException {
|
||||||
|
try {
|
||||||
|
return Class.forName("net.minecraft.network.protocol.login.PacketLoginOutSetCompression");
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
String prefix = Bukkit.getServer().getClass().getPackage().getName().replace("org.bukkit.craftbukkit", "net.minecraft.server");
|
||||||
|
return Class.forName(prefix + ".PacketLoginOutSetCompression");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Class<?> findLoginSuccessPacket() throws ClassNotFoundException {
|
||||||
|
try {
|
||||||
|
return Class.forName("net.minecraft.network.protocol.login.PacketLoginOutSuccess");
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
String prefix = Bukkit.getServer().getClass().getPackage().getName().replace("org.bukkit.craftbukkit", "net.minecraft.server");
|
||||||
|
return Class.forName(prefix + ".PacketLoginOutSuccess");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -115,10 +115,14 @@ public class GeyserSpigotInjector extends GeyserInjector {
|
|||||||
|
|
||||||
ChannelFuture channelFuture = (new ServerBootstrap()
|
ChannelFuture channelFuture = (new ServerBootstrap()
|
||||||
.channel(LocalServerChannelWrapper.class)
|
.channel(LocalServerChannelWrapper.class)
|
||||||
.childHandler(new ChannelInitializer<Channel>() {
|
.childHandler(new ChannelInitializer<>() {
|
||||||
@Override
|
@Override
|
||||||
protected void initChannel(Channel ch) throws Exception {
|
protected void initChannel(Channel ch) throws Exception {
|
||||||
initChannel.invoke(childHandler, ch);
|
initChannel.invoke(childHandler, ch);
|
||||||
|
|
||||||
|
if (bootstrap.getGeyserConfig().isDisableCompression() && GeyserSpigotCompressionDisabler.ENABLED) {
|
||||||
|
ch.pipeline().addAfter("encoder", "geyser-compression-disabler", new GeyserSpigotCompressionDisabler());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
// Set to MAX_PRIORITY as MultithreadEventLoopGroup#newDefaultThreadFactory which DefaultEventLoopGroup implements does by default
|
// Set to MAX_PRIORITY as MultithreadEventLoopGroup#newDefaultThreadFactory which DefaultEventLoopGroup implements does by default
|
||||||
|
@ -0,0 +1,84 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019-2021 GeyserMC. http://geysermc.org
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* @author GeyserMC
|
||||||
|
* @link https://github.com/GeyserMC/Geyser
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.geysermc.geyser.platform.velocity;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelOutboundHandlerAdapter;
|
||||||
|
import io.netty.channel.ChannelPromise;
|
||||||
|
import org.geysermc.geyser.GeyserImpl;
|
||||||
|
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
|
||||||
|
public class GeyserVelocityCompressionDisabler extends ChannelOutboundHandlerAdapter {
|
||||||
|
static final boolean ENABLED;
|
||||||
|
private static final Class<?> COMPRESSION_PACKET_CLASS;
|
||||||
|
private static final Class<?> LOGIN_SUCCESS_PACKET_CLASS;
|
||||||
|
private static final Method SET_COMPRESSION_METHOD;
|
||||||
|
|
||||||
|
static {
|
||||||
|
boolean enabled = false;
|
||||||
|
Class<?> compressionPacketClass = null;
|
||||||
|
Class<?> loginSuccessPacketClass = null;
|
||||||
|
Method setCompressionMethod = null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
compressionPacketClass = Class.forName("com.velocitypowered.proxy.protocol.packet.SetCompression");
|
||||||
|
loginSuccessPacketClass = Class.forName("com.velocitypowered.proxy.protocol.packet.ServerLoginSuccess");
|
||||||
|
setCompressionMethod = Class.forName("com.velocitypowered.proxy.connection.MinecraftConnection")
|
||||||
|
.getMethod("setCompressionThreshold", int.class);
|
||||||
|
enabled = true;
|
||||||
|
} catch (Exception e) {
|
||||||
|
GeyserImpl.getInstance().getLogger().error("Could not initialize compression disabler!", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
ENABLED = enabled;
|
||||||
|
COMPRESSION_PACKET_CLASS = compressionPacketClass;
|
||||||
|
LOGIN_SUCCESS_PACKET_CLASS = loginSuccessPacketClass;
|
||||||
|
SET_COMPRESSION_METHOD = setCompressionMethod;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GeyserVelocityCompressionDisabler() {
|
||||||
|
if (!ENABLED) {
|
||||||
|
throw new RuntimeException("Geyser compression disabler cannot be initialized in its current state!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
|
||||||
|
Class<?> msgClass = msg.getClass();
|
||||||
|
if (!COMPRESSION_PACKET_CLASS.isAssignableFrom(msgClass)) {
|
||||||
|
if (LOGIN_SUCCESS_PACKET_CLASS.isAssignableFrom(msgClass)) {
|
||||||
|
// We're past the point that compression can be enabled
|
||||||
|
// Invoke the method as it calls a Netty event and handles removing cleaner than we could
|
||||||
|
Object minecraftConnection = ctx.pipeline().get("handler");
|
||||||
|
SET_COMPRESSION_METHOD.invoke(minecraftConnection, -1);
|
||||||
|
|
||||||
|
ctx.pipeline().remove(this);
|
||||||
|
}
|
||||||
|
super.write(ctx, msg, promise);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -34,6 +34,7 @@ import org.geysermc.geyser.network.netty.GeyserInjector;
|
|||||||
import org.geysermc.geyser.network.netty.LocalServerChannelWrapper;
|
import org.geysermc.geyser.network.netty.LocalServerChannelWrapper;
|
||||||
|
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
public class GeyserVelocityInjector extends GeyserInjector {
|
public class GeyserVelocityInjector extends GeyserInjector {
|
||||||
@ -67,9 +68,23 @@ public class GeyserVelocityInjector extends GeyserInjector {
|
|||||||
workerGroupField.setAccessible(true);
|
workerGroupField.setAccessible(true);
|
||||||
EventLoopGroup workerGroup = (EventLoopGroup) workerGroupField.get(connectionManager);
|
EventLoopGroup workerGroup = (EventLoopGroup) workerGroupField.get(connectionManager);
|
||||||
|
|
||||||
|
// This method is what initializes the connection in Java Edition, after Netty is all set.
|
||||||
|
Method initChannel = ChannelInitializer.class.getDeclaredMethod("initChannel", Channel.class);
|
||||||
|
initChannel.setAccessible(true);
|
||||||
|
|
||||||
ChannelFuture channelFuture = (new ServerBootstrap()
|
ChannelFuture channelFuture = (new ServerBootstrap()
|
||||||
.channel(LocalServerChannelWrapper.class)
|
.channel(LocalServerChannelWrapper.class)
|
||||||
.childHandler(channelInitializer)
|
.childHandler(new ChannelInitializer<>() {
|
||||||
|
@Override
|
||||||
|
protected void initChannel(Channel ch) throws Exception {
|
||||||
|
initChannel.invoke(channelInitializer, ch);
|
||||||
|
|
||||||
|
if (bootstrap.getGeyserConfig().isDisableCompression() && GeyserVelocityCompressionDisabler.ENABLED) {
|
||||||
|
ch.pipeline().addAfter("minecraft-encoder", "geyser-compression-disabler",
|
||||||
|
new GeyserVelocityCompressionDisabler());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
.group(bossGroup, workerGroup) // Cannot be DefaultEventLoopGroup
|
.group(bossGroup, workerGroup) // Cannot be DefaultEventLoopGroup
|
||||||
.childOption(ChannelOption.WRITE_BUFFER_WATER_MARK, serverWriteMark) // Required or else rare network freezes can occur
|
.childOption(ChannelOption.WRITE_BUFFER_WATER_MARK, serverWriteMark) // Required or else rare network freezes can occur
|
||||||
.localAddress(LocalAddress.ANY))
|
.localAddress(LocalAddress.ANY))
|
||||||
|
@ -182,6 +182,8 @@ public interface GeyserConfiguration {
|
|||||||
|
|
||||||
boolean isUseDirectConnection();
|
boolean isUseDirectConnection();
|
||||||
|
|
||||||
|
boolean isDisableCompression();
|
||||||
|
|
||||||
int getConfigVersion();
|
int getConfigVersion();
|
||||||
|
|
||||||
static void checkGeyserConfiguration(GeyserConfiguration geyserConfig, GeyserLogger geyserLogger) {
|
static void checkGeyserConfiguration(GeyserConfiguration geyserConfig, GeyserLogger geyserLogger) {
|
||||||
|
@ -335,6 +335,9 @@ public abstract class GeyserJacksonConfiguration implements GeyserConfiguration
|
|||||||
@JsonProperty("use-direct-connection")
|
@JsonProperty("use-direct-connection")
|
||||||
private boolean useDirectConnection = true;
|
private boolean useDirectConnection = true;
|
||||||
|
|
||||||
|
@JsonProperty("disable-compression")
|
||||||
|
private boolean isDisableCompression = true;
|
||||||
|
|
||||||
@JsonProperty("config-version")
|
@JsonProperty("config-version")
|
||||||
private int configVersion = 0;
|
private int configVersion = 0;
|
||||||
|
|
||||||
|
@ -219,4 +219,9 @@ mtu: 1400
|
|||||||
# If disabled on plugin versions, expect performance decrease and latency increase
|
# If disabled on plugin versions, expect performance decrease and latency increase
|
||||||
use-direct-connection: true
|
use-direct-connection: true
|
||||||
|
|
||||||
|
# Whether Geyser should attempt to disable compression for Bedrock players. This should be a benefit as there is no need to compress data
|
||||||
|
# when Java packets aren't being handled over the network.
|
||||||
|
# This requires use-direct-connection to be true.
|
||||||
|
disable-compression: true
|
||||||
|
|
||||||
config-version: 4
|
config-version: 4
|
||||||
|
Laden…
In neuem Issue referenzieren
Einen Benutzer sperren