mirror of
https://gitee.com/milvus-io/milvus.git
synced 2026-01-06 19:02:18 +08:00
135 lines
3.9 KiB
Go
135 lines
3.9 KiB
Go
package txn
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/milvus-io/milvus/internal/streamingnode/server/resource"
|
|
"github.com/milvus-io/milvus/internal/streamingnode/server/wal/metricsutil"
|
|
"github.com/milvus-io/milvus/internal/util/streamingutil/status"
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
"github.com/milvus-io/milvus/pkg/streaming/util/message"
|
|
"github.com/milvus-io/milvus/pkg/streaming/util/types"
|
|
"github.com/milvus-io/milvus/pkg/util/lifetime"
|
|
"github.com/milvus-io/milvus/pkg/util/paramtable"
|
|
)
|
|
|
|
// NewTxnManager creates a new transaction manager.
|
|
func NewTxnManager(pchannel types.PChannelInfo) *TxnManager {
|
|
return &TxnManager{
|
|
mu: sync.Mutex{},
|
|
sessions: make(map[message.TxnID]*TxnSession),
|
|
closed: nil,
|
|
metrics: metricsutil.NewTxnMetrics(pchannel.Name),
|
|
logger: resource.Resource().Logger().With(log.FieldComponent("txn-manager")),
|
|
}
|
|
}
|
|
|
|
// TxnManager is the manager of transactions.
|
|
// We don't support cross wal transaction by now and
|
|
// We don't support the transaction lives after the wal transferred to another streaming node.
|
|
type TxnManager struct {
|
|
mu sync.Mutex
|
|
sessions map[message.TxnID]*TxnSession
|
|
closed lifetime.SafeChan
|
|
metrics *metricsutil.TxnMetrics
|
|
logger *log.MLogger
|
|
}
|
|
|
|
// BeginNewTxn starts a new transaction with a session.
|
|
// We only support a transaction work on a streaming node, once the wal is transferred to another node,
|
|
// the transaction is treated as expired (rollback), and user will got a expired error, then perform a retry.
|
|
func (m *TxnManager) BeginNewTxn(ctx context.Context, timetick uint64, keepalive time.Duration) (*TxnSession, error) {
|
|
if keepalive == 0 {
|
|
// If keepalive is 0, the txn set the keepalive with default keepalive.
|
|
keepalive = paramtable.Get().StreamingCfg.TxnDefaultKeepaliveTimeout.GetAsDurationByParse()
|
|
}
|
|
if keepalive < 1*time.Millisecond {
|
|
return nil, status.NewInvaildArgument("keepalive must be greater than 1ms")
|
|
}
|
|
id, err := resource.Resource().IDAllocator().Allocate(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
// The manager is on graceful shutdown.
|
|
// Avoid creating new transactions.
|
|
if m.closed != nil {
|
|
return nil, status.NewTransactionExpired("manager closed")
|
|
}
|
|
session := &TxnSession{
|
|
mu: sync.Mutex{},
|
|
lastTimetick: timetick,
|
|
txnContext: message.TxnContext{
|
|
TxnID: message.TxnID(id),
|
|
Keepalive: keepalive,
|
|
},
|
|
inFlightCount: 0,
|
|
state: message.TxnStateBegin,
|
|
doneWait: nil,
|
|
rollback: false,
|
|
}
|
|
|
|
m.sessions[session.TxnContext().TxnID] = session
|
|
m.metrics.BeginTxn()
|
|
return session, nil
|
|
}
|
|
|
|
// CleanupTxnUntil cleans up the transactions until the specified timestamp.
|
|
func (m *TxnManager) CleanupTxnUntil(ts uint64) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
for id, session := range m.sessions {
|
|
if session.IsExpiredOrDone(ts) {
|
|
session.Cleanup()
|
|
m.metrics.Finish(session.State())
|
|
delete(m.sessions, id)
|
|
}
|
|
}
|
|
|
|
// If the manager is on graceful shutdown and all transactions are cleaned up.
|
|
if len(m.sessions) == 0 && m.closed != nil {
|
|
m.closed.Close()
|
|
}
|
|
}
|
|
|
|
// GetSessionOfTxn returns the session of the transaction.
|
|
func (m *TxnManager) GetSessionOfTxn(id message.TxnID) (*TxnSession, error) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
session, ok := m.sessions[id]
|
|
if !ok {
|
|
return nil, status.NewTransactionExpired("not found in manager")
|
|
}
|
|
return session, nil
|
|
}
|
|
|
|
// GracefulClose waits for all transactions to be cleaned up.
|
|
func (m *TxnManager) GracefulClose(ctx context.Context) error {
|
|
defer m.metrics.Close()
|
|
|
|
m.mu.Lock()
|
|
if m.closed == nil {
|
|
m.closed = lifetime.NewSafeChan()
|
|
if len(m.sessions) == 0 {
|
|
m.closed.Close()
|
|
}
|
|
}
|
|
m.logger.Info("there's still txn session in txn manager, waiting for them to be consumed", zap.Int("session count", len(m.sessions)))
|
|
m.mu.Unlock()
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
case <-m.closed.CloseCh():
|
|
return nil
|
|
}
|
|
}
|