GAE + spring linkageError попытался дублировать определение класса для имени

Я разрабатываю веб-приложение с Spring 4.0.5 на GAE. Я использую весенние данные jpa 1.3.5

Когда я добавляю тег @Transactional в метод @Service, во время загрузки я получаю следующую ошибку:

org.springframework.beans.factory.BeanCreationException: ошибка при создании bean-компонента с именем «protectedSiteService», определенным в файле [/development/workspace_experimental/example_app_engine/war/WEB-INF/classes/com/example/web/controller/ProtectedSiteService.class]: Инициализация bean-компонента не удалась; вложенным исключением является org.springframework.aop.framework.AopConfigException: не удалось сгенерировать подкласс CGLIB класса [класс com.example.web.controller.ProtectedSiteService]: общие причины этой проблемы включают использование конечного класса или невидимого класса; вложенным исключением является org.springframework.cglib.core.CodeGenerationException: java.lang.reflect.InvocationTargetException -> null

и предупреждение

2014-08-14 02:59:30.090:WARN::Вложенный в org.springframework.beans.factory.BeanCreationException: ошибка создания bean-компонента с именем «protectedSiteService», определенным в файле [development/workspace_experimental/example_app_engine/war/WEB-INF/ classs/com/example/web/controller/ProtectedSiteService.class]: Инициализация bean-компонента не удалась; вложенным исключением является org.springframework.aop.framework.AopConfigException: не удалось сгенерировать подкласс CGLIB класса [класс com.example.web.controller.ProtectedSiteService]: общие причины этой проблемы включают использование конечного класса или невидимого класса; вложенным исключением является org.springframework.cglib.core.CodeGenerationException: java.lang.reflect.InvocationTargetException --> null: java.lang.LinkageError: загрузчик (экземпляр com/google/appengine/tools/development/IsolatedAppClassLoader): попытка дублирования определение класса для имени: "com/example/web/controller/ProtectedSiteService$$EnhancerBySpringCGLIB$$e2ae408e"

это класс

@Service
public class ProtectedSiteService {

    public ProtectedSiteService() {

    }

    @Autowired
    private UserRepository userRepo;

     @Transactional
     public List<OrganizationMember> method(String name){

         User user = userRepo.findByEmail(name);
         return user.getOrganizationMemberships();
     }

}

Я думаю, что это может быть проблема конфигурации загрузки. Итак, мой web.xml

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE web-app PUBLIC
 "-//Oracle Corporation//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd">

<web-app xmlns="http://java.sun.com/xml/ns/javaee" version="2.5">

    <servlet>
        <servlet-name>dispatcher</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
        <param-name>contextClass</param-name>
        <param-value>
            org.springframework.web.context.support.AnnotationConfigWebApplicationContext
        </param-value>
        </init-param>
        <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>
            com.example.config.WebMvcConfig
        </param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>dispatcher</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

    <!-- Spring basic configurations -->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/spring/spring.xml</param-value>
    </context-param>

    <!-- Spring Security -->
    <filter>
        <filter-name>springSecurityFilterChain</filter-name>
        <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>springSecurityFilterChain</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    <!-- Enconding helper filter -->
    <filter>
        <filter-name>encodingFilter</filter-name>
        <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
        <init-param>
            <param-name>encoding</param-name>
            <param-value>UTF-8</param-value>
        </init-param>
        <init-param>
            <param-name>forceEncoding</param-name>
            <param-value>true</param-value>
        </init-param>
    </filter>
    <filter-mapping>
        <filter-name>encodingFilter</filter-name>
        <servlet-name>dispatcher</servlet-name>
    </filter-mapping>

    <welcome-file-list>
        <welcome-file>public/landingPage.jsp</welcome-file>
    </welcome-file-list>
</web-app>

Мой spring.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
                           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
                           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">

    <!-- Basic Configurations -->
    <context:annotation-config/>

    <context:component-scan base-package="com.qrtify.config"/>


    <!-- SpringMVC -->
    <import resource="spring-mvc.xml"/>

</beans>

В spring-mvc я определил несколько bean-компонентов.

В моем пакете конфигурации у меня есть 3 файла: JpaApplicationConfig, SecurityConfig и WebMvcConfig.

