由于空闲事务,Spring Boot 事务回滚不会在 PostgreSQL 数据库上触发

MHo*_*gge 1 java postgresql transactions spring-boot

我尝试执行事务操作并故意抛出异常,以验证回滚是否已完成,但回滚并未执行。

PostgreSQL 数据库版本为 12.1-1,基于 Docker。

这是包含注释的服务@Transactional

@Service
public class MyTestService {
    @Autowired
    private DocumentDataDao documentDataDao;

    @Transactional
    public void test() {
        DocumentData data = new DocumentData();
        data.setData(UUID.randomUUID().toString());
        documentDataDao.create(data);
        throw new IllegalArgumentException("Test rollback");
    }
}
Run Code Online (Sandbox Code Playgroud)

create函数使用 aNamedParameterJdbcTemplate插入数据:

String statement = String.format("INSERT INTO %s (%s) VALUES (%s) RETURNING %s", tableName,
                String.join(",", insertingColumns), String.join(",", values),
                String.join(",", returningColumns));
return getNamedJdbcTemplate().queryForObject(statement, parameters, getRowMapper());
Run Code Online (Sandbox Code Playgroud)

test函数是从另一个服务调用的:

@Service
public class ApplicationStartupListener {
    private Logger log = LoggerFactory.getLogger(ApplicationStartupListener.class);

    @Autowired
    private MyTestService testService;

    @PostConstruct
    public void init() {
        try {
            testService.test();
        } catch (Exception e) {
            log.error("fail to start", e);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

调试的时候发现如果没有执行回滚是因为事务正在执行IDLE

以下是未执行的rollback函数:PgConnectionexecuteTransactionCommand

public void rollback() throws SQLException {
    checkClosed();

    if (autoCommit) {
      throw new PSQLException(GT.tr("Cannot rollback when autoCommit is enabled."),
          PSQLState.NO_ACTIVE_SQL_TRANSACTION);
    }

    if (queryExecutor.getTransactionState() != TransactionState.IDLE) {
      executeTransactionCommand(rollbackQuery);
    }
  }
Run Code Online (Sandbox Code Playgroud)

任何关于为什么事务被标记为空闲并停止执行回滚方法的提示将不胜感激。

编辑(1)

正如@M。Deinum 提到,不能保证使用时已创建事务代理@PostConstruct。这就是为什么我用以下方法进行测试ApplicationRunner

@Component
public class AppStartupRunner implements ApplicationRunner {
    @Autowired
    private MyTestService testService;

    @Override
    public void run(ApplicationArguments args) throws Exception {
           testService.test();
    }
}
Run Code Online (Sandbox Code Playgroud)

但这也不起作用。

我还尝试test在应用程序启动后不久通过使用 aRestController并向其发送 HTTP 请求来运行该方法,但仍然存在同样的问题。

@RestController
public class AppController {

    @Autowired
    private MyTestService testService;

    @GetMapping("/test")
    public ResponseEntity<Object> test() {
        testService.test();
        return ResponseEntity.ok().build();
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑(2)

我将 PostgreSQL JDBC 版本从 42.2.2 升级到 42.2.18(目前最新),但在尝试回滚时连接仍然处于 IDLE 状态。

编辑(3)

我在 git 存储库中重现了该问题:https://github.com/Martin-Hogge/spring-boot-postgresql-transactional-example/tree/master

Ism*_*maz 6

我检查了您想要在单个应用程序中使用多个架构(数据源、jdbc 模板)的架构。@Transactional仅管理名为 的应用程序默认数据源HikariPool-1。当您调用其余方法时,将创建名为 的新 hikari 池HikariPool-2。您的操作正在进行HikariPool-2,但@Transactional仅进行管理HikariPool-1

@Transactional的事务管理器参数不能动态更改。因此,您可以定义一个新的自定义注释来管理您的事务。或者您可以使用TransactionTemplate注释来代替。

我创建了一个简单的自定义事务管理方面。它与定义的 dao 的数据源和事务生命周期一起工作。

测试服务

@Service
public class MyTestService {
    @Autowired
    private DocumentDataDao documentDataDao;

    @CustomTransactional(DocumentDataDao.class)
    public void test() {
        DocumentData data = new DocumentData();
        data.setData(UUID.randomUUID().toString());
        documentDataDao.create(data);
        throw new IllegalArgumentException("Test rollback");
    }
}
Run Code Online (Sandbox Code Playgroud)

定制交易

package com.example.transactional;

import java.lang.annotation.*;

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface CustomTransactional {
    Class<? extends BaseDao<?>> value();
}
Run Code Online (Sandbox Code Playgroud)

自定义事务方面

package com.example.transactional;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

import java.util.HashMap;
import java.util.Map;

@Aspect
@Component
public class CustomTransactionalAspect implements ApplicationContextAware {
    private ApplicationContext applicationContext;
    private Map<Class<? extends BaseDao<?>>, BaseDao<?>> classMap = new HashMap<>();

    @Around("@annotation(customTransactional)")
    public Object customTransaction(ProceedingJoinPoint joinPoint, CustomTransactional customTransactional) throws Throwable {
        BaseDao<?> baseDao = getBaseDao(customTransactional.value());

        // custom transaction management
        return baseDao.getConnectionHandler().getTransactionTemplate().execute(status -> {
            try {
                return joinPoint.proceed();
            } catch (Throwable throwable) {
                throw new RuntimeException(throwable);
            }
        });
    }

    /**
     * Search {@link BaseDao} class on spring beans
     *
     * @param clazz Target dao class type
     * @return Spring bean object
     */
    private BaseDao<?> getBaseDao(Class<? extends BaseDao<?>> clazz) {
        return classMap.computeIfAbsent(clazz, c -> applicationContext.getBean(c));
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
}
Run Code Online (Sandbox Code Playgroud)

连接处理程序

我添加的transactionTemplate是交易操作

public class ConnectionHandler {
    private NamedParameterJdbcTemplate namedJdbcTemplate;
    private JdbcTemplate jdbcTemplate;
    private TransactionTemplate transactionTemplate;
    private String schema;

    public ConnectionHandler(DataSource dataSource, String schema) {
        this.namedJdbcTemplate = new NamedParameterJdbcTemplate(dataSource);
        this.jdbcTemplate = new JdbcTemplate(dataSource);
        this.schema = schema;

        this.transactionTemplate = new TransactionTemplate(new DataSourceTransactionManager(dataSource));
    }

    public NamedParameterJdbcTemplate getNamedJdbcTemplate() {
        return namedJdbcTemplate;
    }

    public JdbcTemplate getJdbcTemplate() {
        return jdbcTemplate;
    }

    public String getSchema() {
        return schema;
    }

    public TransactionTemplate getTransactionTemplate() {
        return transactionTemplate;
    }
}
Run Code Online (Sandbox Code Playgroud)

基础道

将修饰符 更改getConnectionHandlerpublic

    public ConnectionHandler getConnectionHandler() {
        return getDataSource().getConnection(getSchemaName());
    }
Run Code Online (Sandbox Code Playgroud)

pom.xml

您可以删除postgresql.version,spring-jdbc.versionHikariCP.version。问题与版本无关。添加spring-boot-starter-aop方面操作的依赖项。

<dependencies>
...
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
</dependencies>
Run Code Online (Sandbox Code Playgroud)