Skip to content

Commit

Permalink
feat: generate kotlin external modules from FTL tooling
Browse files Browse the repository at this point in the history
fixes #970
  • Loading branch information
worstell committed Feb 26, 2024
1 parent 4155488 commit 4ac6d72
Show file tree
Hide file tree
Showing 51 changed files with 2,239 additions and 11 deletions.
4 changes: 4 additions & 0 deletions Bitfile
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,10 @@ kotlin-runtime/scaffolding.zip: kotlin-runtime/scaffolding/**/*
cd kotlin-runtime/scaffolding
build: zip -q --symlinks -r ../scaffolding.zip .

kotlin-runtime/external-module-template.zip: kotlin-runtime/external-module-template/**/*
cd kotlin-runtime/external-module-template
build: zip -q --symlinks -r ../external-module-template.zip .

%{SCHEMA_OUT}: %{SCHEMA_IN}
build:
ftl-schema > %{OUT}
Expand Down
225 changes: 221 additions & 4 deletions buildengine/build_kotlin.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,26 +3,75 @@ package buildengine
import (
"context"
"fmt"
sets "github.com/deckarep/golang-set/v2"
"os"
"path/filepath"
"reflect"
"sort"
"strings"

"github.com/beevik/etree"

"connectrpc.com/connect"
"github.com/TBD54566975/ftl"
ftlv1 "github.com/TBD54566975/ftl/backend/protos/xyz/block/ftl/v1"
"github.com/TBD54566975/ftl/backend/protos/xyz/block/ftl/v1/ftlv1connect"
"github.com/TBD54566975/ftl/backend/schema"
"github.com/TBD54566975/ftl/backend/schema/strcase"
"github.com/TBD54566975/ftl/internal"
"github.com/TBD54566975/ftl/internal/exec"
"github.com/TBD54566975/ftl/internal/log"
"github.com/TBD54566975/ftl/internal/rpc"
kotlinruntime "github.com/TBD54566975/ftl/kotlin-runtime"
"github.com/TBD54566975/scaffolder"
"github.com/beevik/etree"
"golang.org/x/exp/maps"
)

