net.gdface.thrift.ThriftyStructMetadata Maven / Gradle / Ivy
package net.gdface.thrift;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static net.gdface.thrift.TypeTransformer.getInstance;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Map;
import java.util.Map.Entry;
import javax.annotation.concurrent.Immutable;
import com.google.common.base.Throwables;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.ImmutableMap;
import com.microsoft.thrifty.Struct;
import com.microsoft.thrifty.StructBuilder;
@Immutable
public class ThriftyStructMetadata {
public static final LoadingCache,ThriftyStructMetadata>
STRUCTS_CACHE =
CacheBuilder.newBuilder().build(
new CacheLoader,ThriftyStructMetadata>(){
@Override
public ThriftyStructMetadata load(Class> key) throws Exception {
return new ThriftyStructMetadata(key);
}});
private final Class> structType;
/** 字段ID对应的字段对象 */
private final ImmutableMap fields;
/** builder对象所有字段的set方法 */
private final ImmutableMap buildSetters;
private final Class extends StructBuilder>> builderClass;
private final Method buildMethod;
@SuppressWarnings("unchecked")
private ThriftyStructMetadata(Class> structType ) {
this.structType = checkNotNull(structType,"struct is null");
checkArgument(Struct.class.isAssignableFrom(structType),
"structType %s not immplement the %s",structType.getName(),Struct.class.getName());
try {
String className = structType.getName() + "$Builder";
builderClass = (Class extends StructBuilder>>) Class.forName(className);
} catch (Exception e) {
Throwables.throwIfUnchecked(e);
throw new RuntimeException(e);
}
ImmutableMap.Builder fieldBuilder = ImmutableMap.builder();
ImmutableMap.Builder methodBuilder = ImmutableMap.builder();
for(Field field:structType.getDeclaredFields()){
if((field.getModifiers() & Modifier.STATIC)==0){
fieldBuilder.put(field.getName(), field);
try {
Method method = builderClass.getMethod(field.getName(), field.getType());
methodBuilder.put(field.getName(), method);
} catch (NoSuchMethodException e) {
throw new RuntimeException(e);
}
}
}
fields = fieldBuilder.build();
buildSetters = methodBuilder.build();
try {
buildMethod = builderClass.getDeclaredMethod("build");
} catch (Exception e) {
Throwables.throwIfUnchecked(e);
throw new RuntimeException(e);
}
}
public ImmutableMap getFields() {
return fields;
}
public Class> getStructType() {
return structType;
}
@SuppressWarnings("unchecked")
public void setValue(Object instance,short id,L value){
checkNotNull(instance,"instance is null");
checkArgument(structType.isInstance(instance),"invalid value type,required %s",structType.getName());
Field field = fields.get(id);
checkNotNull(field,"invalid field id=%s for %s",Short.toString(id),structType.getName());
Class> fieldType = field.getType();
try {
field.setAccessible(true);
if(value == null){
checkArgument(!fieldType.isPrimitive(),
"primitive field %s of %s required not null value",field.getName(),structType.getName());
field.set(instance,null);
}
else{
field.set(instance, getInstance().to(value, (Class) value.getClass(), fieldType));
}
} catch (Exception e) {
Throwables.throwIfUnchecked(e);
throw new RuntimeException(e);
}
}
@SuppressWarnings("unchecked")
private V getValue(Object instance,String name){
checkArgument(structType.isInstance(instance),"invalid value type,required %s",structType.getName());
Field field = fields.get(name);
checkNotNull(field,"invalid field name=%s for %s",name,structType.getName());
try {
return (V) field.get(instance);
} catch (Exception e) {
Throwables.throwIfUnchecked(e);
throw new RuntimeException(e);
}
}
/**
* 根据字段值构造实例
* @param fieldValues
* @return
*/
@SuppressWarnings("unchecked")
public T constructStruct(Map fieldValues){
checkNotNull(fieldValues,"fieldValues is null");
try {
// new Builder()
StructBuilder builder = (StructBuilder) builderClass.newInstance();
for(Entry entry : fieldValues.entrySet()){
Method method = buildSetters.get(entry.getKey());
checkNotNull(method,"method is null");
TypeValue typeValue= entry.getValue();
// 调用Builder的设置方法设置字段值
method.invoke(builder, getInstance().cast(
typeValue.value,
typeValue.type,
fields.get(entry.getKey()).getGenericType()));
}
// build()
return (T) buildMethod.invoke(builder);
} catch (InvocationTargetException e){
Throwables.throwIfUnchecked(e.getCause());
throw new RuntimeException(e.getCause());
}catch (Exception e) {
Throwables.throwIfUnchecked(e);
throw new RuntimeException(e);
}
}
/**
* 从 {@link com.microsoft.thrifty.Struct} 实例中返回所有字段值
* @param instance
* @return
*/
Map getFieldValues(Object instance){
checkNotNull(instance,"instance is null");
checkArgument(structType.isInstance(instance),"invalid value type,required %s",structType.getName());
ImmutableMap.Builder builder = ImmutableMap.builder();
for(Entry entry:fields.entrySet()){
String name = entry.getKey();
Field field = entry.getValue();
builder.put(name, new TypeValue(field.getType(), getValue(instance, name)));
}
return builder.build();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy