package com.hypixel.hytale.codec.codecs.map;
import com.hypixel.hytale.codec.Codec;
import com.hypixel.hytale.codec.ExtraInfo;
import com.hypixel.hytale.codec.WrappedCodec;
import com.hypixel.hytale.codec.codecs.EnumCodec;
import com.hypixel.hytale.codec.exception.CodecException;
import com.hypixel.hytale.codec.schema.SchemaContext;
import com.hypixel.hytale.codec.schema.config.ObjectSchema;
import com.hypixel.hytale.codec.schema.config.Schema;
import com.hypixel.hytale.codec.schema.config.StringSchema;
import com.hypixel.hytale.codec.util.RawJsonReader;
import it.unimi.dsi.fastutil.objects.Object2ObjectLinkedOpenHashMap;
import java.io.IOException;
import java.util.Collections;
import java.util.EnumMap;
import java.util.Map;
import java.util.function.Supplier;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.bson.BsonDocument;
import org.bson.BsonValue;
public class EnumMapCodec<K extends Enum<K>, V> implements Codec<Map<K, V>>, WrappedCodec<V> {
@Nonnull
private final Class<K> clazz;
private final K[] enumConstants;
@Nonnull
private final String[] enumKeys;
private final EnumCodec.EnumStyle enumStyle;
private final Codec<V> codec;
private final Supplier<Map<K, V>> supplier;
private final boolean unmodifiable;
@Nonnull
private final EnumMap<K, String> keyDocumentation;
public EnumMapCodec(@Nonnull Class<K> clazz, Codec<V> codec) {
this(clazz, codec, true);
}
public EnumMapCodec(@Nonnull Class<K> clazz, Codec<V> codec, boolean unmodifiable) {
this(clazz, EnumCodec.EnumStyle.CAMEL_CASE, codec, () -> new EnumMap(clazz), unmodifiable);
}
public EnumMapCodec(@Nonnull Class<K> clazz, Codec<V> codec, Supplier<Map<K, V>> supplier) {
this(clazz, EnumCodec.EnumStyle.CAMEL_CASE, codec, supplier, true);
}
public EnumMapCodec(@Nonnull Class<K> clazz, Codec<V> codec, Supplier<Map<K, V>> supplier, boolean unmodifiable) {
this(clazz, EnumCodec.EnumStyle.CAMEL_CASE, codec, supplier, unmodifiable);
}
public EnumMapCodec(@Nonnull Class<K> clazz, EnumCodec.EnumStyle enumStyle, Codec<V> codec, Supplier<Map<K, V>> supplier, boolean unmodifiable) {
this.clazz = clazz;
this.enumConstants = (K[])(clazz.getEnumConstants());
this.enumStyle = enumStyle;
this.codec = codec;
this.supplier = supplier;
this.unmodifiable = unmodifiable;
this.keyDocumentation = new EnumMap(clazz);
EnumCodec.EnumStyle currentStyle = EnumCodec.EnumStyle.detect(this.enumConstants);
this.enumKeys = new String[this.enumConstants.length];
for(int i = 0; i < this.enumConstants.length; ++i) {
K e = this.enumConstants[i];
this.enumKeys[i] = currentStyle.formatCamelCase(e.name());
}
}
@Nonnull
public EnumMapCodec<K, V> documentKey(K key, String doc) {
this.keyDocumentation.put(key, doc);
return this;
}
public Codec<V> getChildCodec() {
return this.codec;
}
public Map<K, V> decode(@Nonnull BsonValue bsonValue, @Nonnull ExtraInfo extraInfo) {
BsonDocument bsonDocument = bsonValue.asDocument();
Map<K, V> map = (Map)this.supplier.get();
for(Map.Entry<String, BsonValue> entry : bsonDocument.entrySet()) {
String key = (String)entry.getKey();
BsonValue value = (BsonValue)entry.getValue();
K enumKey = this.getEnum(key);
extraInfo.pushKey(key);
try {
map.put(enumKey, this.codec.decode(value, extraInfo));
} catch (Exception e) {
throw new CodecException("Failed to decode", value, extraInfo, e);
} finally {
extraInfo.popKey();
}
}
if (this.unmodifiable) {
map = Collections.unmodifiableMap(map);
}
return map;
}
@Nonnull
public BsonValue encode(@Nonnull Map<K, V> map, ExtraInfo extraInfo) {
BsonDocument bsonDocument = new BsonDocument();
for(Map.Entry<K, V> entry : map.entrySet()) {
BsonValue value = this.codec.encode(entry.getValue(), extraInfo);
if (value != null && !value.isNull() && (!value.isDocument() || !value.asDocument().isEmpty()) && (!value.isArray() || !value.asArray().isEmpty())) {
String var10000;
switch (this.enumStyle) {
case CAMEL_CASE -> var10000 = this.enumKeys[((Enum)entry.getKey()).ordinal()];
case LEGACY -> var10000 = ((Enum)entry.getKey()).name();
default -> throw new MatchException((String)null, (Throwable)null);
}
String key = var10000;
bsonDocument.put(key, value);
}
}
return bsonDocument;
}
public Map<K, V> decodeJson(@Nonnull RawJsonReader reader, @Nonnull ExtraInfo extraInfo) throws IOException {
reader.expect('{');
reader.consumeWhiteSpace();
if (reader.tryConsume('}')) {
return this.unmodifiable ? Collections.emptyMap() : (Map)this.supplier.get();
} else {
Map<K, V> map = (Map)this.supplier.get();
while(true) {
String key = reader.readString();
K enumKey = this.getEnum(key);
reader.consumeWhiteSpace();
reader.expect(':');
reader.consumeWhiteSpace();
extraInfo.pushKey(key, reader);
try {
map.put(enumKey, this.codec.decodeJson(reader, extraInfo));
} catch (Exception e) {
throw new CodecException("Failed to decode", reader, extraInfo, e);
} finally {
extraInfo.popKey();
}
reader.consumeWhiteSpace();
if (reader.tryConsumeOrExpect('}', ',')) {
if (this.unmodifiable) {
map = Collections.unmodifiableMap(map);
}
return map;
}
reader.consumeWhiteSpace();
}
}
}
@Nonnull
public Schema toSchema(@Nonnull SchemaContext context) {
ObjectSchema schema = new ObjectSchema();
schema.getHytale().setType("EnumMap");
schema.setTitle("Map of " + this.clazz.getSimpleName());
StringSchema values = new StringSchema();
schema.setPropertyNames(values);
Map<String, Schema> properties = new Object2ObjectLinkedOpenHashMap<String, Schema>();
schema.setProperties(properties);
Schema childSchema = context.refDefinition(this.codec);
schema.setAdditionalProperties(childSchema);
for(int i = 0; i < this.enumConstants.length; ++i) {
Schema subSchema = context.refDefinition(this.codec);
subSchema.setMarkdownDescription((String)this.keyDocumentation.get(this.enumConstants[i]));
properties.put(this.enumKeys[i], subSchema);
}
values.setEnum(this.enumKeys);
return schema;
}
@Nullable
protected K getEnum(String value) {
return (K)this.enumStyle.match(this.enumConstants, this.enumKeys, value);
}
}