1 Star 0 Fork 0

go-better/go

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
tcc_mq.go 2.93 KB
一键复制 编辑 原始数据 按行查看 历史
bughou 提交于 2022-03-21 22:24 +08:00 . save
package tcc
import (
"database/sql"
"errors"
"strings"
"sync"
"time"
)
// func for mq handling.
func (tcc *TCC) confirmOrCancel(tx *sql.Tx) (time.Duration, bool, error) {
data := tcc.msg.Data.(*tccData)
if data.Status != statusConfirmed && data.Status != statusCanceled {
// cancel tcc if trying timeout
data.Status = statusCanceled
if canCommit, err := tcc.update(setCanceled, statusTrying, "Cancel", tx); err != nil {
return 0, canCommit, err
}
}
confirm := data.Status == statusConfirmed
if data.Concurrent {
if confirm {
return tcc.confirmConcurrently(data, tx)
} else {
return tcc.cancelConcurrently(data, tx)
}
} else {
if confirm {
return tcc.confirmSerially(data, tx)
} else {
return tcc.cancelSerially(data, tx)
}
}
}
func (tcc *TCC) confirmConcurrently(data *tccData, tx *sql.Tx) (time.Duration, bool, error) {
var retryAfter time.Duration
var canCommit = true
var errs []string
var wg sync.WaitGroup
for i, action := range data.Actions {
if action.Status != statusConfirmed {
wg.Add(1)
go func(action tccAction, i int) {
if _retryAfter, _canCommit, err := action.confirm(tcc, tx, i); err != nil {
if _retryAfter > retryAfter {
retryAfter = _retryAfter
}
canCommit = canCommit && _canCommit
errs = append(errs, action.Name+": "+err.Error())
}
wg.Done()
}(action, i)
}
}
wg.Wait()
if len(errs) == 0 {
return 0, true, nil
}
return retryAfter, canCommit, errors.New(strings.Join(errs, "; "))
}
func (tcc *TCC) cancelConcurrently(data *tccData, tx *sql.Tx) (time.Duration, bool, error) {
var retryAfter time.Duration
var canCommit = true
var errs []string
var wg sync.WaitGroup
for i, action := range data.Actions {
if action.Status != statusCanceled {
wg.Add(1)
go func(action tccAction, i int) {
if _retryAfter, _canCommit, err := action.cancel(tcc, tx, i); err != nil {
if _retryAfter > retryAfter {
retryAfter = _retryAfter
}
canCommit = canCommit && _canCommit
errs = append(errs, action.Name+": "+err.Error())
}
wg.Done()
}(action, i)
}
}
wg.Wait()
if len(errs) == 0 {
return 0, true, nil
}
return retryAfter, canCommit, errors.New(strings.Join(errs, "; "))
}
func (tcc *TCC) confirmSerially(data *tccData, tx *sql.Tx) (time.Duration, bool, error) {
for i, action := range data.Actions {
if action.Status != statusConfirmed {
if retryAfter, canCommit, err := action.confirm(tcc, tx, i); err != nil {
return retryAfter, canCommit, errors.New(action.Name + ": " + err.Error())
}
}
}
return 0, true, nil
}
func (tcc *TCC) cancelSerially(data *tccData, tx *sql.Tx) (time.Duration, bool, error) {
for i := len(data.Actions) - 1; i >= 0; i-- {
action := data.Actions[i]
if action.Status != statusCanceled {
if retryAfter, canCommit, err := action.cancel(tcc, tx, i); err != nil {
return retryAfter, canCommit, errors.New(action.Name + ": " + err.Error())
}
}
}
return 0, true, nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/go-better/go.git
git@gitee.com:go-better/go.git
go-better
go
go
d31700df43a9

搜索帮助