
cn.org.atool.fluentmachine.persistence.ContextRepositoryImpl Maven / Gradle / Ivy
package cn.org.atool.fluentmachine.persistence;
import cn.org.atool.fluentmachine.context.Context;
import cn.org.atool.fluentmachine.context.FireContext;
import cn.org.atool.fluentmachine.exception.LockException;
import lombok.Getter;
import lombok.Setter;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
import javax.sql.DataSource;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import static cn.org.atool.fluentmachine.persistence.ContextRepositorySql.*;
import static java.util.stream.Collectors.toMap;
/**
* 保存状态机上下文数据库实现
*
* @author darui.wu
*/
public class ContextRepositoryImpl extends JdbcDaoSupport implements ContextRepository {
/**
* 默认超时时间10s
*/
@Getter
@Setter
private long lockDuration = 1000L * 10;
private final String env;
private final String ctxTable;
private final String logTable;
public ContextRepositoryImpl(String env, DataSource dataSource, String ctxTable, String logTable) {
super.setDataSource(dataSource);
this.env = env;
this.ctxTable = ctxTable;
this.logTable = logTable;
}
@Override
public void saveContext(Context ctx, FireContext fire) {
ContextEntity bean = ContextHelper.toSaveContext(ctx);
if (this.isExistContext(ctx.getMachineId(), ctx.getTradeNo())) {
super.getJdbcTemplate().update(
SQL_UPDATE_CONTEXT_STATUS(this.ctxTable),
bean.getCtxState(), bean.getRegionStates(), bean.getContext(), bean.getErrors(), bean.getSwitcher(),
this.lockDuration,
ctx.getTradeNo(), ctx.getMachineId(), this.env);
} else {
super.getJdbcTemplate().update(
SQL_INSERT_CONTEXT_STATUS(this.ctxTable),
ctx.getMachineId(), ctx.getTradeNo(),
bean.getCtxState(), bean.getRegionStates(), bean.getContext(), bean.getErrors(), bean.getSwitcher(),
ctx.getLockVersion(), this.env, ctx.getLockExpireSecond());
}
super.getJdbcTemplate().update(SQL_INSERT_CONTEXT_LOG(this.logTable),
ctx.getMachineId(), ctx.getTradeNo(),
bean.getCtxState(), bean.getRegionStates(), bean.getContext(), bean.getErrors(), bean.getSwitcher(),
fire.getFireEvent(), fire.getSourceState(), fire.getTargetState(), this.env);
}
@Override
public boolean lock(Context ctx, String lockVersion, Object event) {
boolean exists = this.isExistContext(ctx.getMachineId(), ctx.getTradeNo());
if (exists) {
return updateLock(ctx, lockVersion, this.lockDuration);
} else {
return true;
}
}
private static long aYear = 1000L * 60 * 60 * 24 * 365;
@Override
public boolean unlock(Context ctx) {
return updateLock(ctx, "0", -aYear);
}
/**
* 更新全局锁
*
* @param ctx 上下文
* @param newLock 新的版本锁
* @param expireSeconds 版本锁过期时间
* @return
*/
private boolean updateLock(Context ctx, String newLock, long expireSeconds) {
int result = super.getJdbcTemplate().update(
SQL_UPDATE_CONTEXT_LOCK(this.ctxTable),
newLock, expireSeconds,
ctx.getTradeNo(), ctx.getMachineId(), this.env, ctx.getLockVersion());
if (result > 0) {
ctx.setLockVersion(newLock);
return true;
} else {
return false;
}
}
@Override
public boolean isExistContext(String machineId, String tradeNo) {
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy