2024-06-11 02:38:01 +00:00
|
|
|
package message
|
|
|
|
|
2024-07-22 12:59:42 +00:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
|
|
|
|
"github.com/cockroachdb/errors"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/msgpb"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewMutableMessage creates a new mutable message.
|
|
|
|
// Only used at server side for streamingnode internal service, don't use it at client side.
|
|
|
|
func NewMutableMessage(payload []byte, properties map[string]string) MutableMessage {
|
|
|
|
return &messageImpl{
|
|
|
|
payload: payload,
|
|
|
|
properties: properties,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-02 06:42:08 +00:00
|
|
|
// NewImmutableMessage creates a new immutable message.
|
|
|
|
func NewImmutableMesasge(
|
|
|
|
id MessageID,
|
|
|
|
payload []byte,
|
|
|
|
properties map[string]string,
|
|
|
|
) ImmutableMessage {
|
|
|
|
return &immutableMessageImpl{
|
|
|
|
id: id,
|
|
|
|
messageImpl: messageImpl{
|
|
|
|
payload: payload,
|
|
|
|
properties: properties,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-22 12:59:42 +00:00
|
|
|
// List all type-safe mutable message builders here.
|
|
|
|
var (
|
|
|
|
NewTimeTickMessageBuilderV1 = createNewMessageBuilderV1[*TimeTickMessageHeader, *msgpb.TimeTickMsg]()
|
|
|
|
NewInsertMessageBuilderV1 = createNewMessageBuilderV1[*InsertMessageHeader, *msgpb.InsertRequest]()
|
|
|
|
NewDeleteMessageBuilderV1 = createNewMessageBuilderV1[*DeleteMessageHeader, *msgpb.DeleteRequest]()
|
|
|
|
NewCreateCollectionMessageBuilderV1 = createNewMessageBuilderV1[*CreateCollectionMessageHeader, *msgpb.CreateCollectionRequest]()
|
|
|
|
NewDropCollectionMessageBuilderV1 = createNewMessageBuilderV1[*DropCollectionMessageHeader, *msgpb.DropCollectionRequest]()
|
|
|
|
NewCreatePartitionMessageBuilderV1 = createNewMessageBuilderV1[*CreatePartitionMessageHeader, *msgpb.CreatePartitionRequest]()
|
|
|
|
NewDropPartitionMessageBuilderV1 = createNewMessageBuilderV1[*DropPartitionMessageHeader, *msgpb.DropPartitionRequest]()
|
|
|
|
)
|
|
|
|
|
|
|
|
// createNewMessageBuilderV1 creates a new message builder with v1 marker.
|
2024-07-25 03:57:45 +00:00
|
|
|
func createNewMessageBuilderV1[H proto.Message, B proto.Message]() func() *mutableMesasgeBuilder[H, B] {
|
|
|
|
return func() *mutableMesasgeBuilder[H, B] {
|
|
|
|
return newMutableMessageBuilder[H, B](VersionV1)
|
2024-07-22 12:59:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// newMutableMessageBuilder creates a new builder.
|
2024-07-02 06:42:08 +00:00
|
|
|
// Should only used at client side.
|
2024-07-25 03:57:45 +00:00
|
|
|
func newMutableMessageBuilder[H proto.Message, B proto.Message](v Version) *mutableMesasgeBuilder[H, B] {
|
2024-07-22 12:59:42 +00:00
|
|
|
var h H
|
2024-07-25 03:57:45 +00:00
|
|
|
messageType := mustGetMessageTypeFromHeader(h)
|
2024-07-22 12:59:42 +00:00
|
|
|
properties := make(propertiesImpl)
|
|
|
|
properties.Set(messageTypeKey, messageType.marshal())
|
|
|
|
properties.Set(messageVersion, v.String())
|
2024-07-25 03:57:45 +00:00
|
|
|
return &mutableMesasgeBuilder[H, B]{
|
2024-07-22 12:59:42 +00:00
|
|
|
properties: properties,
|
2024-06-11 02:38:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-22 12:59:42 +00:00
|
|
|
// mutableMesasgeBuilder is the builder for message.
|
2024-07-25 03:57:45 +00:00
|
|
|
type mutableMesasgeBuilder[H proto.Message, B proto.Message] struct {
|
2024-07-22 12:59:42 +00:00
|
|
|
header H
|
2024-07-25 03:57:45 +00:00
|
|
|
body B
|
2024-06-11 02:38:01 +00:00
|
|
|
properties propertiesImpl
|
|
|
|
}
|
|
|
|
|
2024-07-22 12:59:42 +00:00
|
|
|
// WithMessageHeader creates a new builder with determined message type.
|
2024-07-25 03:57:45 +00:00
|
|
|
func (b *mutableMesasgeBuilder[H, B]) WithHeader(h H) *mutableMesasgeBuilder[H, B] {
|
2024-07-22 12:59:42 +00:00
|
|
|
b.header = h
|
2024-06-11 02:38:01 +00:00
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2024-07-25 03:57:45 +00:00
|
|
|
// WithBody creates a new builder with message body.
|
|
|
|
func (b *mutableMesasgeBuilder[H, B]) WithBody(body B) *mutableMesasgeBuilder[H, B] {
|
|
|
|
b.body = body
|
2024-06-11 02:38:01 +00:00
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithProperty creates a new builder with message property.
|
2024-07-16 07:49:38 +00:00
|
|
|
// A key started with '_' is reserved for streaming system, should never used at user of client.
|
2024-07-25 03:57:45 +00:00
|
|
|
func (b *mutableMesasgeBuilder[H, B]) WithProperty(key string, val string) *mutableMesasgeBuilder[H, B] {
|
2024-07-22 12:59:42 +00:00
|
|
|
if b.properties.Exist(key) {
|
|
|
|
panic(fmt.Sprintf("message builder already set property field, key = %s", key))
|
|
|
|
}
|
2024-06-11 02:38:01 +00:00
|
|
|
b.properties.Set(key, val)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithProperties creates a new builder with message properties.
|
2024-07-16 07:49:38 +00:00
|
|
|
// A key started with '_' is reserved for streaming system, should never used at user of client.
|
2024-07-25 03:57:45 +00:00
|
|
|
func (b *mutableMesasgeBuilder[H, B]) WithProperties(kvs map[string]string) *mutableMesasgeBuilder[H, B] {
|
2024-06-11 02:38:01 +00:00
|
|
|
for key, val := range kvs {
|
|
|
|
b.properties.Set(key, val)
|
|
|
|
}
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
// BuildMutable builds a mutable message.
|
2024-07-02 06:42:08 +00:00
|
|
|
// Panic if not set payload and message type.
|
|
|
|
// should only used at client side.
|
2024-07-25 03:57:45 +00:00
|
|
|
func (b *mutableMesasgeBuilder[H, B]) BuildMutable() (MutableMessage, error) {
|
2024-07-22 12:59:42 +00:00
|
|
|
// payload and header must be a pointer
|
|
|
|
if reflect.ValueOf(b.header).IsNil() {
|
|
|
|
panic("message builder not ready for header field")
|
|
|
|
}
|
2024-07-25 03:57:45 +00:00
|
|
|
if reflect.ValueOf(b.body).IsNil() {
|
|
|
|
panic("message builder not ready for body field")
|
2024-07-02 06:42:08 +00:00
|
|
|
}
|
2024-07-22 12:59:42 +00:00
|
|
|
|
|
|
|
// setup header.
|
|
|
|
sp, err := EncodeProto(b.header)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to encode header")
|
|
|
|
}
|
|
|
|
b.properties.Set(messageSpecialiedHeader, sp)
|
|
|
|
|
2024-07-25 03:57:45 +00:00
|
|
|
payload, err := proto.Marshal(b.body)
|
2024-07-22 12:59:42 +00:00
|
|
|
if err != nil {
|
2024-07-25 03:57:45 +00:00
|
|
|
return nil, errors.Wrap(err, "failed to marshal body")
|
2024-06-11 02:38:01 +00:00
|
|
|
}
|
|
|
|
return &messageImpl{
|
2024-07-22 12:59:42 +00:00
|
|
|
payload: payload,
|
2024-06-11 02:38:01 +00:00
|
|
|
properties: b.properties,
|
2024-07-22 12:59:42 +00:00
|
|
|
}, nil
|
2024-06-11 02:38:01 +00:00
|
|
|
}
|