I am working on Spring MVC project. I am using Hibernate directly. The error No Session found for current thread
has occurred in my project. I know that on stackoverflow and on other places on the web this error is very common. But I haven't found working solution to my issue.
I have had unresolved problem with proper configuration of Hibernate Session Factory. I have described it here: Can I use Hibernate Session Factory declared in DispatcherServlet Context instead of hibernate.cfg.xml? But finally I am using hibernate.cfg.xml
file and also Hibernate Session Factory defined in DispatcherServlet Context file. I have done some simple operations like persisting classes in database. And everything was OK.
At this moment my project has a lot of in common with Spring MVC Hibernate. I have added some annotations to my classes and I have created some new services classes.
My ServletDispatcher Context file contains this definitions:
<annotation-driven />
<context:annotation-config />
<context:component-scan base-package="finances.webapp" />
<!-- Data source -->
<beans:bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<beans:property name="driverClassName" value="com.mysql.jdbc.Driver" />
<beans:property name="url" value="jdbc:mysql://localhost/finances" />
<beans:property name="username" value="root" />
<beans:property name="password" value="root" />
</beans:bean>
<!-- Data source end -->
<!-- Hibernate session factory -->
<beans:bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<beans:property name="dataSource">
<beans:ref bean="dataSource" />
</beans:property>
<beans:property name="configLocation" value="classpath:hibernate.cfg.xml" />
</beans:bean>
<!-- Hibernate session factory end -->
<beans:bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<beans:property name="sessionFactory" ref="sessionFactory" />
</beans:bean>
<beans:bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor">
<beans:property name="transactionManager" ref="transactionManager" />
<beans:property name="transactionAttributes">
<beans:props>
<beans:prop key="save">PROPAGATION_REQUIRED</beans:prop>
</beans:props>
</beans:property>
</beans:bean>
In my pom.xml file I have only these two dependencies according to ORM:
<!-- Hibernate -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-entitymanager</artifactId>
<version>4.1.4.Final</version>
</dependency>
<!-- EJB -->
<dependency>
<groupId>javax.ejb</groupId>
<artifactId>ejb-api</artifactId>
<version>3.0</version>
<scope>provided</scope>
</dependency>
Maybe problem is here? But I haven't any problem with missing packages earlier.
This is code from my controller:
UsersEntity user2 = new UsersEntity("qwerty", "qwerty", true);
usersService.addUser(user2);
The problem No Session found for current thread
has occured during invoking addUser()
method.
My DAO class:
@Repository
public class UsersHome {
@Autowired
private SessionFactory sessionFactory;
public void persist(UsersEntity transientInstance) {
sessionFactory.getCurrentSession().persist(transientInstance);
}
}
My service:
@Service
public class UsersService {
@Autowired
private UsersHome usersHome;
@Transactional(propagation = Propagation.REQUIRED)
public void addUser(UsersEntity user) {
usersHome.persist(user);
}
}
This code has stopped working when I have change method body in UsersHome
class. Previously I have been starting session before my action and ending session after my action. But using annotations and transactions it should work? Please give me some suggestions.
EDIT
<hibernate-configuration>
<session-factory>
<property name="dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="show_sql">true</property>
<property name="hbm2ddl.auto">update</property>
<property name="connection.url">jdbc:mysql://localhost/finances</property>
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="connection.username">root</property>
<property name="connection.password">root</property>
<mapping class="finances.webapp.entities.AuthoritiesEntity"/>
<mapping class="finances.webapp.entities.ExpensesEntity"/>
<mapping class="finances.webapp.entities.ExpensesCategoriesEntity"/>
<mapping class="finances.webapp.entities.ExpensesObjectsEntity"/>
<mapping class="finances.webapp.entities.IncomesEntity"/>
<mapping class="finances.webapp.entities.UsersEntity"/>
</session-factory>
</hibernate-configuration>
EDIT #2
org.springframework.beans.factory.BeanCreationException: Error creating bean with
name 'usersService': Injection of autowired dependencies failed; nested exception
is org.springframework.beans.factory.BeanCreationException: Could not autowire
field: private finances.webapp.dao.UsersHome
finances.webapp.services.UsersService.usersHome;
nested exception is org.springframework.beans.factory.BeanCreationException:
Error creating bean with name 'usersHome': Injection of autowired dependencies
failed; nested exception is org.springframework.beans.factory.BeanCreationException:
Could not autowire field: private org.hibernate.SessionFactory
finances.webapp.dao.UsersHome.sessionFactory; nested exception is
org.springframework.beans.factory.BeanCreationException: Error creating bean with
name 'sessionFactory' defined in ServletContext resource
[/WEB-INF/spring/appServlet/servlet-context.xml]: Invocation of init method
failed; nested exception is java.lang.NoClassDefFoundError:
org/hibernate/cfg/EJB3DTDEntityResolver
EDIT #3
My root-context.xml
file:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<!-- Root Context: defines shared resources visible to all other web components -->
</beans>
And my whole spring-config.xml
file:
<?xml version="1.0" encoding="UTF-8"?>
<beans:beans xmlns="http://www.springframework.org/schema/mvc"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:beans="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
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">
<!-- DispatcherServlet Context: defines this servlet's request-processing infrastructure -->
<!-- Enables the Spring MVC @Controller programming model -->
<annotation-driven />
<context:annotation-config />
<context:component-scan base-package="finances.webapp" />
<!-- Handles HTTP GET requests for resources by efficiently serving up static resources in the ${webappRoot}/resources directory -->
<resources mapping="/resources/**" location="/resources/" />
<!--<beans:bean id="tilesConfigurer" class="org.springframework.web.servlet.view.tiles2.TilesConfigurer">-->
<!--<beans:property name="definitions">-->
<!--<beans:list>-->
<!--<beans:value>/WEB-INF/tiles-definitions.xml</beans:value>-->
<!--</beans:list>-->
<!--</beans:property>-->
<!--</beans:bean>-->
<!--<beans:bean id="viewResolver" class="org.springframework.web.servlet.view.UrlBasedViewResolver">-->
<!--<beans:property name="viewClass" value="org.springframework.web.servlet.view.tiles2.TilesView" />-->
<!--</beans:bean>-->
<!-- Data source -->
<beans:bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<beans:property name="driverClassName" value="com.mysql.jdbc.Driver" />
<beans:property name="url" value="jdbc:mysql://localhost/finances" />
<beans:property name="username" value="root" />
<beans:property name="password" value="root" />
</beans:bean>
<!-- Data source end -->
<!-- Hibernate session factory -->
<beans:bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<beans:property name="dataSource" ref="dataSource" />
<beans:property name="configLocation" value="classpath:hibernate.cfg.xml" />
</beans:bean>
<!-- Hibernate session factory end -->
<beans:bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<beans:property name="sessionFactory" ref="sessionFactory" />
</beans:bean>
<beans:bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor">
<beans:property name="transactionManager" ref="transactionManager" />
<beans:property name="transactionAttributes">
<beans:props>
<beans:prop key="save">PROPAGATION_REQUIRED</beans:prop>
<beans:prop key="persist">PROPAGATION_REQUIRED</beans:prop>
</beans:props>
</beans:property>
</beans:bean>
</beans:beans>
Try to remove "transactionInterceptor", you already have
@Transaction
attribute set and also have tag<annotation-driven/>
. Or add to transactionInterceptor key forpersist()
methodMaybe it help.
EDIT: I recreated your configuration and it works.
this it spring config:
this is hibernate config:
dependency in pom.xml
unit-test:
and unit-test output: