refactor: rewrite CLI and internal architecture #41
@@ -1,14 +1,11 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"git.maximhutz.com/max/lambda/internal/cli"
|
"git.maximhutz.com/max/lambda/internal/cli"
|
||||||
"git.maximhutz.com/max/lambda/internal/config"
|
"git.maximhutz.com/max/lambda/internal/config"
|
||||||
"git.maximhutz.com/max/lambda/internal/plugins"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/convert"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/normalorder"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
@@ -20,48 +17,42 @@ func main() {
|
|||||||
logger.Info("using program arguments", "args", os.Args)
|
logger.Info("using program arguments", "args", os.Args)
|
||||||
logger.Info("parsed CLI options", "options", options)
|
logger.Info("parsed CLI options", "options", options)
|
||||||
|
|
||||||
|
r := GetRegistry()
|
||||||
|
fmt.Println(1)
|
||||||
// Get input.
|
// Get input.
|
||||||
input, err := options.Source.Extract()
|
input, err := options.Source.Extract()
|
||||||
cli.HandleError(err)
|
cli.HandleError(err)
|
||||||
|
fmt.Println(2)
|
||||||
|
|
||||||
// Parse code into syntax tree.
|
// Parse code into syntax tree.
|
||||||
ast, err := saccharine.Parse(input)
|
repr, err := r.Unmarshal(input, "saccharine")
|
||||||
cli.HandleError(err)
|
cli.HandleError(err)
|
||||||
logger.Info("parsed syntax tree", "tree", ast)
|
logger.Info("parsed syntax tree", "tree", repr)
|
||||||
|
|
||||||
|
fmt.Println(3)
|
||||||
// Compile expression to lambda calculus.
|
// Compile expression to lambda calculus.
|
||||||
compiled := convert.SaccharineToLambda(ast)
|
compiled, err := r.ConvertTo(repr, "lambda")
|
||||||
logger.Info("compiled λ expression", "tree", compiled.String())
|
cli.HandleError(err)
|
||||||
|
logger.Info("compiled λ expression", "tree", compiled)
|
||||||
|
fmt.Println(4)
|
||||||
// Create reducer with the compiled expression.
|
// Create reducer with the compiled expression.
|
||||||
runtime := normalorder.NewRuntime(compiled)
|
engine, err := r.GetDefaultEngine("lambda")
|
||||||
|
cli.HandleError(err)
|
||||||
// If the user selected to track CPU performance, attach a profiler.
|
fmt.Println(5)
|
||||||
if options.Profile != "" {
|
|
||||||
plugins.NewPerformance(options.Profile, runtime)
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the user selected to produce a step-by-step explanation, attach an
|
|
||||||
// observer.
|
|
||||||
if options.Explanation {
|
|
||||||
plugins.NewExplanation(runtime)
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the user opted to track statistics, attach a tracker.
|
|
||||||
if options.Statistics {
|
|
||||||
plugins.NewStatistics(runtime)
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the user selected for verbose debug logs, attach a reduction tracker.
|
|
||||||
if options.Verbose {
|
|
||||||
plugins.NewLogs(logger, runtime)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
err = engine.Set(compiled)
|
||||||
|
cli.HandleError(err)
|
||||||
// Run reduction.
|
// Run reduction.
|
||||||
runtime.Run()
|
for engine.Step(1) {
|
||||||
|
}
|
||||||
|
fmt.Println(6)
|
||||||
// Return the final reduced result.
|
// Return the final reduced result.
|
||||||
result := runtime.Expression().String()
|
result, err := engine.Get()
|
||||||
err = options.Destination.Write(result)
|
cli.HandleError(err)
|
||||||
|
fmt.Println(7, result)
|
||||||
|
output, err := r.Marshal(result)
|
||||||
|
cli.HandleError(err)
|
||||||
|
fmt.Println(8)
|
||||||
|
err = options.Destination.Write(output)
|
||||||
cli.HandleError(err)
|
cli.HandleError(err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,85 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/convert"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/normalorder"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Helper function to run a single sample through the lambda runtime.
|
|
||||||
func runSample(samplePath string) (string, error) {
|
|
||||||
// Read the sample file.
|
|
||||||
input, err := os.ReadFile(samplePath)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse code into syntax tree.
|
|
||||||
ast, err := saccharine.Parse(string(input))
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compile expression to lambda calculus.
|
|
||||||
compiled := convert.SaccharineToLambda(ast)
|
|
||||||
|
|
||||||
// Create and run the reducer.
|
|
||||||
reducer := normalorder.NewRuntime(compiled)
|
|
||||||
reducer.Run()
|
|
||||||
|
|
||||||
return reducer.Expression().String() + "\n", nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Test that all samples produce expected output.
|
|
||||||
func TestSamplesValidity(t *testing.T) {
|
|
||||||
// Discover all .test files in the tests directory.
|
|
||||||
testFiles, err := filepath.Glob("../../tests/*.test")
|
|
||||||
assert.NoError(t, err, "Failed to read tests directory.")
|
|
||||||
assert.NotEmpty(t, testFiles, "No '*.test' files found in directory.")
|
|
||||||
|
|
||||||
for _, testPath := range testFiles {
|
|
||||||
// Build expected file path.
|
|
||||||
expectedPath := strings.TrimSuffix(testPath, filepath.Ext(testPath)) + ".expected"
|
|
||||||
|
|
||||||
name := strings.TrimSuffix(filepath.Base(testPath), filepath.Ext(testPath))
|
|
||||||
|
|
||||||
t.Run(name, func(t *testing.T) {
|
|
||||||
// Run the sample and capture output.
|
|
||||||
actual, err := runSample(testPath)
|
|
||||||
assert.NoError(t, err, "Failed to run sample.")
|
|
||||||
|
|
||||||
// Read expected output.
|
|
||||||
expectedBytes, err := os.ReadFile(expectedPath)
|
|
||||||
assert.NoError(t, err, "Failed to read expected output.")
|
|
||||||
expected := string(expectedBytes)
|
|
||||||
|
|
||||||
// Compare outputs.
|
|
||||||
assert.Equal(t, expected, actual, "Output does not match expected.")
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Benchmark all samples using sub-benchmarks.
|
|
||||||
func BenchmarkSamples(b *testing.B) {
|
|
||||||
// Discover all .test files in the tests directory.
|
|
||||||
testFiles, err := filepath.Glob("../../tests/*.test")
|
|
||||||
assert.NoError(b, err, "Failed to read tests directory.")
|
|
||||||
assert.NotEmpty(b, testFiles, "No '*.test' files found in directory.")
|
|
||||||
|
|
||||||
for _, path := range testFiles {
|
|
||||||
name := strings.TrimSuffix(filepath.Base(path), filepath.Ext(path))
|
|
||||||
|
|
||||||
b.Run(name, func(b *testing.B) {
|
|
||||||
for b.Loop() {
|
|
||||||
_, err := runSample(path)
|
|
||||||
assert.NoError(b, err, "Failed to run sample.")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -9,14 +9,14 @@ import (
|
|||||||
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
||||||
)
|
)
|
||||||
|
|
||||||
func MakeRegistry() *registry.Registry {
|
func GetRegistry() *registry.Registry {
|
||||||
r := registry.New()
|
r := registry.New()
|
||||||
|
|
||||||
// Codecs
|
// Codecs
|
||||||
r.AddCodec(cli.ConvertCodec(convert.Saccharine2Lambda{}, "saccharine", "lambda"))
|
r.AddConversions(cli.ConvertCodec(convert.Saccharine2Lambda{}, "saccharine", "lambda")...)
|
||||||
|
|
||||||
// Engines
|
// Engines
|
||||||
r.AddEngine(cli.ConvertEngine(normalorder.Engine{}, "normalorder", "lambda"))
|
r.AddEngine(cli.ConvertEngine(&normalorder.Engine{}, "normalorder", "lambda"))
|
||||||
|
|
||||||
// Marshalers
|
// Marshalers
|
||||||
r.AddMarshaler(cli.ConvertMarshaler(lambda.Marshaler{}, "lambda"))
|
r.AddMarshaler(cli.ConvertMarshaler(lambda.Marshaler{}, "lambda"))
|
||||||
|
|||||||
@@ -6,33 +6,19 @@ import (
|
|||||||
"git.maximhutz.com/max/lambda/pkg/codec"
|
"git.maximhutz.com/max/lambda/pkg/codec"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Codec interface {
|
type Conversion interface {
|
||||||
codec.Codec[Repr, Repr]
|
|
||||||
|
|
||||||
InType() string
|
InType() string
|
||||||
OutType() string
|
OutType() string
|
||||||
|
|
||||||
|
Run(Repr) (Repr, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type convertedCodec[T, U any] struct {
|
type forwardCodec[T, U any] struct {
|
||||||
codec codec.Codec[T, U]
|
codec codec.Codec[T, U]
|
||||||
inType, outType string
|
inType, outType string
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c convertedCodec[T, U]) Decode(r Repr) (Repr, error) {
|
func (c forwardCodec[T, U]) Run(r Repr) (Repr, error) {
|
||||||
u, ok := r.Data().(U)
|
|
||||||
if !ok {
|
|
||||||
return nil, fmt.Errorf("could not parse '%v' as '%s'", r, c.inType)
|
|
||||||
}
|
|
||||||
|
|
||||||
t, err := c.codec.Decode(u)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return NewRepr(c.outType, t), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c convertedCodec[T, U]) Encode(r Repr) (Repr, error) {
|
|
||||||
t, ok := r.Data().(T)
|
t, ok := r.Data().(T)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, fmt.Errorf("could not parse '%v' as '%s'", t, c.outType)
|
return nil, fmt.Errorf("could not parse '%v' as '%s'", t, c.outType)
|
||||||
@@ -46,10 +32,36 @@ func (c convertedCodec[T, U]) Encode(r Repr) (Repr, error) {
|
|||||||
return NewRepr(c.inType, u), nil
|
return NewRepr(c.inType, u), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c convertedCodec[T, U]) InType() string { return c.inType }
|
func (c forwardCodec[T, U]) InType() string { return c.inType }
|
||||||
|
|
||||||
func (c convertedCodec[T, U]) OutType() string { return c.outType }
|
func (c forwardCodec[T, U]) OutType() string { return c.outType }
|
||||||
|
|
||||||
func ConvertCodec[T, U any](e codec.Codec[T, U], inType, outType string) Codec {
|
type backwardCodec[T, U any] struct {
|
||||||
return convertedCodec[T, U]{e, inType, outType}
|
codec codec.Codec[T, U]
|
||||||
|
inType, outType string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c backwardCodec[T, U]) Run(r Repr) (Repr, error) {
|
||||||
|
u, ok := r.Data().(U)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("could not parse '%v' as '%s'", r, c.inType)
|
||||||
|
}
|
||||||
|
|
||||||
|
t, err := c.codec.Decode(u)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewRepr(c.outType, t), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c backwardCodec[T, U]) InType() string { return c.outType }
|
||||||
|
|
||||||
|
func (c backwardCodec[T, U]) OutType() string { return c.inType }
|
||||||
|
|
||||||
|
func ConvertCodec[T, U any](e codec.Codec[T, U], inType, outType string) []Conversion {
|
||||||
|
return []Conversion{
|
||||||
|
forwardCodec[T, U]{e, inType, outType},
|
||||||
|
backwardCodec[T, U]{e, inType, outType},
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
27
internal/registry/converter.go
Normal file
27
internal/registry/converter.go
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
package registry
|
||||||
|
|
||||||
|
import (
|
||||||
|
"git.maximhutz.com/max/lambda/internal/cli"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Converter struct {
|
||||||
|
data map[string][]cli.Conversion
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewConverter() *Converter {
|
||||||
|
return &Converter{data: map[string][]cli.Conversion{}}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Converter) Add(c cli.Conversion) {
|
||||||
|
conversionsFromIn, ok := g.data[c.InType()]
|
||||||
|
if !ok {
|
||||||
|
conversionsFromIn = []cli.Conversion{}
|
||||||
|
}
|
||||||
|
|
||||||
|
conversionsFromIn = append(conversionsFromIn, c)
|
||||||
|
g.data[c.InType()] = conversionsFromIn
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Converter) ConversionsFrom(t string) []cli.Conversion {
|
||||||
|
return g.data[t]
|
||||||
|
}
|
||||||
@@ -8,20 +8,23 @@ import (
|
|||||||
|
|
||||||
type Registry struct {
|
type Registry struct {
|
||||||
marshalers map[string]cli.Marshaler
|
marshalers map[string]cli.Marshaler
|
||||||
codecs []cli.Codec
|
converter *Converter
|
||||||
engines map[string]cli.Engine
|
engines map[string]cli.Engine
|
||||||
}
|
}
|
||||||
|
|
||||||
func New() *Registry {
|
func New() *Registry {
|
||||||
return &Registry{
|
return &Registry{
|
||||||
marshalers: map[string]cli.Marshaler{},
|
marshalers: map[string]cli.Marshaler{},
|
||||||
codecs: []cli.Codec{},
|
converter: NewConverter(),
|
||||||
engines: map[string]cli.Engine{},
|
engines: map[string]cli.Engine{},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Registry) AddCodec(c cli.Codec) error {
|
func (r *Registry) AddConversions(conversions ...cli.Conversion) error {
|
||||||
r.codecs = append(r.codecs, c)
|
for _, conversion := range conversions {
|
||||||
|
r.converter.Add(conversion)
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -52,8 +55,31 @@ func (r *Registry) GetEngine(name string) (cli.Engine, error) {
|
|||||||
return e, nil
|
return e, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *Registry) GetDefaultEngine(id string) (cli.Engine, error) {
|
||||||
|
for _, engine := range r.engines {
|
||||||
|
if engine.InType() == id {
|
||||||
|
return engine, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf("no engine for '%s'", id)
|
||||||
|
}
|
||||||
|
|
||||||
func (r *Registry) ConvertTo(repr cli.Repr, outType string) (cli.Repr, error) {
|
func (r *Registry) ConvertTo(repr cli.Repr, outType string) (cli.Repr, error) {
|
||||||
panic("")
|
path, err := r.ConversionPath(repr.Id(), outType)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
result := repr
|
||||||
|
for _, conversion := range path {
|
||||||
|
result, err = conversion.Run(result)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("converting '%s' to '%s': %w", conversion.InType(), conversion.OutType(), err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Registry) Marshal(repr cli.Repr) (string, error) {
|
func (r *Registry) Marshal(repr cli.Repr) (string, error) {
|
||||||
@@ -73,3 +99,52 @@ func (r *Registry) Unmarshal(s string, outType string) (cli.Repr, error) {
|
|||||||
|
|
||||||
return m.Decode(s)
|
return m.Decode(s)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func reverse[T any](list []T) []T {
|
||||||
|
if list == nil {
|
||||||
|
return list
|
||||||
|
}
|
||||||
|
|
||||||
|
reversed := []T{}
|
||||||
|
|
||||||
|
for i := len(list) - 1; i >= 0; i-- {
|
||||||
|
reversed = append(reversed, list[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
return reversed
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Registry) ConversionPath(from, to string) ([]cli.Conversion, error) {
|
||||||
|
backtrack := map[string]cli.Conversion{}
|
||||||
|
iteration := []string{from}
|
||||||
|
for len(iteration) > 0 {
|
||||||
|
nextIteration := []string{}
|
||||||
|
|
||||||
|
for _, item := range iteration {
|
||||||
|
for _, conversion := range r.converter.ConversionsFrom(item) {
|
||||||
|
if _, ok := backtrack[conversion.OutType()]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
nextIteration = append(nextIteration, conversion.OutType())
|
||||||
|
backtrack[conversion.OutType()] = conversion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iteration = nextIteration
|
||||||
|
}
|
||||||
|
|
||||||
|
reversedPath := []cli.Conversion{}
|
||||||
|
current := to
|
||||||
|
for current != from {
|
||||||
|
conversion, ok := backtrack[current]
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("no valid conversion from '%s' to '%s'", from, to)
|
||||||
|
}
|
||||||
|
|
||||||
|
reversedPath = append(reversedPath, conversion)
|
||||||
|
current = conversion.InType()
|
||||||
|
}
|
||||||
|
|
||||||
|
return reverse(reversedPath), nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,19 +13,19 @@ func (e Engine) Get() (lambda.Expression, error) {
|
|||||||
return e.expr, nil
|
return e.expr, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e Engine) Set(l lambda.Expression) error {
|
func (e *Engine) Set(l lambda.Expression) error {
|
||||||
e.expr = l
|
e.expr = l
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e Engine) Step(i int) bool {
|
func (e *Engine) Step(i int) bool {
|
||||||
var reduced bool
|
|
||||||
|
|
||||||
for range i {
|
for range i {
|
||||||
e.expr, reduced = ReduceOnce(e.expr)
|
next, reduced := ReduceOnce(e.expr)
|
||||||
if !reduced {
|
if !reduced {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
e.expr = next
|
||||||
}
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
|
|||||||
Reference in New Issue
Block a user