From dfbeef73580d201a204de23eb6b79696f8b40eb1 Mon Sep 17 00:00:00 2001 From: 0xjac Date: Mon, 7 Oct 2024 12:55:32 +0200 Subject: [PATCH] feat: Output plain map/slice for nullable types Go maps and slices can be nil. An avro nullable union can thus be represented directly by a map or a slice without using an explicit pointer (`*`). This feature is hidden being the `PlainMap` and `PlainSlice` options to preserve the existing behavior. Closes #461 --- gen/gen.go | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/gen/gen.go b/gen/gen.go index 010fc85..b604a42 100644 --- a/gen/gen.go +++ b/gen/gen.go @@ -24,6 +24,8 @@ type Config struct { FullName bool Encoders bool FullSchema bool + PlainMap bool + PlainSlice bool StrictTypes bool Initialisms []string LogicalTypes []LogicalType @@ -43,6 +45,9 @@ const ( Kebab TagStyle = "kebab" // UpperCamel is a style like ImWrittenInUpperCamel. UpperCamel TagStyle = "upper-camel" + + prefixMap string = "map[" + prefixSlice string = "[]" ) //go:embed output_template.tmpl @@ -85,6 +90,8 @@ func StructFromSchema(schema avro.Schema, w io.Writer, cfg Config) error { WithInitialisms(cfg.Initialisms), WithStrictTypes(cfg.StrictTypes), WithFullSchema(cfg.FullSchema), + WithPlainMap(cfg.PlainMap), + WithPlainSlice(cfg.PlainSlice), } for _, opt := range cfg.LogicalTypes { opts = append(opts, WithLogicalType(opt)) @@ -168,6 +175,20 @@ func WithFullSchema(b bool) OptsFunc { } } +// WithPlainMap configures the generator to emit a plain map and not a map ptr for nullable unions. +func WithPlainMap(b bool) OptsFunc { + return func(g *Generator) { + g.plainMap = b + } +} + +// WithPlainSlice configures the generator to emit a plain map and not a map ptr for nullable unions. +func WithPlainSlice(b bool) OptsFunc { + return func(g *Generator) { + g.plainSlice = b + } +} + // LogicalType used when the name of the "LogicalType" field in the Avro schema matches the Name attribute. type LogicalType struct { // Name of the LogicalType @@ -210,6 +231,8 @@ type Generator struct { fullName bool encoders bool fullSchema bool + plainMap bool + plainSlice bool strictTypes bool initialisms []string logicalTypes map[avro.LogicalType]LogicalType @@ -356,6 +379,14 @@ func (g *Generator) resolveUnionTypes(s *avro.UnionSchema) string { types = append(types, g.generate(elem)) } if s.Nullable() { + if g.plainMap && strings.HasPrefix(types[0], prefixMap) { + return types[0] + } + + if g.plainSlice && strings.HasPrefix(types[0], prefixSlice) { + return types[0] + } + return "*" + types[0] } return "any"