Org.hibernate.SessionException: Session is closed!

Pessoal bom dia!

Estou começando a aprender sobre programação web e estou com um problema com abertura de session utilizando EntityManager, é o seguinte:

quando rodo a aplicação, o selectItens aparece normalmente com os itens recuperados do banco, mas quando dou um submit, aparece a seguinte mensagem:

exception

javax.servlet.ServletException: Session is closed!

root cause

org.hibernate.SessionException: Session is closed!

segue as classes:

ListenerFasesJSF

public class ListenerFasesJSF extends Aux  implements PhaseListener {

    @Override
    public void beforePhase(PhaseEvent fase) {    
       
        
        if (fase.getPhaseId().equals(PhaseId.RESTORE_VIEW)) {                     
            session.beginTransaction();            
            FacesContextUtil.setRequestSession(session);
        }
    }

    @Override
    public void afterPhase(PhaseEvent fase) {
                
        if (fase.getPhaseId().equals(PhaseId.RENDER_RESPONSE)) {
            session = FacesContextUtil.getRequestSession();
            try {
                session.getTransaction().commit();
            } catch (Exception e) {
                System.out.println("Erro no commit da transação");
                System.out.println(e.getMessage());
                e.printStackTrace();
                if (session.getTransaction().isActive()) {
                    session.getTransaction().rollback();
                }
            } finally {
                session.close();
            }
        }
    }

    @Override
    public PhaseId getPhaseId() {
        return PhaseId.ANY_PHASE;
    }
}

Auxiliar

public class Aux {

    public EntityManagerFactory emf;
    public EntityManager em ;
    public Session session;
    

    public Aux() {
        emf = Persistence.createEntityManagerFactory("rcrwebPU");
        em = emf.createEntityManager();
        
        if (em.getDelegate() instanceof EntityManagerImpl) {
            EntityManagerImpl entityManagerImpl = (EntityManagerImpl) em.getDelegate();
            session = entityManagerImpl.getSession();
        } else {
            session = (Session) em.getDelegate();
        }
    }
}

FacesContextUtil

public class FacesContextUtil {

    private static final String HIBERNATE_SESSION = "hibernate_session";

    public static void setRequestSession(Session session) {
        FacesContext.getCurrentInstance().getExternalContext().getRequestMap().put(HIBERNATE_SESSION, session);
    }

    public static Session getRequestSession() {
        return (Session) FacesContext.getCurrentInstance().getExternalContext().getRequestMap().get(HIBERNATE_SESSION);
    }
}

Já pesquisei pela WEB e não encontrei resposta de como abrir sessão utilizando EntityManager,

desde já agradeço a ajuda!!

muda a phase usada no phaseListener… de any_phase para restore_view… provavelmente seu phaselistener deve estar executando várias vezes…

se continuar dando erro posta o stack trace completo…

editando… dei uma olhada, confirma via debug ou poe um sysout para teste no if e no else do contrutor do Aux confirmando que a sessão está sendo aberta, depois da linha que você popula ela.

EDIMIL, você ja fez debug pra ver se o método beforePhase está caindo no IF e criando a session com sucesso?

[quote=maior_abandonado]muda a phase usada no phaseListener… de any_phase para restore_view… provavelmente seu phaselistener deve estar executando várias vezes…

se continuar dando erro posta o stack trace completo…

editando… dei uma olhada, confirma via debug ou poe um sysout para teste no if e no else do contrutor do Aux confirmando que a sessão está sendo aberta, depois da linha que você popula ela.[/quote]

maior_abandonado, alterei de any_phase para restore_view e não apresentou mais o erro de sessão fechada, mas neste caso será minha sessão vai ficar aberta sempre?

o erro que dava era este:


