mirror of
https://github.com/MontFerret/ferret.git
synced 2025-01-10 03:16:58 +02:00
fe7b45df6e
* Refactored networking * Some work * Added event loop * Renamed EventHandler to Handler * wip * Removed console logs * Added DOMManager * Refactored frame managment * Fixes * Fixed concurrency issues * Fixed unit tests * Improved EventLoop api * Some fixes * Refactored event loop. * Improved logic of initial page load * Cleaned up * Fixed linting issues * Fixed dom.Manager.Close * SOme works * Fixes * Removed fmt.Println statements * Refactored WaitForNavigation * Removed filter for e2e tests * Made Cookies Measurable * Made Cookies KeyedCollection * Fixes after code review * Updated e2e tests for iframes * Fixed iframe lookup in e2e tests * Added comments
143 lines
2.5 KiB
Go
143 lines
2.5 KiB
Go
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
|
|
}
|