aboutsummaryrefslogtreecommitdiff
path: root/weed/mq/schema/struct_to_schema.go
blob: 2f0f2180bbc076c75bc8213232e17afaef684e4e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
package schema

import (
	"reflect"

	"github.com/seaweedfs/seaweedfs/weed/pb/schema_pb"
)

func StructToSchema(instance any) *schema_pb.RecordType {
	myType := reflect.TypeOf(instance)
	if myType.Kind() != reflect.Struct {
		return nil
	}
	st := reflectTypeToSchemaType(myType)
	return st.GetRecordType()
}

// CreateCombinedRecordType creates a combined RecordType that includes fields from both key and value schemas
// Key fields are prefixed with "key_" to distinguish them from value fields
func CreateCombinedRecordType(keyRecordType *schema_pb.RecordType, valueRecordType *schema_pb.RecordType) *schema_pb.RecordType {
	var combinedFields []*schema_pb.Field

	// Add key fields with "key_" prefix
	if keyRecordType != nil {
		for _, field := range keyRecordType.Fields {
			keyField := &schema_pb.Field{
				Name:       "key_" + field.Name,
				FieldIndex: field.FieldIndex, // Will be reindexed later
				Type:       field.Type,
				IsRepeated: field.IsRepeated,
				IsRequired: field.IsRequired,
			}
			combinedFields = append(combinedFields, keyField)
		}
	}

	// Add value fields (no prefix)
	if valueRecordType != nil {
		for _, field := range valueRecordType.Fields {
			combinedFields = append(combinedFields, field)
		}
	}

	// Reindex all fields to have sequential indices
	for i, field := range combinedFields {
		field.FieldIndex = int32(i)
	}

	return &schema_pb.RecordType{
		Fields: combinedFields,
	}
}

func reflectTypeToSchemaType(t reflect.Type) *schema_pb.Type {
	switch t.Kind() {
	case reflect.Bool:
		return TypeBoolean
	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32:
		return TypeInt32
	case reflect.Int64:
		return TypeInt64
	case reflect.Float32:
		return TypeFloat
	case reflect.Float64:
		return TypeDouble
	case reflect.String:
		return TypeString
	case reflect.Slice:
		switch t.Elem().Kind() {
		case reflect.Uint8:
			return TypeBytes
		default:
			if st := reflectTypeToSchemaType(t.Elem()); st != nil {
				return &schema_pb.Type{
					Kind: &schema_pb.Type_ListType{
						ListType: &schema_pb.ListType{
							ElementType: st,
						},
					},
				}
			}
		}
	case reflect.Struct:
		recordType := &schema_pb.RecordType{}
		for i := 0; i < t.NumField(); i++ {
			field := t.Field(i)
			fieldType := field.Type
			fieldName := field.Name
			schemaField := reflectTypeToSchemaType(fieldType)
			if schemaField == nil {
				return nil
			}
			recordType.Fields = append(recordType.Fields, &schema_pb.Field{
				Name: fieldName,
				Type: schemaField,
			})
		}
		return &schema_pb.Type{
			Kind: &schema_pb.Type_RecordType{
				RecordType: recordType,
			},
		}
	}
	return nil
}