代码拉取完成,页面将自动刷新
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
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。