Spring 数据剩余 - PersistentEntityResourceAssembler 中的空指针
Spring Data Rest - Null Pointer in PersistentEntityResourceAssembler
由于空指针,@RepositoryRestResource 返回 500 错误,我遇到了问题。
我有一个名为 Resource
的摘要 class,它使用 InheritanceType.JOINED
和 @JsonSubTypes
来提供 [=11= 的大约 5 个子class ] 例如 ServerResource
。当在它加入的另一个实体中引用时,特定的 Resource
会正常返回。在代码中调用时,PagingAndSortingRepository 也能正常工作。问题是@RepositoryRestResource 为 Resource
实体公开的端点 returns 直接调用时出现 500 错误(由于 NPE)。
调试后发现问题出在PersistentEntityResourceAssembler:154
行。映射对象为空,因为框架无法将 ServerResource.class
映射到父 Resource.class
。缓存是一张地图,并使用.getClass()
。
有什么方法可以为 Resource
为这些不同类型工作的实体获取一个端点?这几乎像是一个错误。
为了纠正这个问题,我覆盖了 resourceMappings()
方法,并注入了我需要的映射。这解决了问题,现在 /resources uri 按预期工作。
package com.environment.config;
import java.lang.reflect.Field;
import java.util.Map;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.type.filter.AssignableTypeFilter;
import org.springframework.data.rest.core.config.RepositoryRestConfiguration;
import org.springframework.data.rest.core.mapping.RepositoryResourceMappings;
import org.springframework.data.rest.core.mapping.ResourceMappings;
import org.springframework.data.rest.core.mapping.ResourceMetadata;
import org.springframework.data.rest.webmvc.config.RepositoryRestMvcConfiguration;
import com.environment.domain.Resource;
import com.environment.util.Tuple;
@Configuration
public class RepositoryConfig extends RepositoryRestMvcConfiguration {
@Override
public ResourceMappings resourceMappings() {
RepositoryResourceMappings rm = RepositoryResourceMappings)super.resourceMappings();
try {
Field cacheField = rm.getClass().getDeclaredField("cache");
cacheField.setAccessible(true);
Map<Class<?>, ResourceMetadata> cache = (Map<Class<?>, ResourceMetadata>) cacheField.get(rm);
//Get metadata for parent object
ResourceMetadata resourceMetadata = cache.entrySet().stream()
.map(Tuple<Class<?>,ResourceMetadata>::new)
.filter(t -> Resource.class.equals(t.t))
.map(t -> t.u)
.findFirst()
.get();
ClassPathScanningCandidateComponentProvider provider = new ClassPathScanningCandidateComponentProvider(false);
provider.addIncludeFilter(new AssignableTypeFilter(Resource.class));
provider.findCandidateComponents("com/environment")
.stream()
.map(beanDef -> {
try {
return Class.forName(beanDef.getBeanClassName());
} catch (Exception e) {
e.printStackTrace();
}
return null;
})
.filter(c -> !Resource.class.equals(c))
.filter(c -> Resource.class.isAssignableFrom(c))
.peek(c -> System.out.println("Mapping: " + c)
.forEach(c -> cache.put((Class<?>) c, resourceMetadata));
} catch (Exception e) {
e.printStackTrace();
}
return rm;
}
}
由于空指针,@RepositoryRestResource 返回 500 错误,我遇到了问题。
我有一个名为 Resource
的摘要 class,它使用 InheritanceType.JOINED
和 @JsonSubTypes
来提供 [=11= 的大约 5 个子class ] 例如 ServerResource
。当在它加入的另一个实体中引用时,特定的 Resource
会正常返回。在代码中调用时,PagingAndSortingRepository 也能正常工作。问题是@RepositoryRestResource 为 Resource
实体公开的端点 returns 直接调用时出现 500 错误(由于 NPE)。
调试后发现问题出在PersistentEntityResourceAssembler:154
行。映射对象为空,因为框架无法将 ServerResource.class
映射到父 Resource.class
。缓存是一张地图,并使用.getClass()
。
有什么方法可以为 Resource
为这些不同类型工作的实体获取一个端点?这几乎像是一个错误。
为了纠正这个问题,我覆盖了 resourceMappings()
方法,并注入了我需要的映射。这解决了问题,现在 /resources uri 按预期工作。
package com.environment.config;
import java.lang.reflect.Field;
import java.util.Map;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.type.filter.AssignableTypeFilter;
import org.springframework.data.rest.core.config.RepositoryRestConfiguration;
import org.springframework.data.rest.core.mapping.RepositoryResourceMappings;
import org.springframework.data.rest.core.mapping.ResourceMappings;
import org.springframework.data.rest.core.mapping.ResourceMetadata;
import org.springframework.data.rest.webmvc.config.RepositoryRestMvcConfiguration;
import com.environment.domain.Resource;
import com.environment.util.Tuple;
@Configuration
public class RepositoryConfig extends RepositoryRestMvcConfiguration {
@Override
public ResourceMappings resourceMappings() {
RepositoryResourceMappings rm = RepositoryResourceMappings)super.resourceMappings();
try {
Field cacheField = rm.getClass().getDeclaredField("cache");
cacheField.setAccessible(true);
Map<Class<?>, ResourceMetadata> cache = (Map<Class<?>, ResourceMetadata>) cacheField.get(rm);
//Get metadata for parent object
ResourceMetadata resourceMetadata = cache.entrySet().stream()
.map(Tuple<Class<?>,ResourceMetadata>::new)
.filter(t -> Resource.class.equals(t.t))
.map(t -> t.u)
.findFirst()
.get();
ClassPathScanningCandidateComponentProvider provider = new ClassPathScanningCandidateComponentProvider(false);
provider.addIncludeFilter(new AssignableTypeFilter(Resource.class));
provider.findCandidateComponents("com/environment")
.stream()
.map(beanDef -> {
try {
return Class.forName(beanDef.getBeanClassName());
} catch (Exception e) {
e.printStackTrace();
}
return null;
})
.filter(c -> !Resource.class.equals(c))
.filter(c -> Resource.class.isAssignableFrom(c))
.peek(c -> System.out.println("Mapping: " + c)
.forEach(c -> cache.put((Class<?>) c, resourceMetadata));
} catch (Exception e) {
e.printStackTrace();
}
return rm;
}
}