Reputation: 3841
I'm trying to run entityManager.merge(myEntity) within the following method but it seems that the @Transactional annotation is ignored. The Hibernate configuration seems to be fine because I can successfully fetch data from the db but it's not possible to write to the db. I'm using Spring version 3.2.3. Why are the writing db operations not working?
my method that does not work
package com.reflections.importer.bls;
...
@Service
class BlsGovImporter {
...
@Transactional
private void importSeries(String externalId) {
// This works. The dao is using EntityManager too
Series series = seriesDao.findByExternalId(externalId);
series.getValues().addAll(fetchNewValues());
// This does not work and no exception is thrown
entityManager.merge(series);
}
Upvotes: 7
Views: 24577
Reputation: 59
@Matheus Santz, you are correct!! When using multiple datasources, it is always a best practice to annotate the @Transactional with the TransactionManager reference.
TransactDataSourceConfig.java
package com.spsllc.config;
import javax.sql.DataSource;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.TransactionManager;
import org.springframework.transaction.annotation.EnableTransactionManagement;
import com.zaxxer.hikari.HikariDataSource;
@Configuration
@EnableTransactionManagement
public class TransactDataSourceConfig {
@Bean
@ConfigurationProperties("spsllc.datasource.transactdb")
public DataSourceProperties transactDataSourceProperties() {
return new DataSourceProperties();
}
@Bean(name = "transactdbDS")
@ConfigurationProperties("spsllc.datasource.transactdb.configuration")
public DataSource transactDataSource() {
return transactDataSourceProperties().initializeDataSourceBuilder().type(HikariDataSource.class).build();
}
@Bean
public JdbcTemplate transactdbJdbcTemplate(@Qualifier("transactdbDS") DataSource dataSource) {
return new JdbcTemplate(dataSource);
}
@Bean
public NamedParameterJdbcTemplate transactdbNamedParameterJdbcTemplate(@Qualifier("transactdbDS") DataSource dataSource) {
return new NamedParameterJdbcTemplate(dataSource);
}
@Bean(name = "transactTransactionManager")
public TransactionManager transactTransactionManager() {
return new DataSourceTransactionManager(transactDataSource());
}
}
SearchLogRepository.java
package com.spsllc.repository.transactdb;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.namedparam.BeanPropertySqlParameterSource;
import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate;
import org.springframework.jdbc.support.GeneratedKeyHolder;
import org.springframework.stereotype.Repository;
import org.springframework.transaction.annotation.Transactional;
import com.spsllc.domain.SearchLogEntry;
import lombok.extern.slf4j.Slf4j;
@Repository
@Slf4j
public class SearchLogRepository {
@Autowired
@Qualifier("transactdbJdbcTemplate")
JdbcTemplate jdbcTemplate;
@Autowired
@Qualifier("transactdbNamedParameterJdbcTemplate")
NamedParameterJdbcTemplate nmParamjdbcTemplate;
@Transactional(transactionManager = "transactTransactionManager")
public void insert(List<SearchLogEntry> entries) {
String sql = "INSERT INTO search_log(member_id, weekof_metadata, week_begin_date, " +
"week_end_date, activity_date,type, name, contact_person, contact_type) " +
"VALUES(:memberId, :weekofMetadata, :weekBeginDate, " +
":weekEndDate, :activityDate, :type, :name, :contactPerson, :contactType);";
GeneratedKeyHolder generatedKeyHolder = new GeneratedKeyHolder();
int index = 0;
for (SearchLogEntry e : entries) {
nmParamjdbcTemplate.update(sql, new BeanPropertySqlParameterSource(e), generatedKeyHolder,
new String[] { "search_log_id" });
Integer id = generatedKeyHolder.getKey().intValue();
e.setSearchLogId(id);
log.info(e.toString());
index++;
/** uncomment to test the rollback
if(index > 5)
throw new RuntimeException("Testing @Transactional commit/rollback behavior");
*/
}
}
}
Upvotes: 0
Reputation: 31
In my case, I had to add @EnableTransactionManagement
(you can add it in any @Configuration
class or on your @SpringBootApplication
class).
Upvotes: 2
Reputation: 690
In my case, I had to manually set the transactionManager bean name to make it work:
@Transactional(transactionManager = "myTransactionManager")
Upvotes: 1
Reputation: 8286
luboskrnac actualy answered it exactly for me, but just adding this to Spring newbie who might be confused on using proxy or not.
Please refer to this page explaining the case where even if you call an @Transactional method within the same class, since you call it without the same class, it will not be called via proxy.
Upvotes: 3
Reputation: 24591
Because it is used on private method. Spring Docs:
Method visibility and @Transactional
When using proxies, you should apply the @Transactional annotation only to methods with public visibility. If you do annotate protected, private or package-visible methods with the @Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. Consider the use of AspectJ (see below) if you need to annotate non-public methods.
When its private, it is obviously called from within the same class. But Spring call needs to go through proxy in order to make it working. So the method will need to be called from another bean.
Other option is to annotate class with @Transactional.
Upvotes: 21