org.hibernate.SessionException: Session is closed!
	at org.hibernate.impl.AbstractSessionImpl.errorIfClosed(AbstractSessionImpl.java:72)
	at org.hibernate.impl.SessionImpl.beginTransaction(SessionImpl.java:1466)
	at br.com.rcrweb.controller.ListenerFasesJSF.beforePhase(ListenerFasesJSF.java:29)
	at com.sun.faces.lifecycle.Phase.handleBeforePhase(Phase.java:228)
	at com.sun.faces.lifecycle.Phase.doPhase(Phase.java:99)
	at com.sun.faces.lifecycle.RestoreViewPhase.doPhase(RestoreViewPhase.java:113)
	at com.sun.faces.lifecycle.LifecycleImpl.execute(LifecycleImpl.java:118)
	at javax.faces.webapp.FacesServlet.service(FacesServlet.java:409)
	at org.apache.catalina.core.StandardWrapper.service(StandardWrapper.java:1534)
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:343)
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:215)
	at org.apache.myfaces.trinidadinternal.webapp.TrinidadFilterImpl._doFilterImpl(TrinidadFilterImpl.java:293)
	at org.apache.myfaces.trinidadinternal.webapp.TrinidadFilterImpl.doFilter(TrinidadFilterImpl.java:199)
	at org.apache.myfaces.trinidad.webapp.TrinidadFilter.doFilter(TrinidadFilter.java:92)
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256)
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:215)
	at org.apache.myfaces.webapp.filter.ExtensionsFilter.doFilter(ExtensionsFilter.java:147)
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256)
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:215)
	at org.netbeans.modules.web.monitor.server.MonitorFilter.doFilter(MonitorFilter.java:393)
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256)
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:215)
	at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:279)
	at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:175)
	at org.apache.catalina.core.StandardPipeline.doInvoke(StandardPipeline.java:655)
	at org.apache.catalina.core.StandardPipeline.invoke(StandardPipeline.java:595)
	at com.sun.enterprise.web.WebPipeline.invoke(WebPipeline.java:98)
	at com.sun.enterprise.web.PESessionLockingStandardPipeline.invoke(PESessionLockingStandardPipeline.java:91)
	at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:162)
	at org.apache.catalina.connector.CoyoteAdapter.doService(CoyoteAdapter.java:326)
	at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:227)
	at com.sun.enterprise.v3.services.impl.ContainerMapper.service(ContainerMapper.java:170)
	at com.sun.grizzly.http.ProcessorTask.invokeAdapter(ProcessorTask.java:822)
	at com.sun.grizzly.http.ProcessorTask.doProcess(ProcessorTask.java:719)
	at com.sun.grizzly.http.ProcessorTask.process(ProcessorTask.java:1013)
	at com.sun.grizzly.http.DefaultProtocolFilter.execute(DefaultProtocolFilter.java:225)
	at com.sun.grizzly.DefaultProtocolChain.executeProtocolFilter(DefaultProtocolChain.java:137)
	at com.sun.grizzly.DefaultProtocolChain.execute(DefaultProtocolChain.java:104)
	at com.sun.grizzly.DefaultProtocolChain.execute(DefaultProtocolChain.java:90)
	at com.sun.grizzly.http.HttpProtocolChain.execute(HttpProtocolChain.java:79)
	at com.sun.grizzly.ProtocolChainContextTask.doCall(ProtocolChainContextTask.java:54)
	at com.sun.grizzly.SelectionKeyContextTask.call(SelectionKeyContextTask.java:59)
	at com.sun.grizzly.ContextTask.run(ContextTask.java:71)
	at com.sun.grizzly.util.AbstractThreadPool$Worker.doWork(AbstractThreadPool.java:532)
	at com.sun.grizzly.util.AbstractThreadPool$Worker.run(AbstractThreadPool.java:513)
	at java.lang.Thread.run(Thread.java:619)

leonhard32, o erro cai no IF em cima do session.beginTransation:

public void beforePhase(PhaseEvent fase) {
        System.out.println("Antes Fase: " + fase.getPhaseId());  
        
        if (fase.getPhaseId().equals(PhaseId.RESTORE_VIEW)) {
            System.out.println("chegou aqui" );            
    ==> session.beginTransaction();
            System.out.println("não chegou aqui " );
            FacesContextUtil.setRequestSession(session);
        }
    }

leonhard32, o erro cai no IF em cima do session.beginTransation:

[code]
public void beforePhase(PhaseEvent fase) {
System.out.println("Antes Fase: " + fase.getPhaseId());

    if (fase.getPhaseId().equals(PhaseId.RESTORE_VIEW)) {
        System.out.println("chegou aqui" );            
==> session.beginTransaction();
        System.out.println("não chegou aqui " );
        FacesContextUtil.setRequestSession(session);
    }
}

