Итак, я играю с геоинструментами и подумал, что могу проксировать один из их классов доступа к данным и отслеживать, как он используется в их коде.
Я написал динамический прокси-сервер и завернул в него FeatureSource (интерфейс), и все прошло успешно. Затем я хотел посмотреть на некоторые транзитивные объекты, возвращаемые featureSource, так как главное, что FeatureSource делает, это возвращает FeatureCollection (FeatureSource аналогичен sql DataSource, а featurecollection — оператору sql).
в моем обработчике вызовов я просто передал вызов базовому объекту, распечатав целевой класс/метод/аргументы и результат, когда я пошел, но для вызовов, которые возвращали FeatureCollection (другой интерфейс), я завернул этот объект в свой прокси ( тот же класс, но новый экземпляр, не должно ли это иметь значение, не так ли?) и вернул его. БАМ! Исключение класса:
java.lang.ClassCastException: $Proxy5 cannot be cast to org.geotools.feature.FeatureCollection
at $Proxy4.getFeatures(Unknown Source)
at MyClass.myTestMethod(MyClass.java:295)
код вызова:
FeatureSource<SimpleFeatureType, SimpleFeature> featureSource = ... // create the FS
featureSource = (FeatureSource<SimpleFeatureType, SimpleFeature>) FeatureSourceProxy.newInstance(featureSource, features);
featureSource.getBounds();// ok
featureSource.getSupportedHints();// ok
DefaultQuery query1 = new DefaultQuery(DefaultQuery.ALL);
FeatureCollection<SimpleFeatureType, SimpleFeature> results = featureSource.getFeatures(query1); //<- explosion here
Прокси:
public class FeatureSourceProxy implements java.lang.reflect.InvocationHandler {
private Object target;
private List<SimpleFeature> features;
public static Object newInstance(Object obj, List<SimpleFeature> features) {
return java.lang.reflect.Proxy.newProxyInstance(
obj.getClass().getClassLoader(),
obj.getClass().getInterfaces(),
new FeatureSourceProxy(obj, features)
);
}
private FeatureSourceProxy(Object obj, List<SimpleFeature> features) {
this.target = obj;
this.features = features;
}
public Object invoke(Object proxy, Method m, Object[] args)throws Throwable{
Object result = null;
try {
if("getFeatures".equals(m.getName())){
result = interceptGetFeatures(m, args);
}
else{
result = m.invoke(target, args);
}
}
catch (Exception e) {
throw new RuntimeException("unexpected invocation exception: " + e.getMessage(), e);
}
return result;
}
private Object interceptGetFeatures(Method m, Object[] args) throws Exception{
return newInstance(m.invoke(target, args), features);
}
}
Можно ли динамически возвращать прокси интерфейсов из проксируемого интерфейса или я что-то не так делаю? ваше здоровье!