Skip to content

Commit 87c0967

Browse files
committed
support for Timeless FuncValues
1 parent 44e3c06 commit 87c0967

File tree

2 files changed

+38
-21
lines changed

2 files changed

+38
-21
lines changed

lang/funcs/structs/util.go

+34-14
Original file line numberDiff line numberDiff line change
@@ -51,18 +51,9 @@ func SimpleFnToDirectFunc(name string, fv *types.FuncValue) interfaces.Func {
5151
// *full.FuncValue.
5252
func SimpleFnToFuncValue(name string, fv *types.FuncValue) *full.FuncValue {
5353
return &full.FuncValue{
54-
V: func(txn interfaces.Txn, args []interfaces.Func) (interfaces.Func, error) {
55-
wrappedFunc := SimpleFnToDirectFunc(name, fv)
56-
txn.AddVertex(wrappedFunc)
57-
for i, arg := range args {
58-
argName := fv.T.Ord[i]
59-
txn.AddEdge(arg, wrappedFunc, &interfaces.FuncEdge{
60-
Args: []string{argName},
61-
})
62-
}
63-
return wrappedFunc, nil
64-
},
65-
T: fv.T,
54+
Name: &name,
55+
Timeless: fv,
56+
T: fv.T,
6657
}
6758
}
6859

@@ -76,9 +67,10 @@ func SimpleFnToConstFunc(name string, fv *types.FuncValue) interfaces.Func {
7667
// FuncToFullFuncValue creates a *full.FuncValue which adds the given
7768
// interfaces.Func to the graph. Note that this means the *full.FuncValue
7869
// can only be called once.
79-
func FuncToFullFuncValue(valueTransformingFunc interfaces.Func, typ *types.Type) *full.FuncValue {
70+
func FuncToFullFuncValue(name string, valueTransformingFunc interfaces.Func, typ *types.Type) *full.FuncValue {
8071
return &full.FuncValue{
81-
V: func(txn interfaces.Txn, args []interfaces.Func) (interfaces.Func, error) {
72+
Name: &name,
73+
Timeful: func(txn interfaces.Txn, args []interfaces.Func) (interfaces.Func, error) {
8274
for i, arg := range args {
8375
argName := typ.Ord[i]
8476
txn.AddEdge(arg, valueTransformingFunc, &interfaces.FuncEdge{
@@ -90,3 +82,31 @@ func FuncToFullFuncValue(valueTransformingFunc interfaces.Func, typ *types.Type)
9082
T: typ,
9183
}
9284
}
85+
86+
// Call calls the function with the provided txn and args.
87+
func CallFuncValue(obj *full.FuncValue, txn interfaces.Txn, args []interfaces.Func) (interfaces.Func, error) {
88+
if obj.Timeful != nil {
89+
return obj.Timeful(txn, args)
90+
}
91+
92+
wrappedFunc := SimpleFnToDirectFunc(*obj.Name, obj.Timeless)
93+
txn.AddVertex(wrappedFunc)
94+
for i, arg := range args {
95+
argName := obj.T.Ord[i]
96+
txn.AddEdge(arg, wrappedFunc, &interfaces.FuncEdge{
97+
Args: []string{argName},
98+
})
99+
}
100+
return wrappedFunc, nil
101+
}
102+
103+
// Speculatively call the function with the provided arguments.
104+
// Only makes sense if the function is timeless (produces a single Value, not a
105+
// stream of values).
106+
func CallTimelessFuncValue(obj *full.FuncValue, args []types.Value) (types.Value, error) {
107+
if obj.Timeless != nil {
108+
return obj.Timeless.V(args)
109+
}
110+
111+
panic("cannot call CallIfTimeless on a Timeful function")
112+
}

lang/types/full/full.go

+4-7
Original file line numberDiff line numberDiff line change
@@ -41,8 +41,10 @@ import (
4141
// went horribly wrong. (Think, an internal panic.)
4242
type FuncValue struct {
4343
types.Base
44-
V func(interfaces.Txn, []interfaces.Func) (interfaces.Func, error)
45-
T *types.Type // contains ordered field types, arg names are a bonus part
44+
Name *string
45+
Timeful func(interfaces.Txn, []interfaces.Func) (interfaces.Func, error)
46+
Timeless *types.FuncValue
47+
T *types.Type // contains ordered field types, arg names are a bonus part
4648
}
4749

4850
// String returns a visual representation of this value.
@@ -111,8 +113,3 @@ func (obj *FuncValue) Value() interface{} {
111113
//val := reflect.MakeFunc(typ, fn)
112114
//return val.Interface()
113115
}
114-
115-
// Call calls the function with the provided txn and args.
116-
func (obj *FuncValue) Call(txn interfaces.Txn, args []interfaces.Func) (interfaces.Func, error) {
117-
return obj.V(txn, args)
118-
}

0 commit comments

Comments
 (0)