@Configuration
@EnableWebMvc
@ComponentScan(basePackages={"com.example.web.controller", "com.example.models", "com.example.security"})
public class WebMvcConfig extends WebMvcConfigurerAdapter{

    private static final Logger logger = Logger
            .getLogger(WebMvcConfig.class.getName());

    @Override
    public void addArgumentResolvers(
                    List<HandlerMethodArgumentResolver> argumentResolvers) {

            logger.info("ADDING PAGER...");
            PageableArgumentResolver resolver = new PageableArgumentResolver();
            Pageable pageable = new PageRequest(1, 10);
            resolver.setFallbackPageable(pageable);
            argumentResolvers.add(new ServletWebArgumentResolverAdapter(resolver));
    }
}

Это моя конфигурация JPA

@Configuration
@EnableJpaRepositories(basePackages={"com.example.repository"})
@EnableTransactionManagement
public class JpaApplicationConfig {

    private static final Logger logger = Logger
            .getLogger(JpaApplicationConfig.class.getName());

    @Bean
    public LocalContainerEntityManagerFactoryBean entityManagerFactory(){

        Map<String, String> map = new HashMap<String, String>();
        map.put("datanucleus.NontransactionalRead","true");
        map.put("datanucleus.NontransactionalWrite","false");
        map.put("datanucleus.storeManagerType","rdbms");
        map.put("datanucleus.autoCreateSchema" ,"false");
        map.put("datanucleus.validateTables" ,"false");
        map.put("datanucleus.validateConstraints" ,"false");
        map.put("datanucleus.jpa.addClassTransformer" ,"true");
        map.put("datanucleus.singletonEMFForName", "true");

        LocalContainerEntityManagerFactoryBean lce= new LocalContainerEntityManagerFactoryBean();
        lce.setPersistenceProviderClass(org.datanucleus.api.jpa.PersistenceProviderImpl.class);

        DriverManagerDataSource dmds = new DriverManagerDataSource();
        dmds.setDriverClassName("com.mysql.jdbc.Driver");
        dmds.setUrl("jdbc:mysql://localhost:3306/example");
        dmds.setUsername("example");
        dmds.setPassword("example");

        lce.setDataSource(dmds);
        lce.setPackagesToScan("com.example.models");
        lce.setJpaPropertyMap(map);
        lce.setLoadTimeWeaver(new org.springframework.instrument.classloading.SimpleLoadTimeWeaver());
        return lce; 
    }

    @Bean
    public PlatformTransactionManager transactionManager(){
        logger.info("Loading Transaction Manager...");
        JpaTransactionManager txManager = new JpaTransactionManager();
        txManager.setEntityManagerFactory(entityManagerFactory().getObject());
        return txManager;
    }

    /*@Bean
    public PersistenceAnnotationBeanPostProcessor postProcessor(){
        return new PersistenceAnnotationBeanPostProcessor();
    }*/

}

На самом деле я не знаю, где ошибка, я полагаю, что я загружаю какую-то конфигурацию дважды.


person Michelantonio Trizio    schedule 14.08.2014    source источник


Ответы (1)


Я нашел решение. Ответ "использовать интерфейсы".

Я создаю интерфейс для своего сервиса следующим образом:

public interface ProtectedSiteService {

    public List<OrganizationMember> method(String name);
}

а затем я реализовал интерфейс:

@Service
public class ProtectedSiteServiceImp implements ProtectedSiteService {

    public ProtectedSiteServiceImp() {

    }

    @Autowired
    private UserRepository userRepo;

     @Transactional
     public List<OrganizationMember> method(String name){

         User user = userRepo.findByEmail(name);
         return user.getOrganizationMemberships();
     }

}

Наконец, я использую сервис таким образом, вызывая интерфейс с аннотацией Autowired:

@Controller
public class ProtectedSiteController {

    ...

    @Autowired
    private ProtectedSiteService service;

    public ProtectedSiteController() {

    }

    @RequestMapping(method = RequestMethod.GET, value="/afterLogin")
    public String afterLogin(HttpServletRequest request, HttpServletResponse response){

        Authentication auth = SecurityContextHolder.getContext().getAuthentication();

        List<OrganizationMember> memberList = service.method(auth.getName());

        ...

        return redirect;
    }
}
person Michelantonio Trizio    schedule 17.08.2014