func buildKotlin(ctx context.Context, _ *schema.Schema, module Module) error {
type externalModuleContext struct {
module Module
*schema.Schema
}

func (e externalModuleContext) ExternalModules() []*schema.Module {
depsSet := make(map[string]struct{})
for _, dep := range e.module.Dependencies {
depsSet[dep] = struct{}{}
}

modules := make([]*schema.Module, 0)
for _, module := range e.Modules {
if _, exists := depsSet[module.Name]; exists || module.Name == "builtin" {
modules = append(modules, module)
}
}
return modules
}

func buildKotlin(ctx context.Context, module Module) error {
logger := log.FromContext(ctx)
client := rpc.ClientFromContext[ftlv1connect.ControllerServiceClient](ctx)
resp, err := client.GetSchema(ctx, connect.NewRequest(&ftlv1.GetSchemaRequest{}))
if err != nil {
return err
}
sch, err := schema.FromProto(resp.Msg.Schema)
if err != nil {
return fmt.Errorf("failed to convert schema from proto: %w", err)
}

if err := SetPOMProperties(ctx, filepath.Join(module.Dir, "..")); err != nil {
return fmt.Errorf("unable to update ftl.version in %s: %w", module.Dir, err)
}

if err = generateExternalModules(ctx, module, sch); err != nil {
return fmt.Errorf("unable to generate external modules for %s: %w", module.Module, err)
}

if err = configureDetekt(module); err != nil {
return fmt.Errorf("unable to configure detekt for %s: %w", module.Module, err)
}

logger.Debugf("Using build command '%s'", module.Build)
err := exec.Command(ctx, log.Debug, module.Dir, "bash", "-c", module.Build).RunBuffered(ctx)
err = exec.Command(ctx, log.Debug, module.Dir, "bash", "-c", module.Build).RunBuffered(ctx)
if err != nil {
return fmt.Errorf("failed to build module: %w", err)
}
Expand Down Expand Up @@ -72,3 +121,171 @@ func SetPOMProperties(ctx context.Context, baseDir string) error {

return tree.WriteToFile(pomFile)
}

func configureDetekt(module Module) error {
buildDir := filepath.Join(module.Dir, "target")
if err := os.MkdirAll(buildDir, 0700); err != nil {
return err
}

fileContent := fmt.Sprintf(`
SchemaExtractorRuleSet:
ExtractSchemaRule:
active: true
output: %s
`, buildDir)

detektYmlPath := filepath.Join(buildDir, "detekt.yml")
return os.WriteFile(detektYmlPath, []byte(fileContent), 0600)
}

func generateExternalModules(ctx context.Context, module Module, sch *schema.Schema) error {
logger := log.FromContext(ctx)
config := module.ModuleConfig
funcs := maps.Clone(scaffoldFuncs)

// Wipe the modules directory to ensure we don't have any stale modules.
_ = os.RemoveAll(filepath.Join(config.Dir, "target", "generated-sources", "ftl"))

logger.Debugf("Generating external modules")
if err := internal.ScaffoldZip(kotlinruntime.ExternalModuleTemplates(), config.Dir, externalModuleContext{
module: module,
Schema: sch,
}, scaffolder.Functions(funcs)); err != nil {
return err
}
return nil
}

var scaffoldFuncs = scaffolder.FuncMap{
"snake": strcase.ToLowerSnake,
"screamingSnake": strcase.ToUpperSnake,
"camel": strcase.ToUpperCamel,
"lowerCamel": strcase.ToLowerCamel,
"kebab": strcase.ToLowerKebab,
"screamingKebab": strcase.ToUpperKebab,
"upper": strings.ToUpper,
"lower": strings.ToLower,
"title": strings.Title,
"typename": schema.TypeName,
"comment": func(s []string) string {
if len(s) == 0 {
return ""
}
var sb strings.Builder
sb.WriteString("/**\n")
for _, line := range s {
sb.WriteString(" * ")
sb.WriteString(line)
sb.WriteString("\n")
}
sb.WriteString(" */\n")
return sb.String()
},
"type": genType,
"is": func(kind string, t schema.Node) bool {
return reflect.Indirect(reflect.ValueOf(t)).Type().Name() == kind
},
"imports": func(m *schema.Module) []string {
imports := sets.NewSet[string]()
_ = schema.Visit(m, func(n schema.Node, next func() error) error {
switch n := n.(type) {
case *schema.DataRef:
decl := m.Resolve(schema.Ref{
Module: n.Module,
Name: n.Name,
})
if decl != nil {
if data, ok := decl.Decl.(*schema.Data); ok {
if len(data.Fields) == 0 {
imports.Add("ftl.builtin.Empty")
break
}
}
}

if n.Module == "" {
break
}

imports.Add("ftl." + n.Module + "." + n.Name)

for _, tp := range n.TypeParameters {
tpRef, err := schema.ParseDataRef(tp.String())
if err != nil {
return err
}
if tpRef.Module != "" && tpRef.Module != m.Name {
imports.Add("ftl." + tpRef.Module + "." + tpRef.Name)
}
}
case *schema.Verb:
imports.Append("xyz.block.ftl.Context", "xyz.block.ftl.Ignore", "xyz.block.ftl.Verb")

case *schema.Time:
imports.Add("java.time.OffsetDateTime")

default:
}
return next()
})
importsList := imports.ToSlice()
sort.Strings(importsList)
return importsList
},
}

func genType(module *schema.Module, t schema.Type) string {
switch t := t.(type) {
case *schema.DataRef:
decl := module.Resolve(schema.Ref{
Module: t.Module,
Name: t.Name,
})
if decl != nil {
if data, ok := decl.Decl.(*schema.Data); ok {
if len(data.Fields) == 0 {
return "Empty"
}
}
}

desc := t.Name
if len(t.TypeParameters) > 0 {
desc += "<"
for i, tp := range t.TypeParameters {
if i != 0 {
desc += ", "
}
desc += genType(module, tp)
}
desc += ">"
}
return desc

case *schema.Time:
return "OffsetDateTime"

case *schema.Array:
return "List<" + genType(module, t.Element) + ">"

case *schema.Map:
return "Map<" + genType(module, t.Key) + ", " + genType(module, t.Value) + ">"

case *schema.Optional:
return genType(module, t.Type) + "? = null"

case *schema.Bytes:
return "ByteArray"

case *schema.Bool:
return "Boolean"

case *schema.Int:
return "Long"

case *schema.Float, *schema.String, *schema.Any, *schema.Unit:
return t.String()
}
panic(fmt.Sprintf("unsupported type %T", t))
}
Loading

0 comments on commit 4ac6d72

Please sign in to comment.