feat: marshalers to codecs
This commit is contained in:
@@ -12,14 +12,15 @@ func GetRegistry() *registry.Registry {
|
|||||||
r := registry.New()
|
r := registry.New()
|
||||||
|
|
||||||
// Codecs
|
// Codecs
|
||||||
(registry.RegisterCodec(r, convert.Saccharine2Lambda{}, "saccharine", "lambda"))
|
(registry.RegisterConversion(r, convert.Saccharine2Lambda, "saccharine", "lambda"))
|
||||||
|
(registry.RegisterConversion(r, convert.Lambda2Saccharine, "lambda", "saccharine"))
|
||||||
|
|
||||||
// Engines
|
// Engines
|
||||||
(registry.RegisterEngine(r, normalorder.NewProcess, "normalorder", "lambda"))
|
(registry.RegisterEngine(r, normalorder.NewProcess, "normalorder", "lambda"))
|
||||||
|
|
||||||
// Marshalers
|
// Marshalers
|
||||||
(registry.RegisterMarshaler(r, lambda.Marshaler{}, "lambda"))
|
(registry.RegisterCodec(r, lambda.Marshaler{}, "lambda"))
|
||||||
(registry.RegisterMarshaler(r, saccharine.Marshaler{}, "saccharine"))
|
(registry.RegisterCodec(r, saccharine.Marshaler{}, "saccharine"))
|
||||||
|
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|||||||
50
internal/registry/codec.go
Normal file
50
internal/registry/codec.go
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
package registry
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"git.maximhutz.com/max/lambda/pkg/codec"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Codec interface {
|
||||||
|
codec.Codec[Repr]
|
||||||
|
|
||||||
|
InType() string
|
||||||
|
}
|
||||||
|
|
||||||
|
type convertedCodec[T any] struct {
|
||||||
|
codec codec.Codec[T]
|
||||||
|
inType string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c convertedCodec[T]) Decode(s string) (Repr, error) {
|
||||||
|
t, err := c.codec.Decode(s)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewRepr(c.inType, t), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c convertedCodec[T]) Encode(r Repr) (string, error) {
|
||||||
|
t, ok := r.Data().(T)
|
||||||
|
if !ok {
|
||||||
|
dataType := reflect.TypeOf(r.Data())
|
||||||
|
allowedType := reflect.TypeFor[T]()
|
||||||
|
return "", fmt.Errorf("Codec for '%s' cannot parse '%s'", allowedType, dataType)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.codec.Encode(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c convertedCodec[T]) InType() string { return c.inType }
|
||||||
|
|
||||||
|
func RegisterCodec[T any](registry *Registry, m codec.Codec[T], inType string) error {
|
||||||
|
if _, ok := registry.codecs[inType]; ok {
|
||||||
|
return fmt.Errorf("Codec for '%s' already registered", inType)
|
||||||
|
}
|
||||||
|
|
||||||
|
registry.codecs[inType] = convertedCodec[T]{m, inType}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -13,18 +13,18 @@ type Conversion interface {
|
|||||||
Run(Repr) (Repr, error)
|
Run(Repr) (Repr, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type forwardCodec[T, U any] struct {
|
type convertedConversion[T, U any] struct {
|
||||||
codec codec.Codec[T, U]
|
conversion codec.Conversion[T, U]
|
||||||
inType, outType string
|
inType, outType string
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c forwardCodec[T, U]) Run(r Repr) (Repr, error) {
|
func (c convertedConversion[T, U]) Run(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.inType)
|
return nil, fmt.Errorf("could not parse '%v' as '%s'", t, c.inType)
|
||||||
}
|
}
|
||||||
|
|
||||||
u, err := c.codec.Encode(t)
|
u, err := c.conversion(t)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -32,36 +32,12 @@ func (c forwardCodec[T, U]) Run(r Repr) (Repr, error) {
|
|||||||
return NewRepr(c.outType, u), nil
|
return NewRepr(c.outType, u), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c forwardCodec[T, U]) InType() string { return c.inType }
|
func (c convertedConversion[T, U]) InType() string { return c.inType }
|
||||||
|
|
||||||
func (c forwardCodec[T, U]) OutType() string { return c.outType }
|
func (c convertedConversion[T, U]) OutType() string { return c.outType }
|
||||||
|
|
||||||
type backwardCodec[T, U any] struct {
|
func RegisterConversion[T, U any](registry *Registry, conversion func(T) (U, error), inType, outType string) error {
|
||||||
codec codec.Codec[T, U]
|
registry.converter.Add(convertedConversion[T, U]{conversion, inType, outType})
|
||||||
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.outType)
|
|
||||||
}
|
|
||||||
|
|
||||||
t, err := c.codec.Decode(u)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return NewRepr(c.inType, t), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c backwardCodec[T, U]) InType() string { return c.outType }
|
|
||||||
|
|
||||||
func (c backwardCodec[T, U]) OutType() string { return c.inType }
|
|
||||||
|
|
||||||
func RegisterCodec[T, U any](registry *Registry, c codec.Codec[T, U], inType, outType string) error {
|
|
||||||
registry.converter.Add(forwardCodec[T, U]{c, inType, outType})
|
|
||||||
registry.converter.Add(backwardCodec[T, U]{c, inType, outType})
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,50 +0,0 @@
|
|||||||
package registry
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/codec"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Marshaler interface {
|
|
||||||
codec.Marshaler[Repr]
|
|
||||||
|
|
||||||
InType() string
|
|
||||||
}
|
|
||||||
|
|
||||||
type convertedMarshaler[T any] struct {
|
|
||||||
codec codec.Marshaler[T]
|
|
||||||
inType string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c convertedMarshaler[T]) Decode(s string) (Repr, error) {
|
|
||||||
t, err := c.codec.Decode(s)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return NewRepr(c.inType, t), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c convertedMarshaler[T]) Encode(r Repr) (string, error) {
|
|
||||||
t, ok := r.Data().(T)
|
|
||||||
if !ok {
|
|
||||||
dataType := reflect.TypeOf(r.Data())
|
|
||||||
allowedType := reflect.TypeFor[T]()
|
|
||||||
return "", fmt.Errorf("marshaler for '%s' cannot parse '%s'", allowedType, dataType)
|
|
||||||
}
|
|
||||||
|
|
||||||
return c.codec.Encode(t)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c convertedMarshaler[T]) InType() string { return c.inType }
|
|
||||||
|
|
||||||
func RegisterMarshaler[T any](registry *Registry, m codec.Marshaler[T], inType string) error {
|
|
||||||
if _, ok := registry.marshalers[inType]; ok {
|
|
||||||
return fmt.Errorf("marshaler for '%s' already registered", inType)
|
|
||||||
}
|
|
||||||
|
|
||||||
registry.marshalers[inType] = convertedMarshaler[T]{m, inType}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -7,14 +7,14 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Registry struct {
|
type Registry struct {
|
||||||
marshalers map[string]Marshaler
|
codecs map[string]Codec
|
||||||
converter *Converter
|
converter *Converter
|
||||||
engines map[string]Engine
|
engines map[string]Engine
|
||||||
}
|
}
|
||||||
|
|
||||||
func New() *Registry {
|
func New() *Registry {
|
||||||
return &Registry{
|
return &Registry{
|
||||||
marshalers: map[string]Marshaler{},
|
codecs: map[string]Codec{},
|
||||||
converter: NewConverter(),
|
converter: NewConverter(),
|
||||||
engines: map[string]Engine{},
|
engines: map[string]Engine{},
|
||||||
}
|
}
|
||||||
@@ -61,7 +61,7 @@ func (r *Registry) ConvertTo(repr Repr, outType string) (Repr, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (r *Registry) Marshal(repr Repr) (string, error) {
|
func (r *Registry) Marshal(repr Repr) (string, error) {
|
||||||
m, ok := r.marshalers[repr.ID()]
|
m, ok := r.codecs[repr.ID()]
|
||||||
if !ok {
|
if !ok {
|
||||||
return "", fmt.Errorf("no marshaler for '%s'", repr.ID())
|
return "", fmt.Errorf("no marshaler for '%s'", repr.ID())
|
||||||
}
|
}
|
||||||
@@ -70,7 +70,7 @@ func (r *Registry) Marshal(repr Repr) (string, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (r *Registry) Unmarshal(s string, outType string) (Repr, error) {
|
func (r *Registry) Unmarshal(s string, outType string) (Repr, error) {
|
||||||
m, ok := r.marshalers[outType]
|
m, ok := r.codecs[outType]
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, fmt.Errorf("no marshaler for '%s'", outType)
|
return nil, fmt.Errorf("no marshaler for '%s'", outType)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package codec
|
package codec
|
||||||
|
|
||||||
type Codec[T, U any] interface {
|
type Conversion[T, U any] = func(T) (U, error)
|
||||||
Encode(T) (U, error)
|
|
||||||
Decode(U) (T, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type Marshaler[T any] = Codec[T, string]
|
type Codec[T any] interface {
|
||||||
|
Encode(T) (string, error)
|
||||||
|
Decode(string) (T, error)
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package convert
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/codec"
|
|
||||||
"git.maximhutz.com/max/lambda/pkg/lambda"
|
"git.maximhutz.com/max/lambda/pkg/lambda"
|
||||||
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
"git.maximhutz.com/max/lambda/pkg/saccharine"
|
||||||
)
|
)
|
||||||
@@ -125,14 +124,10 @@ func decodeExression(l lambda.Expression) saccharine.Expression {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type Saccharine2Lambda struct{}
|
func Lambda2Saccharine(l lambda.Expression) (saccharine.Expression, error) {
|
||||||
|
|
||||||
func (c Saccharine2Lambda) Decode(l lambda.Expression) (saccharine.Expression, error) {
|
|
||||||
return decodeExression(l), nil
|
return decodeExression(l), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c Saccharine2Lambda) Encode(s saccharine.Expression) (lambda.Expression, error) {
|
func Saccharine2Lambda(s saccharine.Expression) (lambda.Expression, error) {
|
||||||
return encodeExpression(s), nil
|
return encodeExpression(s), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ codec.Codec[saccharine.Expression, lambda.Expression] = (*Saccharine2Lambda)(nil)
|
|
||||||
|
|||||||
@@ -16,4 +16,4 @@ func (m Marshaler) Encode(e Expression) (string, error) {
|
|||||||
return e.String(), nil
|
return e.String(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ codec.Marshaler[Expression] = (*Marshaler)(nil)
|
var _ codec.Codec[Expression] = (*Marshaler)(nil)
|
||||||
@@ -21,4 +21,4 @@ func (m Marshaler) Encode(e Expression) (string, error) {
|
|||||||
return stringifyExpression(e), nil
|
return stringifyExpression(e), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ codec.Marshaler[Expression] = (*Marshaler)(nil)
|
var _ codec.Codec[Expression] = (*Marshaler)(nil)
|
||||||
Reference in New Issue
Block a user