2016-12-05 22:53:13 +00:00
|
|
|
package events
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var txCounter int64 // replace this with something that won't break
|
|
|
|
|
|
|
|
// nextTXID provides the next transaction identifier.
|
|
|
|
func nexttxID() int64 {
|
|
|
|
// TODO(stevvooe): Need to coordinate this with existing transaction logs.
|
|
|
|
// For now, this is a toy, but not a racy one.
|
|
|
|
return atomic.AddInt64(&txCounter, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
type transaction struct {
|
2016-12-12 22:26:51 +00:00
|
|
|
ctx context.Context
|
2016-12-05 22:53:13 +00:00
|
|
|
id int64
|
|
|
|
parent *transaction // if nil, no parent transaction
|
|
|
|
finish sync.Once
|
|
|
|
start, end time.Time // informational
|
|
|
|
}
|
|
|
|
|
|
|
|
// begin creates a sub-transaction.
|
2016-12-12 22:26:51 +00:00
|
|
|
func (tx *transaction) begin(ctx context.Context, poster Poster) *transaction {
|
2016-12-05 22:53:13 +00:00
|
|
|
id := nexttxID()
|
|
|
|
|
|
|
|
child := &transaction{
|
2016-12-12 22:26:51 +00:00
|
|
|
ctx: ctx,
|
2016-12-05 22:53:13 +00:00
|
|
|
id: id,
|
|
|
|
parent: tx,
|
|
|
|
start: time.Now(),
|
|
|
|
}
|
|
|
|
|
|
|
|
// post the transaction started event
|
2017-02-21 12:24:04 +00:00
|
|
|
poster.Post(ctx, child.makeTransactionEvent("begin")) // transactions are really just events
|
2016-12-05 22:53:13 +00:00
|
|
|
|
|
|
|
return child
|
|
|
|
}
|
|
|
|
|
|
|
|
// commit writes out the transaction.
|
|
|
|
func (tx *transaction) commit(poster Poster) {
|
|
|
|
tx.finish.Do(func() {
|
|
|
|
tx.end = time.Now()
|
2016-12-12 22:26:51 +00:00
|
|
|
poster.Post(tx.ctx, tx.makeTransactionEvent("commit"))
|
2016-12-05 22:53:13 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *transaction) rollback(poster Poster, cause error) {
|
|
|
|
tx.finish.Do(func() {
|
|
|
|
tx.end = time.Now()
|
|
|
|
event := tx.makeTransactionEvent("rollback")
|
|
|
|
event = fmt.Sprintf("%s error=%q", event, cause.Error())
|
2016-12-12 22:26:51 +00:00
|
|
|
poster.Post(tx.ctx, event)
|
2016-12-05 22:53:13 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tx *transaction) makeTransactionEvent(status string) Event {
|
|
|
|
// TODO(stevvooe): obviously, we need more structure than this.
|
|
|
|
event := fmt.Sprintf("%v %v", status, tx.id)
|
|
|
|
if tx.parent != nil {
|
|
|
|
event += " parent=" + fmt.Sprint(tx.parent.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
return event
|
|
|
|
}
|
|
|
|
|
|
|
|
type txKey struct{}
|
|
|
|
|
|
|
|
func getTx(ctx context.Context) (*transaction, bool) {
|
|
|
|
tx := ctx.Value(txKey{})
|
|
|
|
if tx == nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
return tx.(*transaction), true
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithTx returns a new context with an event transaction, such that events
|
|
|
|
// posted to the underlying context will be committed to the event log as a
|
|
|
|
// group, organized by a transaction id, when commit is called.
|
|
|
|
func WithTx(pctx context.Context) (ctx context.Context, commit func(), rollback func(err error)) {
|
|
|
|
poster := G(pctx)
|
|
|
|
parent, _ := getTx(pctx)
|
2016-12-12 22:26:51 +00:00
|
|
|
tx := parent.begin(pctx, poster)
|
2016-12-05 22:53:13 +00:00
|
|
|
|
|
|
|
return context.WithValue(pctx, txKey{}, tx), func() {
|
|
|
|
tx.commit(poster)
|
|
|
|
}, func(err error) {
|
|
|
|
tx.rollback(poster, err)
|
|
|
|
}
|
|
|
|
}
|