mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-25 04:15:56 +03:00
62e6c9bc6c
* Add a storage layer for attachments * Fix some bug * fix test * Fix copyright head and lint * Fix bug * Add setting for minio and flags for migrate-storage * Add documents * fix lint * Add test for minio store type on attachments * fix test * fix test * Apply suggestions from code review Co-authored-by: guillep2k <18600385+guillep2k@users.noreply.github.com> * Add warning when storage migrated successfully * Fix drone * fix test * rebase * Fix test * display the error on console * Move minio test to amd64 since minio docker don't support arm64 * refactor the codes * add trace * Fix test * remove log on xorm * Fi download bug * Add a storage layer for attachments * Add setting for minio and flags for migrate-storage * fix lint * Add test for minio store type on attachments * Apply suggestions from code review Co-authored-by: guillep2k <18600385+guillep2k@users.noreply.github.com> * Fix drone * fix test * Fix test * display the error on console * Move minio test to amd64 since minio docker don't support arm64 * refactor the codes * add trace * Fix test * Add URL function to serve attachments directly from S3/Minio * Add ability to enable/disable redirection in attachment configuration * Fix typo * Add a storage layer for attachments * Add setting for minio and flags for migrate-storage * fix lint * Add test for minio store type on attachments * Apply suggestions from code review Co-authored-by: guillep2k <18600385+guillep2k@users.noreply.github.com> * Fix drone * fix test * Fix test * display the error on console * Move minio test to amd64 since minio docker don't support arm64 * don't change unrelated files * Fix lint * Fix build * update go.mod and go.sum * Use github.com/minio/minio-go/v6 * Remove unused function * Upgrade minio to v7 and some other improvements * fix lint * Fix go mod Co-authored-by: guillep2k <18600385+guillep2k@users.noreply.github.com> Co-authored-by: Tyler <tystuyfzand@gmail.com>
483 lines
14 KiB
Go
Vendored
483 lines
14 KiB
Go
Vendored
package jsoniter
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/modern-go/reflect2"
|
|
"reflect"
|
|
"sort"
|
|
"strings"
|
|
"unicode"
|
|
"unsafe"
|
|
)
|
|
|
|
var typeDecoders = map[string]ValDecoder{}
|
|
var fieldDecoders = map[string]ValDecoder{}
|
|
var typeEncoders = map[string]ValEncoder{}
|
|
var fieldEncoders = map[string]ValEncoder{}
|
|
var extensions = []Extension{}
|
|
|
|
// StructDescriptor describe how should we encode/decode the struct
|
|
type StructDescriptor struct {
|
|
Type reflect2.Type
|
|
Fields []*Binding
|
|
}
|
|
|
|
// GetField get one field from the descriptor by its name.
|
|
// Can not use map here to keep field orders.
|
|
func (structDescriptor *StructDescriptor) GetField(fieldName string) *Binding {
|
|
for _, binding := range structDescriptor.Fields {
|
|
if binding.Field.Name() == fieldName {
|
|
return binding
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Binding describe how should we encode/decode the struct field
|
|
type Binding struct {
|
|
levels []int
|
|
Field reflect2.StructField
|
|
FromNames []string
|
|
ToNames []string
|
|
Encoder ValEncoder
|
|
Decoder ValDecoder
|
|
}
|
|
|
|
// Extension the one for all SPI. Customize encoding/decoding by specifying alternate encoder/decoder.
|
|
// Can also rename fields by UpdateStructDescriptor.
|
|
type Extension interface {
|
|
UpdateStructDescriptor(structDescriptor *StructDescriptor)
|
|
CreateMapKeyDecoder(typ reflect2.Type) ValDecoder
|
|
CreateMapKeyEncoder(typ reflect2.Type) ValEncoder
|
|
CreateDecoder(typ reflect2.Type) ValDecoder
|
|
CreateEncoder(typ reflect2.Type) ValEncoder
|
|
DecorateDecoder(typ reflect2.Type, decoder ValDecoder) ValDecoder
|
|
DecorateEncoder(typ reflect2.Type, encoder ValEncoder) ValEncoder
|
|
}
|
|
|
|
// DummyExtension embed this type get dummy implementation for all methods of Extension
|
|
type DummyExtension struct {
|
|
}
|
|
|
|
// UpdateStructDescriptor No-op
|
|
func (extension *DummyExtension) UpdateStructDescriptor(structDescriptor *StructDescriptor) {
|
|
}
|
|
|
|
// CreateMapKeyDecoder No-op
|
|
func (extension *DummyExtension) CreateMapKeyDecoder(typ reflect2.Type) ValDecoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateMapKeyEncoder No-op
|
|
func (extension *DummyExtension) CreateMapKeyEncoder(typ reflect2.Type) ValEncoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateDecoder No-op
|
|
func (extension *DummyExtension) CreateDecoder(typ reflect2.Type) ValDecoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateEncoder No-op
|
|
func (extension *DummyExtension) CreateEncoder(typ reflect2.Type) ValEncoder {
|
|
return nil
|
|
}
|
|
|
|
// DecorateDecoder No-op
|
|
func (extension *DummyExtension) DecorateDecoder(typ reflect2.Type, decoder ValDecoder) ValDecoder {
|
|
return decoder
|
|
}
|
|
|
|
// DecorateEncoder No-op
|
|
func (extension *DummyExtension) DecorateEncoder(typ reflect2.Type, encoder ValEncoder) ValEncoder {
|
|
return encoder
|
|
}
|
|
|
|
type EncoderExtension map[reflect2.Type]ValEncoder
|
|
|
|
// UpdateStructDescriptor No-op
|
|
func (extension EncoderExtension) UpdateStructDescriptor(structDescriptor *StructDescriptor) {
|
|
}
|
|
|
|
// CreateDecoder No-op
|
|
func (extension EncoderExtension) CreateDecoder(typ reflect2.Type) ValDecoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateEncoder get encoder from map
|
|
func (extension EncoderExtension) CreateEncoder(typ reflect2.Type) ValEncoder {
|
|
return extension[typ]
|
|
}
|
|
|
|
// CreateMapKeyDecoder No-op
|
|
func (extension EncoderExtension) CreateMapKeyDecoder(typ reflect2.Type) ValDecoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateMapKeyEncoder No-op
|
|
func (extension EncoderExtension) CreateMapKeyEncoder(typ reflect2.Type) ValEncoder {
|
|
return nil
|
|
}
|
|
|
|
// DecorateDecoder No-op
|
|
func (extension EncoderExtension) DecorateDecoder(typ reflect2.Type, decoder ValDecoder) ValDecoder {
|
|
return decoder
|
|
}
|
|
|
|
// DecorateEncoder No-op
|
|
func (extension EncoderExtension) DecorateEncoder(typ reflect2.Type, encoder ValEncoder) ValEncoder {
|
|
return encoder
|
|
}
|
|
|
|
type DecoderExtension map[reflect2.Type]ValDecoder
|
|
|
|
// UpdateStructDescriptor No-op
|
|
func (extension DecoderExtension) UpdateStructDescriptor(structDescriptor *StructDescriptor) {
|
|
}
|
|
|
|
// CreateMapKeyDecoder No-op
|
|
func (extension DecoderExtension) CreateMapKeyDecoder(typ reflect2.Type) ValDecoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateMapKeyEncoder No-op
|
|
func (extension DecoderExtension) CreateMapKeyEncoder(typ reflect2.Type) ValEncoder {
|
|
return nil
|
|
}
|
|
|
|
// CreateDecoder get decoder from map
|
|
func (extension DecoderExtension) CreateDecoder(typ reflect2.Type) ValDecoder {
|
|
return extension[typ]
|
|
}
|
|
|
|
// CreateEncoder No-op
|
|
func (extension DecoderExtension) CreateEncoder(typ reflect2.Type) ValEncoder {
|
|
return nil
|
|
}
|
|
|
|
// DecorateDecoder No-op
|
|
func (extension DecoderExtension) DecorateDecoder(typ reflect2.Type, decoder ValDecoder) ValDecoder {
|
|
return decoder
|
|
}
|
|
|
|
// DecorateEncoder No-op
|
|
func (extension DecoderExtension) DecorateEncoder(typ reflect2.Type, encoder ValEncoder) ValEncoder {
|
|
return encoder
|
|
}
|
|
|
|
type funcDecoder struct {
|
|
fun DecoderFunc
|
|
}
|
|
|
|
func (decoder *funcDecoder) Decode(ptr unsafe.Pointer, iter *Iterator) {
|
|
decoder.fun(ptr, iter)
|
|
}
|
|
|
|
type funcEncoder struct {
|
|
fun EncoderFunc
|
|
isEmptyFunc func(ptr unsafe.Pointer) bool
|
|
}
|
|
|
|
func (encoder *funcEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
|
|
encoder.fun(ptr, stream)
|
|
}
|
|
|
|
func (encoder *funcEncoder) IsEmpty(ptr unsafe.Pointer) bool {
|
|
if encoder.isEmptyFunc == nil {
|
|
return false
|
|
}
|
|
return encoder.isEmptyFunc(ptr)
|
|
}
|
|
|
|
// DecoderFunc the function form of TypeDecoder
|
|
type DecoderFunc func(ptr unsafe.Pointer, iter *Iterator)
|
|
|
|
// EncoderFunc the function form of TypeEncoder
|
|
type EncoderFunc func(ptr unsafe.Pointer, stream *Stream)
|
|
|
|
// RegisterTypeDecoderFunc register TypeDecoder for a type with function
|
|
func RegisterTypeDecoderFunc(typ string, fun DecoderFunc) {
|
|
typeDecoders[typ] = &funcDecoder{fun}
|
|
}
|
|
|
|
// RegisterTypeDecoder register TypeDecoder for a typ
|
|
func RegisterTypeDecoder(typ string, decoder ValDecoder) {
|
|
typeDecoders[typ] = decoder
|
|
}
|
|
|
|
// RegisterFieldDecoderFunc register TypeDecoder for a struct field with function
|
|
func RegisterFieldDecoderFunc(typ string, field string, fun DecoderFunc) {
|
|
RegisterFieldDecoder(typ, field, &funcDecoder{fun})
|
|
}
|
|
|
|
// RegisterFieldDecoder register TypeDecoder for a struct field
|
|
func RegisterFieldDecoder(typ string, field string, decoder ValDecoder) {
|
|
fieldDecoders[fmt.Sprintf("%s/%s", typ, field)] = decoder
|
|
}
|
|
|
|
// RegisterTypeEncoderFunc register TypeEncoder for a type with encode/isEmpty function
|
|
func RegisterTypeEncoderFunc(typ string, fun EncoderFunc, isEmptyFunc func(unsafe.Pointer) bool) {
|
|
typeEncoders[typ] = &funcEncoder{fun, isEmptyFunc}
|
|
}
|
|
|
|
// RegisterTypeEncoder register TypeEncoder for a type
|
|
func RegisterTypeEncoder(typ string, encoder ValEncoder) {
|
|
typeEncoders[typ] = encoder
|
|
}
|
|
|
|
// RegisterFieldEncoderFunc register TypeEncoder for a struct field with encode/isEmpty function
|
|
func RegisterFieldEncoderFunc(typ string, field string, fun EncoderFunc, isEmptyFunc func(unsafe.Pointer) bool) {
|
|
RegisterFieldEncoder(typ, field, &funcEncoder{fun, isEmptyFunc})
|
|
}
|
|
|
|
// RegisterFieldEncoder register TypeEncoder for a struct field
|
|
func RegisterFieldEncoder(typ string, field string, encoder ValEncoder) {
|
|
fieldEncoders[fmt.Sprintf("%s/%s", typ, field)] = encoder
|
|
}
|
|
|
|
// RegisterExtension register extension
|
|
func RegisterExtension(extension Extension) {
|
|
extensions = append(extensions, extension)
|
|
}
|
|
|
|
func getTypeDecoderFromExtension(ctx *ctx, typ reflect2.Type) ValDecoder {
|
|
decoder := _getTypeDecoderFromExtension(ctx, typ)
|
|
if decoder != nil {
|
|
for _, extension := range extensions {
|
|
decoder = extension.DecorateDecoder(typ, decoder)
|
|
}
|
|
decoder = ctx.decoderExtension.DecorateDecoder(typ, decoder)
|
|
for _, extension := range ctx.extraExtensions {
|
|
decoder = extension.DecorateDecoder(typ, decoder)
|
|
}
|
|
}
|
|
return decoder
|
|
}
|
|
func _getTypeDecoderFromExtension(ctx *ctx, typ reflect2.Type) ValDecoder {
|
|
for _, extension := range extensions {
|
|
decoder := extension.CreateDecoder(typ)
|
|
if decoder != nil {
|
|
return decoder
|
|
}
|
|
}
|
|
decoder := ctx.decoderExtension.CreateDecoder(typ)
|
|
if decoder != nil {
|
|
return decoder
|
|
}
|
|
for _, extension := range ctx.extraExtensions {
|
|
decoder := extension.CreateDecoder(typ)
|
|
if decoder != nil {
|
|
return decoder
|
|
}
|
|
}
|
|
typeName := typ.String()
|
|
decoder = typeDecoders[typeName]
|
|
if decoder != nil {
|
|
return decoder
|
|
}
|
|
if typ.Kind() == reflect.Ptr {
|
|
ptrType := typ.(*reflect2.UnsafePtrType)
|
|
decoder := typeDecoders[ptrType.Elem().String()]
|
|
if decoder != nil {
|
|
return &OptionalDecoder{ptrType.Elem(), decoder}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func getTypeEncoderFromExtension(ctx *ctx, typ reflect2.Type) ValEncoder {
|
|
encoder := _getTypeEncoderFromExtension(ctx, typ)
|
|
if encoder != nil {
|
|
for _, extension := range extensions {
|
|
encoder = extension.DecorateEncoder(typ, encoder)
|
|
}
|
|
encoder = ctx.encoderExtension.DecorateEncoder(typ, encoder)
|
|
for _, extension := range ctx.extraExtensions {
|
|
encoder = extension.DecorateEncoder(typ, encoder)
|
|
}
|
|
}
|
|
return encoder
|
|
}
|
|
|
|
func _getTypeEncoderFromExtension(ctx *ctx, typ reflect2.Type) ValEncoder {
|
|
for _, extension := range extensions {
|
|
encoder := extension.CreateEncoder(typ)
|
|
if encoder != nil {
|
|
return encoder
|
|
}
|
|
}
|
|
encoder := ctx.encoderExtension.CreateEncoder(typ)
|
|
if encoder != nil {
|
|
return encoder
|
|
}
|
|
for _, extension := range ctx.extraExtensions {
|
|
encoder := extension.CreateEncoder(typ)
|
|
if encoder != nil {
|
|
return encoder
|
|
}
|
|
}
|
|
typeName := typ.String()
|
|
encoder = typeEncoders[typeName]
|
|
if encoder != nil {
|
|
return encoder
|
|
}
|
|
if typ.Kind() == reflect.Ptr {
|
|
typePtr := typ.(*reflect2.UnsafePtrType)
|
|
encoder := typeEncoders[typePtr.Elem().String()]
|
|
if encoder != nil {
|
|
return &OptionalEncoder{encoder}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func describeStruct(ctx *ctx, typ reflect2.Type) *StructDescriptor {
|
|
structType := typ.(*reflect2.UnsafeStructType)
|
|
embeddedBindings := []*Binding{}
|
|
bindings := []*Binding{}
|
|
for i := 0; i < structType.NumField(); i++ {
|
|
field := structType.Field(i)
|
|
tag, hastag := field.Tag().Lookup(ctx.getTagKey())
|
|
if ctx.onlyTaggedField && !hastag && !field.Anonymous() {
|
|
continue
|
|
}
|
|
if tag == "-" || field.Name() == "_" {
|
|
continue
|
|
}
|
|
tagParts := strings.Split(tag, ",")
|
|
if field.Anonymous() && (tag == "" || tagParts[0] == "") {
|
|
if field.Type().Kind() == reflect.Struct {
|
|
structDescriptor := describeStruct(ctx, field.Type())
|
|
for _, binding := range structDescriptor.Fields {
|
|
binding.levels = append([]int{i}, binding.levels...)
|
|
omitempty := binding.Encoder.(*structFieldEncoder).omitempty
|
|
binding.Encoder = &structFieldEncoder{field, binding.Encoder, omitempty}
|
|
binding.Decoder = &structFieldDecoder{field, binding.Decoder}
|
|
embeddedBindings = append(embeddedBindings, binding)
|
|
}
|
|
continue
|
|
} else if field.Type().Kind() == reflect.Ptr {
|
|
ptrType := field.Type().(*reflect2.UnsafePtrType)
|
|
if ptrType.Elem().Kind() == reflect.Struct {
|
|
structDescriptor := describeStruct(ctx, ptrType.Elem())
|
|
for _, binding := range structDescriptor.Fields {
|
|
binding.levels = append([]int{i}, binding.levels...)
|
|
omitempty := binding.Encoder.(*structFieldEncoder).omitempty
|
|
binding.Encoder = &dereferenceEncoder{binding.Encoder}
|
|
binding.Encoder = &structFieldEncoder{field, binding.Encoder, omitempty}
|
|
binding.Decoder = &dereferenceDecoder{ptrType.Elem(), binding.Decoder}
|
|
binding.Decoder = &structFieldDecoder{field, binding.Decoder}
|
|
embeddedBindings = append(embeddedBindings, binding)
|
|
}
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
fieldNames := calcFieldNames(field.Name(), tagParts[0], tag)
|
|
fieldCacheKey := fmt.Sprintf("%s/%s", typ.String(), field.Name())
|
|
decoder := fieldDecoders[fieldCacheKey]
|
|
if decoder == nil {
|
|
decoder = decoderOfType(ctx.append(field.Name()), field.Type())
|
|
}
|
|
encoder := fieldEncoders[fieldCacheKey]
|
|
if encoder == nil {
|
|
encoder = encoderOfType(ctx.append(field.Name()), field.Type())
|
|
}
|
|
binding := &Binding{
|
|
Field: field,
|
|
FromNames: fieldNames,
|
|
ToNames: fieldNames,
|
|
Decoder: decoder,
|
|
Encoder: encoder,
|
|
}
|
|
binding.levels = []int{i}
|
|
bindings = append(bindings, binding)
|
|
}
|
|
return createStructDescriptor(ctx, typ, bindings, embeddedBindings)
|
|
}
|
|
func createStructDescriptor(ctx *ctx, typ reflect2.Type, bindings []*Binding, embeddedBindings []*Binding) *StructDescriptor {
|
|
structDescriptor := &StructDescriptor{
|
|
Type: typ,
|
|
Fields: bindings,
|
|
}
|
|
for _, extension := range extensions {
|
|
extension.UpdateStructDescriptor(structDescriptor)
|
|
}
|
|
ctx.encoderExtension.UpdateStructDescriptor(structDescriptor)
|
|
ctx.decoderExtension.UpdateStructDescriptor(structDescriptor)
|
|
for _, extension := range ctx.extraExtensions {
|
|
extension.UpdateStructDescriptor(structDescriptor)
|
|
}
|
|
processTags(structDescriptor, ctx.frozenConfig)
|
|
// merge normal & embedded bindings & sort with original order
|
|
allBindings := sortableBindings(append(embeddedBindings, structDescriptor.Fields...))
|
|
sort.Sort(allBindings)
|
|
structDescriptor.Fields = allBindings
|
|
return structDescriptor
|
|
}
|
|
|
|
type sortableBindings []*Binding
|
|
|
|
func (bindings sortableBindings) Len() int {
|
|
return len(bindings)
|
|
}
|
|
|
|
func (bindings sortableBindings) Less(i, j int) bool {
|
|
left := bindings[i].levels
|
|
right := bindings[j].levels
|
|
k := 0
|
|
for {
|
|
if left[k] < right[k] {
|
|
return true
|
|
} else if left[k] > right[k] {
|
|
return false
|
|
}
|
|
k++
|
|
}
|
|
}
|
|
|
|
func (bindings sortableBindings) Swap(i, j int) {
|
|
bindings[i], bindings[j] = bindings[j], bindings[i]
|
|
}
|
|
|
|
func processTags(structDescriptor *StructDescriptor, cfg *frozenConfig) {
|
|
for _, binding := range structDescriptor.Fields {
|
|
shouldOmitEmpty := false
|
|
tagParts := strings.Split(binding.Field.Tag().Get(cfg.getTagKey()), ",")
|
|
for _, tagPart := range tagParts[1:] {
|
|
if tagPart == "omitempty" {
|
|
shouldOmitEmpty = true
|
|
} else if tagPart == "string" {
|
|
if binding.Field.Type().Kind() == reflect.String {
|
|
binding.Decoder = &stringModeStringDecoder{binding.Decoder, cfg}
|
|
binding.Encoder = &stringModeStringEncoder{binding.Encoder, cfg}
|
|
} else {
|
|
binding.Decoder = &stringModeNumberDecoder{binding.Decoder}
|
|
binding.Encoder = &stringModeNumberEncoder{binding.Encoder}
|
|
}
|
|
}
|
|
}
|
|
binding.Decoder = &structFieldDecoder{binding.Field, binding.Decoder}
|
|
binding.Encoder = &structFieldEncoder{binding.Field, binding.Encoder, shouldOmitEmpty}
|
|
}
|
|
}
|
|
|
|
func calcFieldNames(originalFieldName string, tagProvidedFieldName string, wholeTag string) []string {
|
|
// ignore?
|
|
if wholeTag == "-" {
|
|
return []string{}
|
|
}
|
|
// rename?
|
|
var fieldNames []string
|
|
if tagProvidedFieldName == "" {
|
|
fieldNames = []string{originalFieldName}
|
|
} else {
|
|
fieldNames = []string{tagProvidedFieldName}
|
|
}
|
|
// private?
|
|
isNotExported := unicode.IsLower(rune(originalFieldName[0])) || originalFieldName[0] == '_'
|
|
if isNotExported {
|
|
fieldNames = []string{}
|
|
}
|
|
return fieldNames
|
|
}
|