Initial project

This commit is contained in:
Timi
2025-07-13 17:58:12 +08:00
parent 6c59beb2a8
commit 791cf9b770
19 changed files with 774 additions and 94 deletions

View File

@ -0,0 +1,36 @@
package com.imyeyu.config;
import java.lang.reflect.Field;
/**
* @author 夜雨
* @version 2024-04-12 00:44
*/
public abstract class BaseConverter<T, K> {
@SuppressWarnings("unchecked")
K serialize0(Field field, Object t) {
return serialize(field, (T) t);
}
@SuppressWarnings("unchecked")
T deserialize0(Field field, Object data) {
return deserialize(field, (K) data);
}
/**
* 当组件值需要写入配置时的转换,默认直接 toString
*
* @param t 组件值类型
* @return 配置值
*/
protected abstract K serialize(Field field, T t);
/**
* 当获取配置并即将设置到组件时调用
*
* @param data 配置值
* @return 返回值
*/
protected abstract T deserialize(Field field, K data);
}

View File

@ -0,0 +1,237 @@
package com.imyeyu.config;
import com.imyeyu.io.IO;
import com.imyeyu.java.TimiJava;
import com.imyeyu.java.bean.timi.TimiCode;
import com.imyeyu.java.bean.timi.TimiException;
import com.imyeyu.java.ref.Ref;
import com.imyeyu.java.ref.RefField;
import org.yaml.snakeyaml.DumperOptions;
import org.yaml.snakeyaml.LoaderOptions;
import org.yaml.snakeyaml.Yaml;
import org.yaml.snakeyaml.constructor.Constructor;
import org.yaml.snakeyaml.representer.Representer;
import java.io.File;
import java.lang.reflect.Array;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
/**
* @author 夜雨
* @version 2024-04-10 00:44
*/
public class ConfigLoader<T> {
private final File file;
private final Yaml yaml;
private final Class<T> clazz;
private final List<RawMapper> rawMappingList = new ArrayList<>();
private final Map<Class<?>, BaseConverter<?, ?>> converterMap = new HashMap<>();
private T value;
private LinkedHashMap<String, Object> rawValue;
public ConfigLoader(String srcPath, Class<T> clazz) {
this(srcPath, srcPath, clazz);
}
public ConfigLoader(String srcPath, String distPath, Class<T> clazz) {
this.clazz = clazz;
LoaderOptions loaderOptions = new LoaderOptions();
Constructor constructor = new Constructor(loaderOptions);
DumperOptions dumperOptions = new DumperOptions();
dumperOptions.setIndent(4);
dumperOptions.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
dumperOptions.setIndicatorIndent(4);
dumperOptions.setDefaultScalarStyle(DumperOptions.ScalarStyle.PLAIN);
dumperOptions.setIndentWithIndicator(true);
Representer representer = new ConfigRepresenter(dumperOptions);
yaml = new Yaml(constructor, representer, dumperOptions);
file = new File(distPath);
if (!file.exists()) {
try {
IO.resourceToDisk(getClass(), srcPath, distPath);
} catch (Exception e) {
throw new TimiException(TimiCode.ERROR, "load default config error", e);
}
}
}
@SuppressWarnings("unchecked")
public T load() {
try {
rawValue = yaml.load(IO.toString(file).replaceAll("\t", " "));
if (clazz == null || clazz.equals(Object.class)) {
value = (T) rawValue;
} else {
// 解析配置对象
rawMappingList.clear();
value = clazz.getDeclaredConstructor().newInstance();
deserializeToValue(value, rawValue, "");
}
return this.value;
} catch (Exception e) {
throw new TimiException(TimiCode.ERROR, "load config error", e);
}
}
public void dump() {
try {
serializeToRawValue();
IO.toFile(file, yaml.dump(rawValue).replaceAll("( ){4}", "\t"));
} catch (Exception e) {
throw new TimiException(TimiCode.ERROR, "dump config error", e);
}
}
public void addConverter(Class<?> type, BaseConverter<?, ?> converter) {
converterMap.put(type, converter);
}
public void removeConverter(Class<?> type) {
converterMap.remove(type);
}
@SuppressWarnings("unchecked")
private void serializeToRawValue() throws Exception {
for (int i = 0; i < rawMappingList.size(); i++) {
RawMapper rawMapper = rawMappingList.get(i);
String[] bukkit = rawMapper.rawStack.split("#");
Map<String, Object> raw = rawValue;
String key;
if (bukkit.length == 1) {
key = bukkit[0];
} else {
for (int j = 0; j < bukkit.length - 1; j++) {
raw = (Map<String, Object>) rawValue.get(bukkit[j]);
}
key = bukkit[bukkit.length - 1];
}
Object value = rawMapper.refField.getGetter().invoke(rawMapper.owner);
// 字段注解序列化
Converter fieldConverter = rawMapper.refField.getType().getAnnotation(Converter.class);
if (fieldConverter != null) {
BaseConverter<?, ?> converter = Ref.newInstance(fieldConverter.value());
raw.put(key, converter.serialize0(rawMapper.refField.getField(), value));
continue;
}
// 类型序列化
if (converterMap.containsKey(rawMapper.refField.getType())) {
BaseConverter<?, ?> converter = converterMap.get(rawMapper.refField.getType());
raw.put(key, converter.serialize0(rawMapper.refField.getField(), value));
continue;
}
// 枚举序列化
if (Enum.class.isAssignableFrom(rawMapper.refField.getType())) {
raw.put(key, rawMapper.refField.getGetter().invoke(rawMapper.owner).toString());
continue;
}
// 默认序列化
raw.put(key, rawMapper.refField.getGetter().invoke(rawMapper.owner));
}
}
@SuppressWarnings("unchecked")
private void deserializeToValue(Object object, Map<?, ?> map, String stack) throws Exception {
for (Map.Entry<?, ?> item : map.entrySet()) {
RefField refField = Ref.field(object.getClass(), item.getKey().toString());
if (!(item.getValue() instanceof Map<?, ?>)) {
// 缓存映射
RawMapper rawMapper = new RawMapper();
rawMapper.owner = object;
rawMapper.refField = refField;
rawMapper.rawStack = stack + item.getKey();
rawMappingList.add(rawMapper);
}
Class<?> type = refField.getType();
// 字段注解反序列化
Converter fieldConverter = refField.getType().getAnnotation(Converter.class);
if (fieldConverter != null) {
BaseConverter<?, ?> converter = Ref.newInstance(fieldConverter.value());
refField.getSetter().invoke(object, converter.deserialize0(refField.getField(), item.getValue()));
continue;
}
// 类型反序列化
if (converterMap.containsKey(type)) {
refField.getSetter().invoke(object, converterMap.get(type).deserialize0(refField.getField(), item.getValue()));
continue;
}
// 默认反序列化
if (item.getValue() instanceof Map<?, ?> valueMap) {
// 子级对象
Object newObject = type.getDeclaredConstructor().newInstance();
refField.getSetter().invoke(object, newObject);
deserializeToValue(newObject, valueMap, stack + item.getKey() + "#");
continue;
}
// 数组
if (item.getValue() instanceof List<?> valueList) {
// 数组
if (type.isArray()) {
Class<?> arrayType = type.getComponentType();
Object array = Array.newInstance(arrayType, valueList.size());
if (arrayType.isPrimitive()) {
// 基本类型数组
if (TimiJava.isNotEmpty(valueList)) {
Method method = Ref.getMethod(valueList.get(0).getClass(), arrayType.getName() + "Value");
for (int i = 0; i < valueList.size(); i++) {
Array.set(array, i, method.invoke(valueList.get(i)));
}
}
} else {
for (int i = 0; i < valueList.size(); i++) {
Array.set(array, i, arrayType.cast(valueList.get(i)));
}
}
refField.getSetter().invoke(object, array);
continue;
}
// 列表
refField.getSetter().invoke(object, valueList);
continue;
}
// 枚举
if (Enum.class.isAssignableFrom(type)) {
refField.getSetter().invoke(object, Ref.toType((Class<Enum<?>>) type, item.getValue().toString()));
continue;
}
refField.getSetter().invoke(object, item.getValue());
}
}
public T getValue() {
return value;
}
public Map<String, Object> getRawValue() {
return rawValue;
}
/**
* @author 夜雨
* @version 2024-04-10 00:45
*/
private static class RawMapper {
Object owner;
RefField refField;
String rawStack;
}
}

