-
Notifications
You must be signed in to change notification settings - Fork 3
Expand file tree
/
Copy pathnamedfunc.go
More file actions
65 lines (61 loc) · 2.17 KB
/
namedfunc.go
File metadata and controls
65 lines (61 loc) · 2.17 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
// This Source Code Form is subject to the terms of the Mozilla Public License,
// v. 2.0. If a copy of the MPL was not distributed with this file, You can
// obtain one at http://mozilla.org/MPL/2.0/.
package lib
import "github.com/islisp-dev/iris/core"
func checkLambdaList(e core.Environment, lambdaList core.Instance) core.Instance {
if err := ensure(e, core.ListClass, lambdaList); err != nil {
return err
}
for i, cadr := range lambdaList.(core.List).Slice() {
if core.DeepEqual(cadr, core.NewSymbol(":REST")) || core.DeepEqual(cadr, core.NewSymbol("&REST")) {
if lambdaList.(core.List).Length() != i+2 {
_, err := SignalCondition(e, core.NewArityError(e), Nil)
return err
}
}
}
return nil
}
func newNamedFunction(e core.Environment, functionName, lambdaList core.Instance, forms ...core.Instance) (core.Instance, core.Instance) {
lexical := e
if err := ensure(e, core.SymbolClass, functionName); err != nil {
return nil, err
}
if err := checkLambdaList(e, lambdaList); err != nil {
return nil, err
}
parameters := []core.Instance{}
variadic := false
for _, cadr := range lambdaList.(core.List).Slice() {
if core.DeepEqual(cadr, core.NewSymbol(":REST")) || core.DeepEqual(cadr, core.NewSymbol("&REST")) {
variadic = true
}
parameters = append(parameters, cadr)
}
return core.NewFunction(functionName, func(e core.Environment, arguments ...core.Instance) (core.Instance, core.Instance) {
e.MergeLexical(lexical)
if (variadic && len(parameters)-2 > len(arguments)) || (!variadic && len(parameters) != len(arguments)) {
return SignalCondition(e, core.NewArityError(e), Nil)
}
for idx := range parameters {
key := parameters[idx]
if core.DeepEqual(key, core.NewSymbol(":REST")) || core.DeepEqual(key, core.NewSymbol("&REST")) {
key := parameters[idx+1]
value, err := List(e, arguments[idx:]...)
if err != nil {
return nil, err
}
if !e.Variable.Define(key, value) {
return SignalCondition(e, core.NewImmutableBinding(e), Nil)
}
break
}
value := arguments[idx]
if !e.Variable.Define(key, value) {
return SignalCondition(e, core.NewImmutableBinding(e), Nil)
}
}
return Progn(e, forms...)
}), nil
}