Compare commits
	
		
			4 Commits
		
	
	
		
			29a3f73f15
			...
			v0.0.260
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6d7cfb86f8 | |||
| 1e9d663ffe | |||
| 5b8d7ebf87 | |||
| 11dc6d2640 | 
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @@ -14,7 +14,7 @@ require ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/bytedance/sonic v1.10.0 // indirect | 	github.com/bytedance/sonic v1.10.1 // indirect | ||||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | ||||||
| 	github.com/chenzhuoyu/iasm v0.9.0 // indirect | 	github.com/chenzhuoyu/iasm v0.9.0 // indirect | ||||||
| 	github.com/gabriel-vasile/mimetype v1.4.2 // indirect | 	github.com/gabriel-vasile/mimetype v1.4.2 // indirect | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.sum
									
									
									
									
									
								
							| @@ -6,6 +6,8 @@ github.com/bytedance/sonic v1.10.0-rc3 h1:uNSnscRapXTwUgTyOF0GVljYD08p9X/Lbr9Mwe | |||||||
| github.com/bytedance/sonic v1.10.0-rc3/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | github.com/bytedance/sonic v1.10.0-rc3/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
| github.com/bytedance/sonic v1.10.0 h1:qtNZduETEIWJVIyDl01BeNxur2rW9OwTQ/yBqFRkKEk= | github.com/bytedance/sonic v1.10.0 h1:qtNZduETEIWJVIyDl01BeNxur2rW9OwTQ/yBqFRkKEk= | ||||||
| github.com/bytedance/sonic v1.10.0/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | github.com/bytedance/sonic v1.10.0/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
|  | github.com/bytedance/sonic v1.10.1 h1:7a1wuFXL1cMy7a3f7/VFcEtriuXQnUBhtoVfOZiaysc= | ||||||
|  | github.com/bytedance/sonic v1.10.1/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d h1:77cEq6EriyTZ0g/qfRdp61a3Uu/AWrgIq2s0ClJV1g0= | github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d h1:77cEq6EriyTZ0g/qfRdp61a3Uu/AWrgIq2s0ClJV1g0= | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.258" | const GoextVersion = "0.0.260" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2023-09-11T11:28:34+0200" | const GoextVersionTimestamp = "2023-09-12T11:40:39+0200" | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ func (c *Coll[TData]) EnsureInitializedReflection(v TData) { | |||||||
|  |  | ||||||
| 	m := make(map[string]fullTypeRef) | 	m := make(map[string]fullTypeRef) | ||||||
|  |  | ||||||
| 	c.initFields("", rval, m, make([]int, 0)) | 	c.initFields("", rval.Type(), m, make([]int, 0), make([]reflect.Type, 0)) | ||||||
|  |  | ||||||
| 	c.implDataTypeMap[rval.Type()] = m | 	c.implDataTypeMap[rval.Type()] = m | ||||||
| } | } | ||||||
| @@ -50,20 +50,16 @@ func (c *Coll[TData]) init() { | |||||||
| 		c.implDataTypeMap = make(map[reflect.Type]map[string]fullTypeRef) | 		c.implDataTypeMap = make(map[reflect.Type]map[string]fullTypeRef) | ||||||
|  |  | ||||||
| 		v := reflect.ValueOf(example) | 		v := reflect.ValueOf(example) | ||||||
| 		c.initFields("", v, c.dataTypeMap, make([]int, 0)) | 		c.initFields("", v.Type(), c.dataTypeMap, make([]int, 0), make([]reflect.Type, 0)) | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Coll[TData]) initFields(prefix string, rval reflect.Value, m map[string]fullTypeRef, idxarr []int) { | func (c *Coll[TData]) initFields(prefix string, rtyp reflect.Type, m map[string]fullTypeRef, idxarr []int, typesInPath []reflect.Type) { | ||||||
|  |  | ||||||
| 	rtyp := rval.Type() |  | ||||||
|  |  | ||||||
| 	for i := 0; i < rtyp.NumField(); i++ { | 	for i := 0; i < rtyp.NumField(); i++ { | ||||||
|  |  | ||||||
| 		rsfield := rtyp.Field(i) | 		rsfield := rtyp.Field(i) | ||||||
| 		rvfield := rval.Field(i) |  | ||||||
|  |  | ||||||
| 		if !rsfield.IsExported() { | 		if !rsfield.IsExported() { | ||||||
| 			continue | 			continue | ||||||
| @@ -91,21 +87,21 @@ func (c *Coll[TData]) initFields(prefix string, rval reflect.Value, m map[string | |||||||
| 		newIdxArr := langext.ArrCopy(idxarr) | 		newIdxArr := langext.ArrCopy(idxarr) | ||||||
| 		newIdxArr = append(newIdxArr, i) | 		newIdxArr = append(newIdxArr, i) | ||||||
|  |  | ||||||
| 		if langext.InArray("inline", bsontags) && rvfield.Kind() == reflect.Struct { | 		if langext.InArray("inline", bsontags) && rsfield.Type.Kind() == reflect.Struct { | ||||||
|  |  | ||||||
| 			// pass-through field | 			// pass-through field | ||||||
| 			c.initFields(prefix, rvfield, m, newIdxArr) | 			c.initFields(prefix, rsfield.Type, m, newIdxArr, typesInPath) | ||||||
|  |  | ||||||
| 		} else { | 		} else { | ||||||
|  |  | ||||||
| 			if rvfield.Type().Kind() == reflect.Pointer { | 			if rsfield.Type.Kind() == reflect.Pointer { | ||||||
|  |  | ||||||
| 				m[fullKey] = fullTypeRef{ | 				m[fullKey] = fullTypeRef{ | ||||||
| 					IsPointer:      true, | 					IsPointer:      true, | ||||||
| 					RealType:       rvfield.Type(), | 					RealType:       rsfield.Type, | ||||||
| 					Kind:           rvfield.Type().Elem().Kind(), | 					Kind:           rsfield.Type.Elem().Kind(), | ||||||
| 					Type:           rvfield.Type().Elem(), | 					Type:           rsfield.Type.Elem(), | ||||||
| 					UnderlyingType: reflectext.Underlying(rvfield.Type().Elem()), | 					UnderlyingType: reflectext.Underlying(rsfield.Type.Elem()), | ||||||
| 					Name:           rsfield.Name, | 					Name:           rsfield.Name, | ||||||
| 					Index:          newIdxArr, | 					Index:          newIdxArr, | ||||||
| 				} | 				} | ||||||
| @@ -114,20 +110,37 @@ func (c *Coll[TData]) initFields(prefix string, rval reflect.Value, m map[string | |||||||
|  |  | ||||||
| 				m[fullKey] = fullTypeRef{ | 				m[fullKey] = fullTypeRef{ | ||||||
| 					IsPointer:      false, | 					IsPointer:      false, | ||||||
| 					RealType:       rvfield.Type(), | 					RealType:       rsfield.Type, | ||||||
| 					Kind:           rvfield.Type().Kind(), | 					Kind:           rsfield.Type.Kind(), | ||||||
| 					Type:           rvfield.Type(), | 					Type:           rsfield.Type, | ||||||
| 					UnderlyingType: reflectext.Underlying(rvfield.Type()), | 					UnderlyingType: reflectext.Underlying(rsfield.Type), | ||||||
| 					Name:           rsfield.Name, | 					Name:           rsfield.Name, | ||||||
| 					Index:          newIdxArr, | 					Index:          newIdxArr, | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if rvfield.Kind() == reflect.Struct { | 			if rsfield.Type.Kind() == reflect.Struct { | ||||||
| 				c.initFields(fullKey+".", rvfield, m, newIdxArr) | 				c.initFields(fullKey+".", rsfield.Type, m, newIdxArr, typesInPath) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			if rsfield.Type.Kind() == reflect.Pointer && rsfield.Type.Elem().Kind() == reflect.Struct { | ||||||
|  | 				innerType := rsfield.Type.Elem() | ||||||
|  |  | ||||||
|  | 				// check if there is recursion | ||||||
|  | 				recursion := false | ||||||
|  | 				for _, typ := range typesInPath { | ||||||
|  | 					recursion = recursion || (typ == innerType) | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				if !recursion { | ||||||
|  | 					// Store all seen types before that deref a pointer to prevent endless recursion | ||||||
|  | 					newTypesInPath := make([]reflect.Type, len(typesInPath)) | ||||||
|  | 					copy(newTypesInPath, typesInPath) | ||||||
|  | 					newTypesInPath = append(newTypesInPath, rtyp) | ||||||
|  | 					c.initFields(fullKey+".", innerType, m, newIdxArr, newTypesInPath) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -23,6 +23,9 @@ func TestReflectionGetFieldType(t *testing.T) { | |||||||
| 		Sub   struct { | 		Sub   struct { | ||||||
| 			A string `bson:"a"` | 			A string `bson:"a"` | ||||||
| 		} `bson:"sub"` | 		} `bson:"sub"` | ||||||
|  | 		SubPtr *struct { | ||||||
|  | 			A string `bson:"a"` | ||||||
|  | 		} `bson:"subPtr"` | ||||||
| 		Str   string                  `bson:"str"` | 		Str   string                  `bson:"str"` | ||||||
| 		Ptr   *int                    `bson:"ptr"` | 		Ptr   *int                    `bson:"ptr"` | ||||||
| 		MDate rfctime.RFC3339NanoTime `bson:"mdate"` | 		MDate rfctime.RFC3339NanoTime `bson:"mdate"` | ||||||
| @@ -43,6 +46,11 @@ func TestReflectionGetFieldType(t *testing.T) { | |||||||
| 		}{ | 		}{ | ||||||
| 			A: "2", | 			A: "2", | ||||||
| 		}, | 		}, | ||||||
|  | 		SubPtr: &struct { | ||||||
|  | 			A string `bson:"a"` | ||||||
|  | 		}{ | ||||||
|  | 			A: "4", | ||||||
|  | 		}, | ||||||
| 		Str:   "3", | 		Str:   "3", | ||||||
| 		Ptr:   langext.Ptr(4), | 		Ptr:   langext.Ptr(4), | ||||||
| 		MDate: t1, | 		MDate: t1, | ||||||
| @@ -82,6 +90,12 @@ func TestReflectionGetFieldType(t *testing.T) { | |||||||
| 	tst.AssertEqual(t, gft("sub.a").IsPointer, false) | 	tst.AssertEqual(t, gft("sub.a").IsPointer, false) | ||||||
| 	tst.AssertEqual(t, gfv("sub.a").(string), "2") | 	tst.AssertEqual(t, gfv("sub.a").(string), "2") | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, gft("subPtr.a").Kind.String(), "string") | ||||||
|  | 	tst.AssertEqual(t, gft("subPtr.a").Type.String(), "string") | ||||||
|  | 	tst.AssertEqual(t, gft("subPtr.a").Name, "A") | ||||||
|  | 	tst.AssertEqual(t, gft("subPtr.a").IsPointer, false) | ||||||
|  | 	tst.AssertEqual(t, gfv("subPtr.a").(string), "4") | ||||||
|  |  | ||||||
| 	tst.AssertEqual(t, gft("str").Kind.String(), "string") | 	tst.AssertEqual(t, gft("str").Kind.String(), "string") | ||||||
| 	tst.AssertEqual(t, gft("str").Type.String(), "string") | 	tst.AssertEqual(t, gft("str").Type.String(), "string") | ||||||
| 	tst.AssertEqual(t, gft("str").Name, "Str") | 	tst.AssertEqual(t, gft("str").Name, "Str") | ||||||
| @@ -99,16 +113,25 @@ func TestReflectionGetTokenValueAsMongoType(t *testing.T) { | |||||||
|  |  | ||||||
| 	type IDType string | 	type IDType string | ||||||
|  |  | ||||||
|  | 	type RecurseiveType struct { | ||||||
|  | 		Other int             `bson:"other"` | ||||||
|  | 		Inner *RecurseiveType `bson:"inner"` | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	type TestData struct { | 	type TestData struct { | ||||||
| 		ID    IDType    `bson:"_id"` | 		ID    IDType    `bson:"_id"` | ||||||
| 		CDate time.Time `bson:"cdate"` | 		CDate time.Time `bson:"cdate"` | ||||||
| 		Sub   struct { | 		Sub   struct { | ||||||
| 			A string `bson:"a"` | 			A string `bson:"a"` | ||||||
| 		} `bson:"sub"` | 		} `bson:"sub"` | ||||||
|  | 		SubPtr *struct { | ||||||
|  | 			A string `bson:"a"` | ||||||
|  | 		} `bson:"subPtr"` | ||||||
| 		Str   string                  `bson:"str"` | 		Str   string                  `bson:"str"` | ||||||
| 		Ptr   *int                    `bson:"ptr"` | 		Ptr   *int                    `bson:"ptr"` | ||||||
| 		Num   int                     `bson:"num"` | 		Num   int                     `bson:"num"` | ||||||
| 		MDate rfctime.RFC3339NanoTime `bson:"mdate"` | 		MDate rfctime.RFC3339NanoTime `bson:"mdate"` | ||||||
|  | 		Rec   RecurseiveType          `bson:"rec"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	coll := W[TestData](&mongo.Collection{}) | 	coll := W[TestData](&mongo.Collection{}) | ||||||
| @@ -130,6 +153,9 @@ func TestReflectionGetTokenValueAsMongoType(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	tst.AssertEqual(t, gtvasmt("hello", "str").(string), "hello") | 	tst.AssertEqual(t, gtvasmt("hello", "str").(string), "hello") | ||||||
|  | 	tst.AssertEqual(t, gtvasmt("hello", "sub.a").(string), "hello") | ||||||
|  | 	tst.AssertEqual(t, gtvasmt("hello", "subPtr.a").(string), "hello") | ||||||
|  | 	tst.AssertEqual(t, gtvasmt("4", "rec.other").(int), 4) | ||||||
| 	tst.AssertEqual(t, gtvasmt("4", "num").(int), 4) | 	tst.AssertEqual(t, gtvasmt("4", "num").(int), 4) | ||||||
| 	tst.AssertEqual(t, gtvasmt("asdf", "_id").(IDType), "asdf") | 	tst.AssertEqual(t, gtvasmt("asdf", "_id").(IDType), "asdf") | ||||||
| 	tst.AssertEqual(t, gtvasmt("", "ptr").(*int), nil) | 	tst.AssertEqual(t, gtvasmt("", "ptr").(*int), nil) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user