1 Star 0 Fork 0

go-better/go

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
engine.go 3.42 KB
一键复制 编辑 原始数据 按行查看 历史
bughou 提交于 2022-03-21 22:24 +08:00 . save
package tcc
import (
"context"
"database/sql"
"encoding/json"
"errors"
"fmt"
"reflect"
"sync"
"time"
"gitee.com/go-better/dev/db/sqlmq"
)
type Engine struct {
sqlmq *sqlmq.SqlMQ
mqName string
mqTableName string
actions map[string]Action
mutex sync.RWMutex
}
type Action interface {
Name() string
Try() error
Confirm() error
Cancel() error
}
// name must be unique for the same mq.
func NewEngine(name string, mq *sqlmq.SqlMQ) *Engine {
stdTable := mq.Table.(*sqlmq.StdTable)
engine := &Engine{
sqlmq: mq,
mqName: "tcc-" + name,
mqTableName: stdTable.Name(),
actions: make(map[string]Action),
}
if err := mq.Register(engine.mqName, engine.handle); err != nil {
panic(time.Now().Format(time.RFC3339Nano) + " " + err.Error())
}
return engine
}
func (engine *Engine) Register(actions ...Action) {
engine.mutex.Lock()
defer engine.mutex.Unlock()
for _, action := range actions {
if name := action.Name(); engine.actions[name] != nil {
panic(time.Now().Format(time.RFC3339Nano) + " action " + name + " already registered")
} else {
engine.actions[name] = action
}
}
}
var errTccId = errors.New("tcc id error")
func (engine *Engine) New(timeout time.Duration, concurrent bool) (*TCC, error) {
now := time.Now()
msg := &sqlmq.StdMessage{
Queue: engine.mqName,
Data: &tccData{
Status: statusTrying,
Concurrent: concurrent,
},
CreatedAt: now,
RetryAt: now.Add(timeout),
}
if err := engine.sqlmq.Produce(nil, msg); err != nil {
return nil, err
}
if msg.Id <= 0 {
return nil, errTccId
}
return &TCC{engine: engine, msg: msg}, nil
}
func (engine *Engine) Run(timeout time.Duration, concurrent bool, actions ...Action) error {
tcc, err := engine.New(timeout, concurrent)
if err != nil {
return err
}
for _, action := range actions {
if err := tcc.Try(action); err != nil {
if err2 := tcc.Cancel(); err2 != nil {
engine.sqlmq.Logger.Error(err2)
}
return err
}
}
return tcc.Confirm()
}
func (engine *Engine) checkAction(tried Action) error {
name := tried.Name()
engine.mutex.RLock()
registered := engine.actions[name]
engine.mutex.RUnlock()
if registered == nil {
return fmt.Errorf("action %s is not registered", name)
}
if reflect.TypeOf(registered) != reflect.TypeOf(tried) {
return fmt.Errorf(
`action %s has been registered with type "%T", but tried with type "%T"`,
name, registered, tried,
)
}
return nil
}
func (engine *Engine) handle(ctx context.Context, tx *sql.Tx, message sqlmq.Message) (
time.Duration, bool, error,
) {
msg := message.(*sqlmq.StdMessage)
data := &tccData{}
if err := json.Unmarshal(msg.Data.([]byte), &data); err != nil {
return time.Hour, true, err
}
msg.Data = data
if retryAfter, canCommit, err := (&TCC{engine: engine, msg: msg}).confirmOrCancel(tx); err != nil {
if retryAfter <= 0 {
retryAfter = sqlmq.GetRetryWait(msg.TriedCount)
}
return retryAfter, canCommit, err
}
return 0, true, nil
}
var errActionNotRegistered = errors.New("action not registered")
func (engine *Engine) unmarshalAction(name string, b []byte) (Action, error) {
engine.mutex.RLock()
action := engine.actions[name]
engine.mutex.RUnlock()
if action == nil {
return nil, errActionNotRegistered
}
actionPointer := reflect.New(reflect.TypeOf(action))
if err := json.Unmarshal(b, actionPointer.Interface()); err != nil {
return nil, err
}
return actionPointer.Elem().Interface().(Action), nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/go-better/go.git
git@gitee.com:go-better/go.git
go-better
go
go
d31700df43a9

搜索帮助