View File

@ -0,0 +1,18 @@
package com.imyeyu.config;
import org.yaml.snakeyaml.DumperOptions;
import org.yaml.snakeyaml.nodes.Tag;
import org.yaml.snakeyaml.representer.Representer;
/**
* @author 夜雨
* @version 2024-05-01 14:03
*/
public class ConfigRepresenter extends Representer {
public ConfigRepresenter(DumperOptions options) {
super(options);
this.nullRepresenter = data -> representScalar(Tag.NULL, "");
}
}

View File

@ -0,0 +1,14 @@
package com.imyeyu.config;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* @author 夜雨
* @version 2024-04-26 00:30
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Converter {
Class<? extends BaseConverter<?, ?>> value();
}

View File

@ -0,0 +1,15 @@
package com.imyeyu.config;
import java.lang.reflect.Field;
/**
* @author 夜雨
* @version 2024-04-27 10:38
*/
public abstract class StringConverter<T> extends BaseConverter<T, String> {
@Override
protected String serialize(Field field, T t) {
return t.toString();
}
}

View File

@ -0,0 +1,42 @@
package test;
import lombok.Data;
import java.util.List;
/**
* @author 夜雨
* @version 2024-04-25 15:37
*/
@Data
public class Config {
public enum EnumTest {
TEST1,
TEST2
}
private String str;
private int num;
private Obj obj;
private boolean bool;
private EnumTest enumTest;
private ConfigConverterTest converterTest;
private int[] list;
@Data
public static class Obj {
private String objStr;
private List<String> objList;
}
}

