mirror of
https://gitee.com/milvus-io/milvus.git
synced 2026-01-07 19:31:51 +08:00
issue: #33285 - support transaction on single wal. - last confirmed message id can still be used when enable transaction. - add fence operation for segment allocation interceptor. --------- Signed-off-by: chyezh <chyezh@outlook.com>
107 lines
2.7 KiB
Go
107 lines
2.7 KiB
Go
package txn
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/milvus-io/milvus/internal/streamingnode/server/resource"
|
|
"github.com/milvus-io/milvus/internal/util/streamingutil/status"
|
|
"github.com/milvus-io/milvus/pkg/streaming/util/message"
|
|
)
|
|
|
|
// NewTxnManager creates a new transaction manager.
|
|
func NewTxnManager() *TxnManager {
|
|
return &TxnManager{
|
|
mu: sync.Mutex{},
|
|
sessions: make(map[message.TxnID]*TxnSession),
|
|
closed: nil,
|
|
}
|
|
}
|
|
|
|
// 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 chan struct{}
|
|
}
|
|
|
|
// 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) {
|
|
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
|
|
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()
|
|
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 {
|
|
close(m.closed)
|
|
}
|
|
}
|
|
|
|
// 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() {
|
|
m.mu.Lock()
|
|
if m.closed == nil {
|
|
m.closed = make(chan struct{})
|
|
if len(m.sessions) == 0 {
|
|
close(m.closed)
|
|
}
|
|
}
|
|
m.mu.Unlock()
|
|
|
|
<-m.closed
|
|
}
|