mirror of
https://github.com/MontFerret/ferret.git
synced 2025-01-04 03:02:02 +02:00
95 lines
1.7 KiB
Go
95 lines
1.7 KiB
Go
package runtime
|
|
|
|
import (
|
|
"context"
|
|
"runtime"
|
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/core"
|
|
"github.com/MontFerret/ferret/pkg/runtime/logging"
|
|
"github.com/MontFerret/ferret/pkg/runtime/values"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type Program struct {
|
|
src string
|
|
body core.Expression
|
|
}
|
|
|
|
func NewProgram(src string, body core.Expression) (*Program, error) {
|
|
if src == "" {
|
|
return nil, core.Error(core.ErrMissedArgument, "source")
|
|
}
|
|
|
|
if body == nil {
|
|
return nil, core.Error(core.ErrMissedArgument, "body")
|
|
}
|
|
|
|
return &Program{src, body}, nil
|
|
}
|
|
|
|
func (p *Program) Source() string {
|
|
return p.src
|
|
}
|
|
|
|
func (p *Program) Run(ctx context.Context, setters ...Option) (result []byte, err error) {
|
|
ctx = NewOptions(setters).WithContext(ctx)
|
|
|
|
logger := logging.FromContext(ctx)
|
|
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
// find out exactly what the error was and set err
|
|
switch x := r.(type) {
|
|
case string:
|
|
err = errors.New(x)
|
|
case error:
|
|
err = x
|
|
default:
|
|
err = errors.New("unknown panic")
|
|
}
|
|
|
|
b := make([]byte, 0, 20)
|
|
runtime.Stack(b, true)
|
|
|
|
logger.Error().
|
|
Timestamp().
|
|
Err(err).
|
|
Str("stack", string(b)).
|
|
Msg("Panic")
|
|
|
|
result = nil
|
|
}
|
|
}()
|
|
|
|
scope, closeFn := core.NewRootScope()
|
|
|
|
defer func() {
|
|
if err := closeFn(); err != nil {
|
|
logger.Error().
|
|
Timestamp().
|
|
Err(err).
|
|
Msg("closing root scope")
|
|
}
|
|
}()
|
|
|
|
out, err := p.body.Exec(ctx, scope)
|
|
|
|
if err != nil {
|
|
js, _ := values.None.MarshalJSON()
|
|
|
|
return js, err
|
|
}
|
|
|
|
return out.MarshalJSON()
|
|
}
|
|
|
|
func (p *Program) MustRun(ctx context.Context, setters ...Option) []byte {
|
|
out, err := p.Run(ctx, setters...)
|
|
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return out
|
|
}
|