Question

J'ai une pile Spring / JPA / Groovy / Hibernate, (Remarque: Pas Graals), et je trouve que je reçois un No session found for current thread Lorsque vous essayez d'en effectuer @Transactional Travaillez sur le démarrage.

J'ai une classe de contrôleur qui, sur @PostConstruct, invoque un @Transactional Méthode sur une autre classe qui essaie de remplir la base de données avec quelques données d'échantillonnage à utiliser dans une simulation.

Voici le controller classer:

@Component
public class SimulationController {

    private TransactionTemplate transactionTemplate;
    @Autowired
    private PlatformTransactionManager transactionManager;
    @Autowired
    private IPublisher publisher;

    @PostConstruct
    public void intialize()
    {
        this.transactionTemplate = new TransactionTemplate(transactionManager);
        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus arg0) {
                racePublisher.populateData();
            }
        });
    }
    //  Also tried, with no success:    
    //  @PostConstruct
    //  public void initialize()
    //  {
    //      publisher.populateData();
    //  }
}

Comme vous pouvez le voir, je me suis éloigné d'un pur @Transactional approche dans le @PostConstruct Pour les raisons discutées ici.

Mon IPublisher est une classe groovy, comme suit:

@Component
class Publisher implements IPublisher {

@Autowired
IStockDAO stockDAO

void populateData()
{
    createStock()
}
@Transactional
void createStock()
{
    def list = [new Stock(ticker: "ADBE", name: "Adobe"),
                new Stock(ticker: "MSFT", venueCode: "Microsoft")]
    list.each { stockDAO.create it }
}

Qui est une implémentation de:

public interface IPublisher {

    public void populateData();
    public void createStock();
}

Remarque, j'ai aussi essayé de marquer populateData() comme @Transactional, sans effet.

Dans ma classe de contexte de printemps, je définis <tx:annotation-driven/>.

Pour autant que je sache, j'ai tout fait correctement. Cependant, je n'arrive pas à faire fonctionner cela.

Qu'est-ce qui est requis?

Actualisé:Voici les haricots configurant mes haricots liés à l'accès aux données:

<beans>
<bean id="transactionManager"
        class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource" />
    </bean>

    <tx:annotation-driven/>

    <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver" />
        <property name="url" value="jdbc:mysql://${database.host}:${database.port}/${database.name}" />
        <property name="username" value="${database.username}" />
        <property name="password" value="${database.password}" />
        <property name="initialSize" value="5" />
        <property name="maxActive" value="50" />
    </bean>
    <bean id="sessionFactory"
        class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="packagesToScan">
            <list>
                <value>com.mangofactory.concorde</value>
            </list>
        </property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">${hibernate.dialect}</prop>
                <prop key="hibernate.cache.provider_class">org.hibernate.cache.HashtableCacheProvider
                </prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.format_sql">true</prop>
                <prop key="hibernate.hbm2ddl.auto">update</prop><!--  use create for full drop/create -->
                <prop key="hibernate.connection.autocommit">true</prop>
                <prop key="hibernate.statement_cache.size">0</prop>
                <prop key="hibernate.jdbc.batch_size">20</prop>
            </props>
        </property>
    </bean>
    <beans>
Était-ce utile?

La solution

Tu devrais utiliser HibernateTransactionManager à la place de DataSourceTransactionManager.

Autres conseils

Il y a des problèmes de base avec le code. La frontière transactionnelle commence dans la méthode Createstock de la classe d'éditeurs et s'y termine. Il ne se propage pas à votre contrôleur. Utilisez les appels de requête DB dans une transaction. c'est-à-dire à l'intérieur d'une méthode qui a l'annotation transactionnelle. Sinon cela ne fonctionnerait jamais.

Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top