View File

@ -0,0 +1,13 @@
package test;
import lombok.Data;
/**
* @author 夜雨
* @version 2024-04-26 18:08
*/
@Data
public class ConfigConverterTest {
private String value;
}

View File

@ -0,0 +1,106 @@
package test;
import com.imyeyu.config.BaseConverter;
import com.imyeyu.config.ConfigLoader;
import com.imyeyu.io.IO;
import org.junit.jupiter.api.Test;
import java.io.File;
import java.lang.reflect.Field;
import java.util.Map;
/**
* @author 夜雨
* @version 2024-04-25 11:13
*/
public class TestConfig {
private static final String FILE_PATH = "test.yaml";
public <T> ConfigLoader<T> load(Class<T> clazz) {
ConfigLoader<T> loader = new ConfigLoader<>(FILE_PATH, clazz);
loader.load();
return loader;
}
public <T> ConfigLoader<T> reload(Class<T> clazz) {
IO.destroy(new File(FILE_PATH));
return load(clazz);
}
@Test
public void testLoad() {
ConfigLoader<Object> loader = reload(null);
loader.load();
assert loader.getRawValue() != null;
}
@Test
public void testRaw() {
ConfigLoader<Object> loader = reload(Object.class);
Map<String, Object> rawValue = loader.getRawValue();
rawValue.put("str", "str value update");
loader.dump();
loader.load();
rawValue = loader.getRawValue();
assert rawValue.get("str").equals("str value update");
}
@Test
public void testObject() {
ConfigLoader<Config> loader = reload(Config.class);
Config value = loader.getValue();
assert value.getStr().equals("str value");
assert value.getNum() == 123;
assert value.getList().length == 4;
assert value.getObj() != null;
assert value.getObj().getObjStr().equals("test str");
assert value.getObj().getObjList().size() == 2;
value.setStr("dump object test");
value.getObj().getObjList().add(0, "new list item test");
loader.dump();
value = loader.load();
assert value.getStr().equals("dump object test");
assert value.getObj().getObjList().get(0).equals("new list item test");
}
@Test
public void testEnum() {
ConfigLoader<Config> loader = reload(Config.class);
Config value = loader.getValue();
assert value.getEnumTest() == Config.EnumTest.TEST2;
value.setEnumTest(Config.EnumTest.TEST1);
loader.dump();
value = loader.load();
assert value.getEnumTest() == Config.EnumTest.TEST1;
}
@Test
public void testConverter() {
IO.destroy(new File("test_converter.yaml"));
ConfigLoader<Config> loader = new ConfigLoader<>("test_converter.yaml", Config.class);
loader.addConverter(ConfigConverterTest.class, new BaseConverter<ConfigConverterTest, String>() {
@Override
protected String serialize(Field field, ConfigConverterTest configConverterTest) {
return configConverterTest.getValue();
}
@Override
protected ConfigConverterTest deserialize(Field field, String data) {
ConfigConverterTest configConverterTest = new ConfigConverterTest();
configConverterTest.setValue(data);
return configConverterTest;
}
});
Config value = loader.load();
assert value.getConverterTest().getValue().equals("converter value");
value.getConverterTest().setValue("converter test update");
loader.dump();
value = loader.load();
assert value.getConverterTest().getValue().equals("converter test update");
}
}

View File

@ -0,0 +1,16 @@
# comment
str: "str value"
num: 123
bool: true
enum_test: TEST2
obj:
# comment 2
obj_str: test str
obj_list:
- str item0
- str item1
list:
- 0
- 1
- 2
- 3

View File

@ -0,0 +1,17 @@
# comment
str: "str value"
num: 123
bool: true
enum_test: TEST2
converter_test: converter value
obj:
# comment 2
obj_str: test str
obj_list:
- str item0
- str item1
list:
- 0
- 1
- 2
- 3