Zhen Ye 19e5e9f910
enhance: broadcaster will lock resource until message acked (#44508)
issue: #43897

- Return LastConfirmedMessageID when wal append operation.
- Add resource-key-based locker for broadcast-ack operation to protect
the coord state when executing ddl.
- Resource-key-based locker is held until the broadcast operation is
acked.
- ResourceKey support shared and exclusive lock.
- Add FastAck execute ack right away after the broadcast done to speed
up ddl.
- Ack callback will support broadcast message result now.
- Add tombstone for broadcaster to avoid to repeatedly commit DDL and
ABA issue.

---------

Signed-off-by: chyezh <chyezh@outlook.com>
2025-09-24 20:58:05 +08:00

108 lines
3.5 KiB
Go

package utility
import (
"context"
"reflect"
"google.golang.org/protobuf/reflect/protoreflect"
"github.com/milvus-io/milvus/internal/streamingnode/server/wal/metricsutil"
"github.com/milvus-io/milvus/pkg/v2/streaming/util/message"
)
// walCtxKey is the key type of extra append result.
type walCtxKey int
var (
extraAppendResultValue walCtxKey = 1
notPersistedValue walCtxKey = 2
metricsValue walCtxKey = 3
flushFromOldArchValue walCtxKey = 4
)
// ExtraAppendResult is the extra append result.
type ExtraAppendResult struct {
TimeTick uint64
TxnCtx *message.TxnContext
Extra protoreflect.ProtoMessage
LastConfirmedMessageID message.MessageID
}
// NotPersistedHint is the hint of not persisted message.
type NotPersistedHint struct {
MessageID message.MessageID // The reused MessageID.
}
// WithNotPersisted set not persisted message to context
func WithNotPersisted(ctx context.Context, hint *NotPersistedHint) context.Context {
return context.WithValue(ctx, notPersistedValue, hint)
}
// GetNotPersisted get not persisted message from context
func GetNotPersisted(ctx context.Context) *NotPersistedHint {
val := ctx.Value(notPersistedValue)
if val == nil {
return nil
}
return val.(*NotPersistedHint)
}
// WithExtraAppendResult set extra to context
func WithExtraAppendResult(ctx context.Context, r *ExtraAppendResult) context.Context {
return context.WithValue(ctx, extraAppendResultValue, r)
}
// ModifyAppendResultExtra modify extra in context
func ModifyAppendResultExtra[M protoreflect.ProtoMessage](ctx context.Context, modifier func(old M) (new M)) {
result := ctx.Value(extraAppendResultValue)
var old M
if result.(*ExtraAppendResult).Extra != nil {
old = result.(*ExtraAppendResult).Extra.(M)
}
new := modifier(old)
if reflect.ValueOf(new).IsNil() {
result.(*ExtraAppendResult).Extra = nil
return
}
result.(*ExtraAppendResult).Extra = new
}
// ReplaceAppendResultLastConfirmedMessageID set last confirmed message id to context
func ReplaceAppendResultLastConfirmedMessageID(ctx context.Context, lastConfirmedMessageID message.MessageID) {
result := ctx.Value(extraAppendResultValue)
result.(*ExtraAppendResult).LastConfirmedMessageID = lastConfirmedMessageID
}
// ReplaceAppendResultTimeTick set time tick to context
func ReplaceAppendResultTimeTick(ctx context.Context, timeTick uint64) {
result := ctx.Value(extraAppendResultValue)
result.(*ExtraAppendResult).TimeTick = timeTick
}
// ReplaceAppendResultTxnContext set txn context to context
func ReplaceAppendResultTxnContext(ctx context.Context, txnCtx *message.TxnContext) {
result := ctx.Value(extraAppendResultValue)
result.(*ExtraAppendResult).TxnCtx = txnCtx
}
// WithAppendMetricsContext create a context with metrics recording.
func WithAppendMetricsContext(ctx context.Context, m *metricsutil.AppendMetrics) context.Context {
return context.WithValue(ctx, metricsValue, m)
}
// MustGetAppendMetrics get append metrics from context
func MustGetAppendMetrics(ctx context.Context) *metricsutil.AppendMetrics {
return ctx.Value(metricsValue).(*metricsutil.AppendMetrics)
}
// WithFlushFromOldArch set flush from old arch to context
func WithFlushFromOldArch(ctx context.Context) context.Context {
return context.WithValue(ctx, flushFromOldArchValue, struct{}{})
}
// GetFlushFromOldArch get flush from old arch from context
func GetFlushFromOldArch(ctx context.Context) bool {
result := ctx.Value(flushFromOldArchValue)
return result != nil
}