весна и претензии интеграции

1

Я пытаюсь реализовать приложение с несколькими ui - vaadin, jsp и т.д.

Он работал с простым jsp но потом решил использовать vaadin как ui.

Я создал ваадин сервлет (и весенний сервлет тоже вышел). Мой web.xml выглядит так

<?xml version="1.0"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
         version="2.5">

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>
            classpath:/pmc-web-context.xml
        </param-value>
    </context-param>

    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

    <servlet>
        <servlet-name>another-pmc-servlet</servlet-name>
        <servlet-class>com.vaadin.server.VaadinServlet</servlet-class>
        <init-param>
            <param-name>UI</param-name>
            <param-value>com.xxxx.app.pmc.vaadin.PmcUi</param-value>
        </init-param>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value></param-value>
        </init-param>
    </servlet>

    <servlet>
        <servlet-name>pmc-servlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value></param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>another-pmc-servlet</servlet-name>
        <url-pattern>/VAADIN/*</url-pattern>
    </servlet-mapping>

    <servlet-mapping>
        <servlet-name>pmc-servlet</servlet-name>
        <url-pattern>/JSP/*</url-pattern>
    </servlet-mapping>

    <servlet>
        <servlet-name>jsp</servlet-name>
        <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>
    </servlet>

    <servlet-mapping>
        <servlet-name>jsp</servlet-name>
        <url-pattern>/WEB-INF/jsp/*</url-pattern>
    </servlet-mapping>

</web-app>

Я создал компонент table vaadin и скорректировал его для моих нужд. Я использовал autowiring для обслуживания.

package com.xxxx.app.pmc.vaadin.components;

import com.xxxx.app.pmc.model.Project;
import com.xxxx.app.pmc.service.project.ProjectService;
import com.vaadin.data.util.IndexedContainer;
import com.vaadin.ui.Table;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.util.List;

@Component("projectTable")
public class ProjectTable extends Table {

    private static final String CAPTION = "Projects";
    private static final String[] headers = { "Project name", "Project owner", "ID" };

    @Autowired
    private ProjectService projectService;

    public Table createTable() {
        this.setContainerDataSource(projectDatasource());
        this.setVisibleColumns(headers);
        this.setSelectable(true);
        this.setImmediate(true);

        return this;
    }

    public IndexedContainer projectDatasource() {
        IndexedContainer indexedContainer = new IndexedContainer();

        for(String header: headers) {
            indexedContainer.addContainerProperty(header, String.class, "");
        }

        List<Project> projects = projectService.findAllProjects();

        for(int i = 0; i < projects.size(); i++) {
            Object id = indexedContainer.addItem();
            Project item = projects.get(i);
            indexedContainer.getContainerProperty(id, headers[0]).setValue(item.getProjectName());
            indexedContainer.getContainerProperty(id, headers[1]).setValue(item.getProjectOwner());
            indexedContainer.getContainerProperty(id, headers[1]).setValue(item.getProjectId());
        }

        return indexedContainer;
    }

}

ProjectService является весенним бобом.

@Service("projectService")
public class ProjectService {

    @Autowired
    private ProjectRepository projectRepository;

    public void insertProject(Project project) {
        projectRepository.store(project);
    }

    public List<Project> findAllProjects() {
        return projectRepository.getAllItems();
    }

    public Project getProject(String id) {
        return projectRepository.get(id);
    }

}

ProjectRepository - еще один весенний боб. Он использует SqlSessionTemplate боб из MyBatis.

@Repository("projectRepository")
public class ProjectRepository implements IRepository<Project> {

    private static final String STORE_PROJECT = "Project.insertProject";
    private static final String GET_PROJECT_BY_ID = "Project.getProjectById";
    private static final String GET_PROJECT_LIST = "Project.getProjectList";

    @Autowired
    private SqlSessionTemplate sqlSessionTemplate;

    @Override
    public void store(Project object) {
        sqlSessionTemplate.insert(STORE_PROJECT, object);
    }

    @Override
    public Project get(String id) {
        return sqlSessionTemplate.selectOne(GET_PROJECT_BY_ID, id);
    }

    @Override
    public List<Project> getAllItems() {
        return sqlSessionTemplate.selectList(GET_PROJECT_LIST);
    }

} 

Когда я написал приложение с использованием controller Spring (используя JSP), он работал нормально. Но когда я добавил vaadin - JSP перестала работать, а приложение vaadin выбрасывает NullPointerException для ProjectService, ProjectRepository... все бобы, которые я использую.

В чем проблема? Все контекстные файлы контекста xml просты.

<import resource="classpath:com/xxxx/app/pmc/pmc-service-context.xml"/>
<context:component-scan base-package="com.xxx.app.pmc"/>

И мой pmc-web-context.xml тоже имеет такие строки.

<mvc:annotation-driven/>
<context:annotation-config/>

Он отлично работал с JSP, поэтому я думаю, что проблема заключается не в самих весенних декларациях, а в интеграции весны в мой ваадин.

Как его решить? Когда я, например, создал объект ProjectTable вручную - он выдает NullPointerException для ProjectService. Когда я создаю ProjectService вручную - он выдает NullPointerException для ProjectRepository и так далее. Кажется, autowiring просто не работает.

PS забыл добавить свой код пользовательского интерфейса

package com.xxxx.app.pmc.vaadin;

import com.xxxx.app.pmc.vaadin.components.ProjectTable;
import com.vaadin.annotations.Title;
import com.vaadin.server.VaadinRequest;
import com.vaadin.ui.UI;
import com.vaadin.ui.VerticalLayout;

import org.springframework.stereotype.Component;

@Title("PMC")
@Component("pmcVaadin")
public class PmcUi extends UI {

    @Override
    protected void init(VaadinRequest request) {
        VerticalLayout mainLayout = new VerticalLayout();
        ProjectTable projectTable = new ProjectTable();
        mainLayout.addComponent(projectTable.createTable());
        mainLayout.setSizeFull();
        setContent(mainLayout);
    }

}
Теги:
servlets
spring
vaadin

2 ответа

0

Довольно простой вариант - использовать Configurable support. Он выполнит некоторую магию аспекта, и все java-объекты, аннотированные @Configurable будут автоматически интегрированы в Spring. Более подробную информацию вы можете найти в документации Spring. Также обратите внимание, что для представления контекстной информации в графическом интерфейсе вам придется использовать сессионные компоненты. Это вызывает проблему с размером сеанса, а в более крупных приложениях невозможно кластеризовать.

0

Подобно тому, как вы подозревали, что ваши Spring-бобы созданы, но в контексте приложения и Spring сервлета. У вашего серваля Vaadin нет доступа ни к одному из них.

Для подробного (ручного) решения с кодом либо проверьте Vaadin wiki, либо найдите подходящие дополнения Vaadin, которые будут выполнять эту работу за вас (лично я рекомендую SpringVaadinIntegration).

Типичная идея - программно передать сервлет Vaadin в один из классов Spring Spring (например: WebApplicationContextUtils) и получить контекст приложения (загруженный ContextLoaderListener). Если вам нужен контекст сервлета, то из примеров кода, которые я видел, вы делаете то же самое, что и выше, но дополнительно вручную читаете контекст (например: using XmlWebApplicationContext) и устанавливаете контекст приложения как родительский.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню