package com.hypixel.hytale.protocol.packets.connection;
import com.hypixel.hytale.protocol.Packet;
import com.hypixel.hytale.protocol.io.PacketIO;
import com.hypixel.hytale.protocol.io.ProtocolException;
import com.hypixel.hytale.protocol.io.ValidationResult;
import com.hypixel.hytale.protocol.io.VarInt;
import io.netty.buffer.ByteBuf;
import java.util.Objects;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
public class Disconnect implements Packet {
public static final int PACKET_ID = 1;
public static final boolean IS_COMPRESSED = false;
public static final int NULLABLE_BIT_FIELD_SIZE = 1;
public static final int FIXED_BLOCK_SIZE = 2;
public static final int VARIABLE_FIELD_COUNT = 1;
public static final int VARIABLE_BLOCK_START = 2;
public static final int MAX_SIZE = 16384007;
@Nullable
public String reason;
@Nonnull
public DisconnectType type;
public int getId() {
return 1;
}
public Disconnect() {
this.type = DisconnectType.Disconnect;
}
public Disconnect(@Nullable String reason, @Nonnull DisconnectType type) {
this.type = DisconnectType.Disconnect;
this.reason = reason;
this.type = type;
}
public Disconnect(@Nonnull Disconnect other) {
this.type = DisconnectType.Disconnect;
this.reason = other.reason;
this.type = other.type;
}
@Nonnull
public static Disconnect deserialize(@Nonnull ByteBuf buf, int offset) {
Disconnect obj = new Disconnect();
byte nullBits = buf.getByte(offset);
obj.type = DisconnectType.fromValue(buf.getByte(offset + 1));
int pos = offset + 2;
if ((nullBits & 1) != 0) {
int reasonLen = VarInt.peek(buf, pos);
if (reasonLen < 0) {
throw ProtocolException.negativeLength("Reason", reasonLen);
}
if (reasonLen > 4096000) {
throw ProtocolException.stringTooLong("Reason", reasonLen, 4096000);
}
int reasonVarLen = VarInt.length(buf, pos);
obj.reason = PacketIO.readVarString(buf, pos, PacketIO.UTF8);
int var10000 = pos + reasonVarLen + reasonLen;
}
return obj;
}
public static int computeBytesConsumed(@Nonnull ByteBuf buf, int offset) {
byte nullBits = buf.getByte(offset);
int pos = offset + 2;
if ((nullBits & 1) != 0) {
int sl = VarInt.peek(buf, pos);
pos += VarInt.length(buf, pos) + sl;
}
return pos - offset;
}
public void serialize(@Nonnull ByteBuf buf) {
byte nullBits = 0;
if (this.reason != null) {
nullBits = (byte)(nullBits | 1);
}
buf.writeByte(nullBits);
buf.writeByte(this.type.getValue());
if (this.reason != null) {
PacketIO.writeVarString(buf, this.reason, 4096000);
}
}
public int computeSize() {
int size = 2;
if (this.reason != null) {
size += PacketIO.stringSize(this.reason);
}
return size;
}
public static ValidationResult validateStructure(@Nonnull ByteBuf buffer, int offset) {
if (buffer.readableBytes() - offset < 2) {
return ValidationResult.error("Buffer too small: expected at least 2 bytes");
} else {
byte nullBits = buffer.getByte(offset);
int pos = offset + 2;
if ((nullBits & 1) != 0) {
int reasonLen = VarInt.peek(buffer, pos);
if (reasonLen < 0) {
return ValidationResult.error("Invalid string length for Reason");
}
if (reasonLen > 4096000) {
return ValidationResult.error("Reason exceeds max length 4096000");
}
pos += VarInt.length(buffer, pos);
pos += reasonLen;
if (pos > buffer.writerIndex()) {
return ValidationResult.error("Buffer overflow reading Reason");
}
}
return ValidationResult.OK;
}
}
public Disconnect clone() {
Disconnect copy = new Disconnect();
copy.reason = this.reason;
copy.type = this.type;
return copy;
}
public boolean equals(Object obj) {
if (this == obj) {
return true;
} else if (!(obj instanceof Disconnect)) {
return false;
} else {
Disconnect other = (Disconnect)obj;
return Objects.equals(this.reason, other.reason) && Objects.equals(this.type, other.type);
}
}
public int hashCode() {
return Objects.hash(new Object[]{this.reason, this.type});
}
}