package runtime import ( "context" "runtime" "strings" "github.com/pkg/errors" "github.com/MontFerret/ferret/pkg/runtime/core" "github.com/MontFerret/ferret/pkg/runtime/logging" "github.com/MontFerret/ferret/pkg/runtime/values" ) type Program struct { src string body core.Expression params map[string]struct{} } func NewProgram(src string, body core.Expression, params map[string]struct{}) (*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, params}, nil } func (p *Program) Source() string { return p.src } func (p *Program) Params() []string { res := make([]string, 0, len(p.params)) for name := range p.params { res = append(res, name) } return res } func (p *Program) Run(ctx context.Context, setters ...Option) (result []byte, err error) { opts := NewOptions(setters) err = p.validateParams(opts) if err != nil { return nil, err } ctx = opts.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 } func (p *Program) validateParams(opts *Options) error { if len(p.params) == 0 { return nil } // There might be no errors. // Thus, we allocate this slice lazily, on a first error. var missedParams []string for n := range p.params { _, exists := opts.params[n] if !exists { if missedParams == nil { missedParams = make([]string, 0, len(p.params)) } missedParams = append(missedParams, "@"+n) } } if len(missedParams) > 0 { return core.Error(ErrMissedParam, strings.Join(missedParams, ", ")) } return nil }