Spring Data Rest - PersistentEntityResourceAssembler中的空指针



我有一个问题,@RepositoryRestResource返回一个500错误由于空指针。

我有一个名为Resource的抽象类,它使用InheritanceType.JOINED@JsonSubTypes来提供Resource的约5个子类,如ServerResource。当在与它连接的另一个实体中引用特定的Resource时,返回良好。在代码中调用PagingAndSortingRepository时也可以正常工作。问题是@RepositoryRestResource为Resource实体暴露的端点在直接调用时返回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;
    }
}

最新更新