[/code][/quote]

EDIMIL, pode postar como você esta obtendo a session?

EDIMIL, o que me parece estar acontecendo é você não estar abrindo a sessão, antes de iniciar…

Tipo:

   [code]@Override
public void beforePhase(PhaseEvent fase) {
	if(fase.getPhaseId().equals(PhaseId.RESTORE_VIEW)){
		Session session = HibernateUtil.getSessionfactory().openSession();
		session.beginTransaction();
		FacesContextUtil.setRequestSession(session);
	}
}[/code]

[quote=leonhard32]EDIMIL, o que me parece estar acontecendo é você não estar abrindo a sessão, antes de iniciar…

Tipo:

   [code]@Override
public void beforePhase(PhaseEvent fase) {
	if(fase.getPhaseId().equals(PhaseId.RESTORE_VIEW)){
		Session session = HibernateUtil.getSessionfactory().openSession();
		session.beginTransaction();
		FacesContextUtil.setRequestSession(session);
	}
}[/code][/quote]

leonhard32 eu não utilizo esta linha [Session session = HibernateUtil.getSessionfactory().openSession();] porque minha aplicação não utiliza o arquivo cfg, eu utilizo anotações, tem outra forma de iniciar a sessão utilizando entitymanager?

Entendi…
Você esta trabalhando com o JSF + Hibernate limpo ou está usando o Seam?

Eu ja tive problemas do tipo, quando trabalhei com o Seam…
Mas, se puder postar o trecho do código onde você trabalha a obtenção da Session, e a gerencia, da pra ver o que está acontecendo

Abs

[quote=leonhard32]Entendi…
Você esta trabalhando com o JSF + Hibernate limpo ou está usando o Seam?

Eu ja tive problemas do tipo, quando trabalhei com o Seam…
Mas, se puder postar o trecho do código onde você trabalha a obtenção da Session, e a gerencia, da pra ver o que está acontecendo

Abs[/quote]

Não uso o Seam, quando eu carrego a página cargo.jsp, entra na classe ListenerFasesJSF que extende a classe Aux para obtenção da sessão, quando vem a fase RENDER_RESPONSE a sessão é fechada e mesmo entrando na RESTORE_VIEW novamente a sessão não abre.

Eu não sei como a sessão é aberta pelo EntityManager, porque tudo inicia por lá, se eu estivesse usando o arquivo cfg poderia abrir a sessão utilizando session.getSessionFactory.openSession(), mas pelo EM não achei ainda a opção…

    public class Aux {  
      
        public EntityManagerFactory emf;  
        public EntityManager em ;  
        public Session session;  
          
      
        public Aux() {  
            emf = Persistence.createEntityManagerFactory("rcrwebPU");  
            em = emf.createEntityManager();  
              
            if (em.getDelegate() instanceof EntityManagerImpl) {  
                EntityManagerImpl entityManagerImpl = (EntityManagerImpl) em.getDelegate();  
                session = entityManagerImpl.getSession();  
            } else {  
                session = (Session) em.getDelegate();  
            }  
        }  
    }  

Ainda tenho a clase BackingBenDepartamento que recupera os departamentos cadastrados:

public class DepartamentoBackingBean {

    private List<SelectItem> selectDepartamento;

    public DepartamentoBackingBean() {
    }

    public List<SelectItem> getDepartamentos() {

        Session session = FacesContextUtil.getRequestSession();
        InterfaceDAO<DepartamentoBean> departamentoDAO = new DAOGeral<DepartamentoBean>(DepartamentoBean.class, session);
        List<DepartamentoBean> departamentos = departamentoDAO.getBeans();        
        selectDepartamento = new ArrayList<SelectItem>();
        selectDepartamento.add(new SelectItem(null, "Selecione o Cargo ..."));
        for (DepartamentoBean departamento : departamentos) {
            selectDepartamento.add(new SelectItem(departamento.getDepCod().toString(), departamento.getDepNome()));
        }

        return selectDepartamento;

    }
}