Skip to content

Commit

Permalink
feat: support custom logical types (#429)
Browse files Browse the repository at this point in the history
  • Loading branch information
Emptyless authored Aug 13, 2024
1 parent 2623a40 commit 51c1d35
Show file tree
Hide file tree
Showing 4 changed files with 199 additions and 18 deletions.
2 changes: 1 addition & 1 deletion cmd/avrosv/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ func realMain(args []string, stdout, stderr io.Writer) int {
}

if cfg.Verbose {
fmt.Fprintln(stdout, schema)
_, _ = fmt.Fprintln(stdout, schema)
}

return 0
Expand Down
69 changes: 55 additions & 14 deletions gen/gen.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,12 +19,13 @@ import (

// Config configures the code generation.
type Config struct {
PackageName string
Tags map[string]TagStyle
FullName bool
Encoders bool
StrictTypes bool
Initialisms []string
PackageName string
Tags map[string]TagStyle
FullName bool
Encoders bool
StrictTypes bool
Initialisms []string
LogicalTypes []LogicalType
}

// TagStyle defines the styling for a tag.
Expand Down Expand Up @@ -83,6 +84,9 @@ func StructFromSchema(schema avro.Schema, w io.Writer, cfg Config) error {
WithInitialisms(cfg.Initialisms),
WithStrictTypes(cfg.StrictTypes),
}
for _, opt := range cfg.LogicalTypes {
opts = append(opts, WithLogicalType(opt))
}
g := NewGenerator(strcase.ToSnake(cfg.PackageName), cfg.Tags, opts...)
g.Parse(rec)

Expand Down Expand Up @@ -155,6 +159,29 @@ func WithPackageDoc(text string) OptsFunc {
}
}

// LogicalType used when the name of the "LogicalType" field in the Avro schema matches the Name attribute.
type LogicalType struct {
// Name of the LogicalType
Name string
// Typ returned, has to be a valid Go type
Typ string
// Import added as import (if not empty)
Import string
// ThirdPartyImport added as import (if not empty)
ThirdPartyImport string
}

// WithLogicalType registers a LogicalType which takes precedence over the default logical types
// defined by this package.
func WithLogicalType(logicalType LogicalType) OptsFunc {
return func(g *Generator) {
if g.logicalTypes == nil {
g.logicalTypes = map[avro.LogicalType]LogicalType{}
}
g.logicalTypes[avro.LogicalType(logicalType.Name)] = logicalType
}
}

func ensureTrailingPeriod(text string) string {
if text == "" {
return text
Expand All @@ -167,14 +194,15 @@ func ensureTrailingPeriod(text string) string {

// Generator generates Go structs from schemas.
type Generator struct {
template string
pkg string
pkgdoc string
tags map[string]TagStyle
fullName bool
encoders bool
strictTypes bool
initialisms []string
template string
pkg string
pkgdoc string
tags map[string]TagStyle
fullName bool
encoders bool
strictTypes bool
initialisms []string
logicalTypes map[avro.LogicalType]LogicalType

imports []string
thirdPartyImports []string
Expand Down Expand Up @@ -313,6 +341,19 @@ func (g *Generator) resolveUnionTypes(s *avro.UnionSchema) string {
}

func (g *Generator) resolveLogicalSchema(logicalType avro.LogicalType) string {
if g.logicalTypes != nil {
if typ, ok := g.logicalTypes[logicalType]; ok {
if val := typ.Import; val != "" {
g.addImport(val)
}
if val := typ.ThirdPartyImport; val != "" {
g.addThirdPartyImport(val)
}

return typ.Typ
}
}

var typ string
switch logicalType {
case "date", "timestamp-millis", "timestamp-micros":
Expand Down
61 changes: 58 additions & 3 deletions gen/gen_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -155,19 +155,74 @@ func TestStruct_ConfigurableFieldTags(t *testing.T) {
}
}

func TestStruct_ConfigurableLogicalTypes(t *testing.T) {
schema := `{
"type": "record",
"name": "test",
"fields": [
{ "name": "id", "type": {"type": "string", "logicalType": "uuid"} }
]
}`

gc := gen.Config{
PackageName: "Something",
LogicalTypes: []gen.LogicalType{{
Name: "uuid",
Typ: "uuid.UUID",
ThirdPartyImport: "github.com/google/uuid",
}},
}
_, lines := generate(t, schema, gc)

for _, expected := range []string{
"package something",
"import (",
"\"github.com/google/uuid\"",
"type Test struct {",
"ID uuid.UUID `avro:\"id\"`",
"}",
} {
assert.Contains(t, lines, expected)
}
}

func TestStruct_GenFromRecordSchema(t *testing.T) {
fileName := "testdata/golden.go"
gc := gen.Config{PackageName: "Something"}
schema, err := os.ReadFile("testdata/golden.avsc")
require.NoError(t, err)

file, _ := generate(t, string(schema), gc)

if *update {
err = os.WriteFile(fileName, file, 0600)
require.NoError(t, err)
}

want, err := os.ReadFile(fileName)
require.NoError(t, err)
assert.Equal(t, string(want), string(file))
}

func TestStruct_GenFromRecordSchemaWithCustomLogicalTypes(t *testing.T) {
fileName := "testdata/golden_logicaltype.go"

gc := gen.Config{PackageName: "Something", LogicalTypes: []gen.LogicalType{{
Name: "uuid",
Typ: "uuid.UUID",
ThirdPartyImport: "github.com/google/uuid",
}}}
schema, err := os.ReadFile("testdata/golden.avsc")
require.NoError(t, err)

gc := gen.Config{PackageName: "Something"}
file, _ := generate(t, string(schema), gc)

if *update {
err = os.WriteFile("testdata/golden.go", file, 0600)
err = os.WriteFile(fileName, file, 0600)
require.NoError(t, err)
}

want, err := os.ReadFile("testdata/golden.go")
want, err := os.ReadFile(fileName)
require.NoError(t, err)
assert.Equal(t, string(want), string(file))
}
Expand Down
85 changes: 85 additions & 0 deletions gen/testdata/golden_logicaltype.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit 51c1d35

Please sign in to comment.