Compare commits

..

9 Commits

Author SHA1 Message Date
boyce
eaf20c4e3a 新增蓝图热更新功能 2025-11-17 15:01:51 +08:00
boyce
027e83b706 整理优化代码 2025-11-17 10:55:17 +08:00
boyce
f9be55e98d 优化蓝图结点 2025-11-13 11:04:35 +08:00
boyce
d7c4cfb1ef 优化ws超时 2025-11-12 16:46:35 +08:00
boyce
4cb6882a1a 新增结点数组下标 2025-11-10 10:33:07 +08:00
boyce
b78d9721f2 新增蓝图结点 2025-11-07 14:16:47 +08:00
boyce
f8953d1764 优化bp结点 2025-11-05 17:01:43 +08:00
boyce
fac7a323e1 清理文件 2025-11-03 14:23:18 +08:00
boyce
1995d91cfc 优化蓝图执行器 2025-10-30 16:00:46 +08:00
15 changed files with 344 additions and 177 deletions

View File

@@ -3,13 +3,14 @@ package cluster
import ( import (
"errors" "errors"
"fmt" "fmt"
"reflect"
"strings"
"sync"
"github.com/duanhf2012/origin/v2/event" "github.com/duanhf2012/origin/v2/event"
"github.com/duanhf2012/origin/v2/log" "github.com/duanhf2012/origin/v2/log"
"github.com/duanhf2012/origin/v2/rpc" "github.com/duanhf2012/origin/v2/rpc"
"github.com/duanhf2012/origin/v2/service" "github.com/duanhf2012/origin/v2/service"
"reflect"
"strings"
"sync"
) )
var configDir = "./config/" var configDir = "./config/"

View File

@@ -10,7 +10,7 @@ const (
Sys_Event_Tcp EventType = -3 Sys_Event_Tcp EventType = -3
Sys_Event_Http_Event EventType = -4 Sys_Event_Http_Event EventType = -4
Sys_Event_WebSocket EventType = -5 Sys_Event_WebSocket EventType = -5
Sys_Event_Kcp EventType = -6 Sys_Event_Kcp EventType = -6
Sys_Event_Node_Conn_Event EventType = -7 Sys_Event_Node_Conn_Event EventType = -7
Sys_Event_Nats_Conn_Event EventType = -8 Sys_Event_Nats_Conn_Event EventType = -8
Sys_Event_DiscoverService EventType = -9 Sys_Event_DiscoverService EventType = -9
@@ -18,6 +18,6 @@ const (
Sys_Event_EtcdDiscovery EventType = -11 Sys_Event_EtcdDiscovery EventType = -11
Sys_Event_Gin_Event EventType = -12 Sys_Event_Gin_Event EventType = -12
Sys_Event_FrameTick EventType = -13 Sys_Event_FrameTick EventType = -13
Sys_Event_ReloadBlueprint EventType = -14
Sys_Event_User_Define EventType = 1 Sys_Event_User_Define EventType = 1
) )

View File

@@ -3,12 +3,13 @@ package network
import ( import (
"crypto/tls" "crypto/tls"
"errors" "errors"
"github.com/duanhf2012/origin/v2/log"
"github.com/gorilla/websocket"
"net" "net"
"net/http" "net/http"
"sync" "sync"
"time" "time"
"github.com/duanhf2012/origin/v2/log"
"github.com/gorilla/websocket"
) )
type WSServer struct { type WSServer struct {
@@ -16,13 +17,16 @@ type WSServer struct {
MaxConnNum int MaxConnNum int
PendingWriteNum int PendingWriteNum int
MaxMsgLen uint32 MaxMsgLen uint32
HTTPTimeout time.Duration
CertFile string CertFile string
KeyFile string KeyFile string
NewAgent func(*WSConn) Agent NewAgent func(*WSConn) Agent
ln net.Listener ln net.Listener
handler *WSHandler handler *WSHandler
messageType int messageType int
HandshakeTimeout time.Duration
ReadTimeout time.Duration
WriteTimeout time.Duration
} }
type WSHandler struct { type WSHandler struct {
@@ -73,14 +77,14 @@ func (handler *WSHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
} }
handler.conns[conn] = struct{}{} handler.conns[conn] = struct{}{}
handler.mutexConns.Unlock() handler.mutexConns.Unlock()
c,ok:=conn.NetConn().(*net.TCPConn) c, ok := conn.NetConn().(*net.TCPConn)
if !ok { if !ok {
tlsConn,ok := conn.NetConn().(*tls.Conn) tlsConn, ok := conn.NetConn().(*tls.Conn)
if !ok { if !ok {
log.Error("conn error") log.Error("conn error")
return return
} }
c,ok = tlsConn.NetConn().(*net.TCPConn) c, ok = tlsConn.NetConn().(*net.TCPConn)
if !ok { if !ok {
log.Error("conn error") log.Error("conn error")
return return
@@ -127,10 +131,19 @@ func (server *WSServer) Start() error {
server.MaxMsgLen = 4096 server.MaxMsgLen = 4096
log.Info("invalid MaxMsgLen", log.Uint32("reset", server.MaxMsgLen)) log.Info("invalid MaxMsgLen", log.Uint32("reset", server.MaxMsgLen))
} }
if server.HTTPTimeout <= 0 { if server.HandshakeTimeout <= 0 {
server.HTTPTimeout = 10 * time.Second server.HandshakeTimeout = 15 * time.Second
log.Info("invalid HTTPTimeout", log.Duration("reset", server.HTTPTimeout)) log.Info("invalid HandshakeTimeout", log.Duration("reset", server.HandshakeTimeout))
} }
if server.ReadTimeout <= 0 {
server.ReadTimeout = 15 * time.Second
log.Info("invalid ReadTimeout", log.Duration("reset", server.ReadTimeout))
}
if server.WriteTimeout <= 0 {
server.WriteTimeout = 15 * time.Second
log.Info("invalid WriteTimeout", log.Duration("reset", server.WriteTimeout))
}
if server.NewAgent == nil { if server.NewAgent == nil {
log.Error("NewAgent must not be nil") log.Error("NewAgent must not be nil")
return errors.New("NewAgent must not be nil") return errors.New("NewAgent must not be nil")
@@ -159,7 +172,7 @@ func (server *WSServer) Start() error {
conns: make(WebsocketConnSet), conns: make(WebsocketConnSet),
messageType: server.messageType, messageType: server.messageType,
upgrader: websocket.Upgrader{ upgrader: websocket.Upgrader{
HandshakeTimeout: server.HTTPTimeout, HandshakeTimeout: server.HandshakeTimeout,
CheckOrigin: func(_ *http.Request) bool { return true }, CheckOrigin: func(_ *http.Request) bool { return true },
}, },
} }
@@ -167,8 +180,8 @@ func (server *WSServer) Start() error {
httpServer := &http.Server{ httpServer := &http.Server{
Addr: server.Addr, Addr: server.Addr,
Handler: server.handler, Handler: server.handler,
ReadTimeout: server.HTTPTimeout, ReadTimeout: server.ReadTimeout,
WriteTimeout: server.HTTPTimeout, WriteTimeout: server.WriteTimeout,
MaxHeaderBytes: 1024, MaxHeaderBytes: 1024,
} }

View File

@@ -2,13 +2,15 @@ package wsmodule
import ( import (
"fmt" "fmt"
"sync"
"time"
"github.com/duanhf2012/origin/v2/event" "github.com/duanhf2012/origin/v2/event"
"github.com/duanhf2012/origin/v2/log" "github.com/duanhf2012/origin/v2/log"
"github.com/duanhf2012/origin/v2/network" "github.com/duanhf2012/origin/v2/network"
"github.com/duanhf2012/origin/v2/network/processor" "github.com/duanhf2012/origin/v2/network/processor"
"github.com/duanhf2012/origin/v2/service" "github.com/duanhf2012/origin/v2/service"
"go.mongodb.org/mongo-driver/bson/primitive" "go.mongodb.org/mongo-driver/bson/primitive"
"sync"
) )
type WSModule struct { type WSModule struct {
@@ -36,6 +38,10 @@ type WSCfg struct {
LittleEndian bool //是否小端序 LittleEndian bool //是否小端序
KeyFile string KeyFile string
CertFile string CertFile string
HandshakeTimeoutSecond time.Duration
ReadTimeoutSecond time.Duration
WriteTimeoutSecond time.Duration
} }
type WSPackType int8 type WSPackType int8
@@ -63,6 +69,9 @@ func (ws *WSModule) OnInit() error {
ws.WSServer.PendingWriteNum = ws.wsCfg.PendingWriteNum ws.WSServer.PendingWriteNum = ws.wsCfg.PendingWriteNum
ws.WSServer.MaxMsgLen = ws.wsCfg.MaxMsgLen ws.WSServer.MaxMsgLen = ws.wsCfg.MaxMsgLen
ws.WSServer.Addr = ws.wsCfg.ListenAddr ws.WSServer.Addr = ws.wsCfg.ListenAddr
ws.WSServer.HandshakeTimeout = ws.wsCfg.HandshakeTimeoutSecond*time.Second
ws.WSServer.ReadTimeout = ws.wsCfg.ReadTimeoutSecond*time.Second
ws.WSServer.WriteTimeout = ws.wsCfg.WriteTimeoutSecond*time.Second
if ws.wsCfg.KeyFile != "" && ws.wsCfg.CertFile != "" { if ws.wsCfg.KeyFile != "" && ws.wsCfg.CertFile != "" {
ws.WSServer.KeyFile = ws.wsCfg.KeyFile ws.WSServer.KeyFile = ws.wsCfg.KeyFile

View File

@@ -3,78 +3,154 @@ package blueprint
import ( import (
"fmt" "fmt"
"sync/atomic" "sync/atomic"
"github.com/duanhf2012/origin/v2/log"
) )
type Blueprint struct { type Blueprint struct {
execNodes []IExecNode // 注册的定义执行结点 execNodes []IExecNode // 注册的定义执行结点
execNodeList []func() IExecNode
execPool ExecPool execPool *ExecPool
graphPool GraphPool graphPool *GraphPool
blueprintModule IBlueprintModule blueprintModule IBlueprintModule
mapGraph map[int64]IGraph mapGraph map[int64]IGraph
seedID int64 seedID int64
cancelTimer func(*uint64)bool cancelTimer func(*uint64) bool
execDefFilePath string // 执行结点定义文件路径
graphFilePath string // 蓝图文件路径
} }
func (bm *Blueprint) RegExecNode(execNode IExecNode) { func (bm *Blueprint) RegisterExecNode(execNodeFunc func() IExecNode) {
bm.execNodes = append(bm.execNodes, execNode) bm.execNodeList = append(bm.execNodeList, execNodeFunc)
} }
func (bm *Blueprint) regSysNode(){ type IExecNodeType[T any] interface {
bm.RegExecNode(&AddInt{}) *T
bm.RegExecNode(&SubInt{}) IExecNode
bm.RegExecNode(&MulInt{})
bm.RegExecNode(&DivInt{})
bm.RegExecNode(&ModInt{})
bm.RegExecNode(&RandNumber{})
bm.RegExecNode(&Entrance_ArrayParam{})
bm.RegExecNode(&Entrance_IntParam{})
bm.RegExecNode(&Entrance_Timer{})
bm.RegExecNode(&Output{})
bm.RegExecNode(&Sequence{})
bm.RegExecNode(&Foreach{})
bm.RegExecNode(&ForeachIntArray{})
bm.RegExecNode(&GetArrayInt{})
bm.RegExecNode(&GetArrayString{})
bm.RegExecNode(&GetArrayLen{})
bm.RegExecNode(&CreateIntArray{})
bm.RegExecNode(&CreateStringArray{})
bm.RegExecNode(&AppendIntegerToArray{})
bm.RegExecNode(&AppendStringToArray{})
bm.RegExecNode(&BoolIf{})
bm.RegExecNode(&GreaterThanInteger{})
bm.RegExecNode(&LessThanInteger{})
bm.RegExecNode(&EqualInteger{})
bm.RegExecNode(&RangeCompare{})
bm.RegExecNode(&Probability{})
bm.RegExecNode(&CreateTimer{})
} }
func (bm *Blueprint) Init(execDefFilePath string, graphFilePath string, blueprintModule IBlueprintModule,cancelTimer func(*uint64)bool) error { // 生成一个泛型函数返回func() IExecNode类型
bm.regSysNode() func NewExecNode[T any, P IExecNodeType[T]]() func() IExecNode {
err := bm.execPool.Load(execDefFilePath) return func() IExecNode {
var t T
return P(&t)
}
}
func (bm *Blueprint) regSysNodes() {
bm.RegisterExecNode(NewExecNode[AddInt]())
bm.RegisterExecNode(NewExecNode[SubInt]())
bm.RegisterExecNode(NewExecNode[MulInt]())
bm.RegisterExecNode(NewExecNode[DivInt]())
bm.RegisterExecNode(NewExecNode[ModInt]())
bm.RegisterExecNode(NewExecNode[RandNumber]())
bm.RegisterExecNode(NewExecNode[Entrance_ArrayParam]())
bm.RegisterExecNode(NewExecNode[Entrance_IntParam]())
bm.RegisterExecNode(NewExecNode[Entrance_Timer]())
bm.RegisterExecNode(NewExecNode[DebugOutput]())
bm.RegisterExecNode(NewExecNode[Sequence]())
bm.RegisterExecNode(NewExecNode[Foreach]())
bm.RegisterExecNode(NewExecNode[ForeachIntArray]())
bm.RegisterExecNode(NewExecNode[GetArrayInt]())
bm.RegisterExecNode(NewExecNode[GetArrayString]())
bm.RegisterExecNode(NewExecNode[GetArrayLen]())
bm.RegisterExecNode(NewExecNode[CreateIntArray]())
bm.RegisterExecNode(NewExecNode[CreateStringArray]())
bm.RegisterExecNode(NewExecNode[AppendIntegerToArray]())
bm.RegisterExecNode(NewExecNode[AppendStringToArray]())
bm.RegisterExecNode(NewExecNode[BoolIf]())
bm.RegisterExecNode(NewExecNode[GreaterThanInteger]())
bm.RegisterExecNode(NewExecNode[LessThanInteger]())
bm.RegisterExecNode(NewExecNode[EqualInteger]())
bm.RegisterExecNode(NewExecNode[RangeCompare]())
bm.RegisterExecNode(NewExecNode[EqualSwitch]())
bm.RegisterExecNode(NewExecNode[Probability]())
bm.RegisterExecNode(NewExecNode[CreateTimer]())
}
func (bm *Blueprint) StartHotReload() (func(),error) {
var execPool ExecPool
var graphPool GraphPool
// 加载配置结点生成名字对应的innerExecNode
err := execPool.Load(bm.execDefFilePath)
if err != nil {
return nil, err
}
// 将注册的实际执行结点与innerExecNode进行关联
for _, newExec := range bm.execNodeList {
e := newExec()
if !execPool.Register(e) {
return nil,fmt.Errorf("register exec failed,exec:%s", e.GetName())
}
}
// 加载所有的vgf蓝图文件
err = graphPool.Load(&execPool, bm.graphFilePath, bm.blueprintModule)
if err != nil {
return nil, err
}
// 返回配置加载后的刷新内存处理
return func() {
// 替换旧的执行池和图池
bm.execPool = &execPool
bm.graphPool = &graphPool
for _, gh := range bm.mapGraph {
gFileName := gh.GetGraphFileName()
bGraph := bm.graphPool.GetBaseGraph(gFileName)
if bGraph == nil {
log.Warn("GetBaseGraph fail", log.String("graph file name", gFileName))
bGraph = &baseGraph{entrance: map[int64]*execNode{}}
}
gh.HotReload(bGraph)
}
}, nil
}
func (bm *Blueprint) Init(execDefFilePath string, graphFilePath string, blueprintModule IBlueprintModule, cancelTimer func(*uint64) bool) error {
var execPool ExecPool
var graphPool GraphPool
// 加载配置结点生成名字对应的innerExecNode
err := execPool.Load(execDefFilePath)
if err != nil { if err != nil {
return err return err
} }
for _, e := range bm.execNodes { // 注册系统执行结点
if !bm.execPool.Register(e) { bm.regSysNodes()
// 将注册的实际执行结点与innerExecNode进行关联
for _, newExec := range bm.execNodeList {
e := newExec()
if !execPool.Register(e) {
return fmt.Errorf("register exec failed,exec:%s", e.GetName()) return fmt.Errorf("register exec failed,exec:%s", e.GetName())
} }
} }
err = bm.graphPool.Load(&bm.execPool, graphFilePath, blueprintModule) // 加载所有的vgf蓝图文件
err = graphPool.Load(&execPool, graphFilePath, blueprintModule)
if err != nil { if err != nil {
return err return err
} }
bm.execPool = &execPool
bm.graphPool = &graphPool
bm.cancelTimer = cancelTimer bm.cancelTimer = cancelTimer
bm.blueprintModule = blueprintModule bm.blueprintModule = blueprintModule
bm.mapGraph = make(map[int64]IGraph,128) bm.mapGraph = make(map[int64]IGraph, 128)
bm.execDefFilePath = execDefFilePath
bm.graphFilePath = graphFilePath
return nil return nil
} }
@@ -82,32 +158,41 @@ func (bm *Blueprint) Create(graphName string) int64 {
if graphName == "" { if graphName == "" {
return 0 return 0
} }
graphID := atomic.AddInt64(&bm.seedID, 1) graphID := atomic.AddInt64(&bm.seedID, 1)
bm.mapGraph[graphID] = bm.graphPool.Create(graphName, graphID) gr := bm.graphPool.Create(graphName, graphID)
if gr == nil {
return 0
}
bm.mapGraph[graphID] = gr
return graphID return graphID
} }
func (bm *Blueprint) TriggerEvent(graphID int64, eventID int64, args ...any) error{ func (bm *Blueprint) TriggerEvent(graphID int64, eventID int64, args ...any) error {
graph := bm.mapGraph[graphID] graph := bm.mapGraph[graphID]
if graph == nil { if graph == nil {
return fmt.Errorf("can not find graph:%d", graphID) return fmt.Errorf("can not find graph:%d", graphID)
} }
_,err:= graph.Do(eventID, args...) _, err := graph.Do(eventID, args...)
return err return err
} }
func (bm *Blueprint) Do(graphID int64, entranceID int64, args ...any) (Port_Array,error){ func (bm *Blueprint) Do(graphID int64, entranceID int64, args ...any) (Port_Array, error) {
graph := bm.mapGraph[graphID] graph := bm.mapGraph[graphID]
if graph == nil { if graph == nil {
return nil,fmt.Errorf("can not find graph:%d", graphID) return nil, fmt.Errorf("can not find graph:%d", graphID)
} }
return graph.Do(entranceID, args...) return graph.Do(entranceID, args...)
} }
func (bm *Blueprint) ReleaseGraph(graphID int64) { func (bm *Blueprint) ReleaseGraph(graphID int64) {
if graphID == 0 {
return
}
defer delete(bm.mapGraph, graphID) defer delete(bm.mapGraph, graphID)
graph := bm.mapGraph[graphID] graph := bm.mapGraph[graphID]
if graph == nil { if graph == nil {
@@ -117,7 +202,7 @@ func (bm *Blueprint) ReleaseGraph(graphID int64) {
graph.Release() graph.Release()
} }
func (bm *Blueprint) CancelTimerId(graphID int64, timerId *uint64) bool{ func (bm *Blueprint) CancelTimerId(graphID int64, timerId *uint64) bool {
tId := *timerId tId := *timerId
bm.cancelTimer(timerId) bm.cancelTimer(timerId)
@@ -126,11 +211,11 @@ func (bm *Blueprint) CancelTimerId(graphID int64, timerId *uint64) bool{
return false return false
} }
gr,ok := graph.(*Graph) gr, ok := graph.(*Graph)
if !ok { if !ok {
return false return false
} }
delete(gr.mapTimerID, tId) delete(gr.mapTimerID, tId)
return true return true
} }

View File

@@ -5,6 +5,11 @@ import (
) )
func TestExecMgr(t *testing.T) { func TestExecMgr(t *testing.T) {
var bp Blueprint
err := bp.Init("E:\\WorkSpace\\c4\\OriginNodeEditor\\json", "E:\\WorkSpace\\c4\\OriginNodeEditor\\vgf", nil, nil)
if err != nil {
t.Fatalf("Init failed,err:%v", err)
}
//graphTest2 := bp.Create("testForeach") //graphTest2 := bp.Create("testForeach")
//err = graphTest2.Do(EntranceID_IntParam, 1, 2, 3) //err = graphTest2.Do(EntranceID_IntParam, 1, 2, 3)

View File

@@ -2,15 +2,7 @@ package blueprint
import "fmt" import "fmt"
type IBaseExecNode interface { // IInnerExecNode 配置生成的结点
initInnerExecNode(innerNode *innerExecNode)
initExecNode(gr *Graph, en *execNode) error
GetPorts() ([]IPort, []IPort)
getExecNodeInfo() (*ExecContext, *execNode)
setExecNodeInfo(gr *ExecContext, en *execNode)
GetBlueprintModule() IBlueprintModule
}
type IInnerExecNode interface { type IInnerExecNode interface {
GetName() string GetName() string
SetExec(exec IExecNode) SetExec(exec IExecNode)
@@ -26,16 +18,27 @@ type IInnerExecNode interface {
GetOutPortParamStartIndex() int GetOutPortParamStartIndex() int
} }
// IBaseExecNode 实际注册的执行结点的基础结构体
type IBaseExecNode interface {
initInnerExecNode(innerNode *innerExecNode)
initExecNode(gr *Graph, en *execNode) error
GetPorts() ([]IPort, []IPort)
getExecNodeInfo() (*ExecContext, *execNode)
setExecNodeInfo(gr *ExecContext, en *execNode)
GetBlueprintModule() IBlueprintModule
}
// IExecNode 实际注册的执行结点
type IExecNode interface { type IExecNode interface {
IBaseExecNode
GetName() string GetName() string
DoNext(index int) error DoNext(index int) error
Exec() (int, error) // 返回后续执行的Node的Index Exec() (int, error) // 返回后续执行的Node的Index
GetNextExecLen() int GetNextExecLen() int
getInnerExecNode() IInnerExecNode getInnerExecNode() IInnerExecNode
setVariableName(name string) bool
} }
// 配置对应的基础信息+端口数据
type innerExecNode struct { type innerExecNode struct {
Name string Name string
Title string Title string
@@ -47,11 +50,11 @@ type innerExecNode struct {
outPortParamStartIndex int // 输出参数的起始索引,用于排除执行出口 outPortParamStartIndex int // 输出参数的起始索引,用于排除执行出口
IExecNode IExecNode // 实际注册的执行结点
} }
type BaseExecNode struct { type BaseExecNode struct {
*innerExecNode *innerExecNode // 内部注册的执行结点
// 执行时初始化的数据 // 执行时初始化的数据
*ExecContext *ExecContext
@@ -108,26 +111,12 @@ func (bc *BaseExecConfig) GetMaxOutPortId() int {
return maxPortId return maxPortId
} }
//func (em *innerExecNode) AppendInPort(port ...IPort) {
// if len(em.inPort) == 0 {
// em.inPortParamStartIndex = -1
// }
//
// for i := 0; i < len(port); i++ {
// if !port[i].IsPortExec() && em.inPortParamStartIndex < 0 {
// em.inPortParamStartIndex = len(em.inPort)
// }
//
// em.inPort = append(em.inPort, port[i])
// }
//}
func (em *innerExecNode) PrepareMaxInPortId(maxInPortId int) { func (em *innerExecNode) PrepareMaxInPortId(maxInPortId int) {
em.inPort = make([]IPort, 0, maxInPortId+1) em.inPort = make([]IPort, maxInPortId+1)
} }
func (em *innerExecNode) PrepareMaxOutPortId(maxOutPortId int) { func (em *innerExecNode) PrepareMaxOutPortId(maxOutPortId int) {
em.outPort = make([]IPort, 0, maxOutPortId+1) em.outPort = make([]IPort, maxOutPortId+1)
} }
func (em *innerExecNode) SetInPortById(id int, port IPort) bool { func (em *innerExecNode) SetInPortById(id int, port IPort) bool {
@@ -243,6 +232,10 @@ func (em *innerExecNode) GetOutPort(index int) IPort {
return em.outPort[index] return em.outPort[index]
} }
func (en *BaseExecNode) GetVariableName() string {
return en.execNode.variableName
}
func (en *BaseExecNode) GetBluePrintModule() IBlueprintModule { func (en *BaseExecNode) GetBluePrintModule() IBlueprintModule {
return en.gr.IBlueprintModule return en.gr.IBlueprintModule
} }
@@ -579,10 +572,6 @@ func (en *BaseExecNode) getInnerExecNode() IInnerExecNode {
return en.innerExecNode.IExecNode.(IInnerExecNode) return en.innerExecNode.IExecNode.(IInnerExecNode)
} }
func (en *BaseExecNode) setVariableName(name string) bool {
return false
}
func (en *BaseExecNode) GetBlueprintModule() IBlueprintModule { func (en *BaseExecNode) GetBlueprintModule() IBlueprintModule {
if en.gr == nil { if en.gr == nil {
return nil return nil

View File

@@ -5,19 +5,19 @@ import (
"fmt" "fmt"
"os" "os"
"path/filepath" "path/filepath"
"sort"
"strconv" "strconv"
"strings" "strings"
"sort"
) )
// 格式说明Entrance_ID // Entrance 格式:Entrance_XXXX_ID
const ( const (
Entrance = "Entrance_" Entrance = "Entrance_"
) )
type ExecPool struct { type ExecPool struct {
innerExecNodeMap map[string]IInnerExecNode innerExecNodeMap map[string]IInnerExecNode // 所有配置对应的结点信息
execNodeMap map[string]IExecNode execNodeMap map[string]IExecNode // 实际注册的执行结点
} }
func (em *ExecPool) Load(execDefFilePath string) error { func (em *ExecPool) Load(execDefFilePath string) error {
@@ -49,16 +49,19 @@ func (em *ExecPool) Load(execDefFilePath string) error {
// 只处理JSON文件 // 只处理JSON文件
if filepath.Ext(path) == ".json" { if filepath.Ext(path) == ".json" {
// 将配置的结点初始化为innerExecNode将加入到innerExecNodeMap中
return em.processJSONFile(path) return em.processJSONFile(path)
} }
return nil return nil
}) })
if err != nil { if err != nil {
return fmt.Errorf("failed to walk path %s: %v", execDefFilePath, err) return fmt.Errorf("failed to walk path %s: %v", execDefFilePath, err)
} }
return em.loadSysExec() // 生成变量配置对应的配置结点GetVar_类型、SetVar_类型
return em.regVariablesNode()
} }
// 处理单个JSON文件 // 处理单个JSON文件
@@ -84,21 +87,24 @@ func (em *ExecPool) processJSONFile(filePath string) error {
} }
for i := range baseExecConfig { for i := range baseExecConfig {
// 排序 // 对PortId进行排序
sort.Slice(baseExecConfig[i].Inputs, func(left, right int) bool { sort.Slice(baseExecConfig[i].Inputs, func(left, right int) bool {
return baseExecConfig[i].Inputs[left].PortId < baseExecConfig[i].Inputs[right].PortId return baseExecConfig[i].Inputs[left].PortId < baseExecConfig[i].Inputs[right].PortId
}) })
// 对PortId进行排序
sort.Slice(baseExecConfig[i].Outputs, func(left, right int) bool { sort.Slice(baseExecConfig[i].Outputs, func(left, right int) bool {
return baseExecConfig[i].Outputs[left].PortId < baseExecConfig[i].Outputs[right].PortId return baseExecConfig[i].Outputs[left].PortId < baseExecConfig[i].Outputs[right].PortId
}) })
exec, err := em.createExecFromJSON(baseExecConfig[i]) // 根据配置的结点信息创建innerExecNode
if err != nil { var execError error
return err exec, execError := em.createExecFromJSON(baseExecConfig[i])
if execError != nil {
return execError
} }
if !em.loadBaseExec(exec) { // 加载到innerExecNodeMap中
if !em.addInnerExec(exec) {
return fmt.Errorf("exec %s already registered", exec.GetName()) return fmt.Errorf("exec %s already registered", exec.GetName())
} }
} }
@@ -126,6 +132,7 @@ func (em *ExecPool) createPortByDataType(nodeName, portName, dataType string) (I
func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExecNode, error) { func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExecNode, error) {
var baseExec innerExecNode var baseExec innerExecNode
// 如果是入口名则按入口名Entrance_ArrayParam_000002生成结点名:Entrance_ArrayParam
entranceName, _, ok := getEntranceNodeNameAndID(baseExecConfig.Name) entranceName, _, ok := getEntranceNodeNameAndID(baseExecConfig.Name)
if ok { if ok {
baseExec.Name = entranceName baseExec.Name = entranceName
@@ -138,7 +145,7 @@ func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExe
baseExec.PrepareMaxInPortId(baseExecConfig.GetMaxInPortId()) baseExec.PrepareMaxInPortId(baseExecConfig.GetMaxInPortId())
baseExec.PrepareMaxOutPortId(baseExecConfig.GetMaxOutPortId()) baseExec.PrepareMaxOutPortId(baseExecConfig.GetMaxOutPortId())
// exec数量 // 初始化所有的输入端口
inExecNum := 0 inExecNum := 0
for index, input := range baseExecConfig.Inputs { for index, input := range baseExecConfig.Inputs {
portType := strings.ToLower(input.PortType) portType := strings.ToLower(input.PortType)
@@ -146,6 +153,7 @@ func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExe
return nil, fmt.Errorf("input %s data type %s not support", input.Name, input.DataType) return nil, fmt.Errorf("input %s data type %s not support", input.Name, input.DataType)
} }
// 输入执行结点只能有一个,且只能放在第一个
if portType == Config_PortType_Exec { if portType == Config_PortType_Exec {
if inExecNum > 0 { if inExecNum > 0 {
return nil, fmt.Errorf("inPort only allows one Execute,node name %s", baseExec.Name) return nil, fmt.Errorf("inPort only allows one Execute,node name %s", baseExec.Name)
@@ -155,19 +163,22 @@ func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExe
} }
inExecNum++ inExecNum++
// 设置执行端口
baseExec.SetInPortById(input.PortId, NewPortExec()) baseExec.SetInPortById(input.PortId, NewPortExec())
// baseExec.AppendInPort(NewPortExec())
continue continue
} }
// 根据类型设置对应的端口
port, err := em.createPortByDataType(baseExec.Name, input.Name, input.DataType) port, err := em.createPortByDataType(baseExec.Name, input.Name, input.DataType)
if err != nil { if err != nil {
return nil, err return nil, err
} }
// 根据PortId设置端口
baseExec.SetInPortById(input.PortId, port) baseExec.SetInPortById(input.PortId, port)
} }
// 初始化所有的输出端口
hasData := false hasData := false
for _, output := range baseExecConfig.Outputs { for _, output := range baseExecConfig.Outputs {
portType := strings.ToLower(output.PortType) portType := strings.ToLower(output.PortType)
@@ -180,11 +191,13 @@ func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExe
return nil, fmt.Errorf("the exec port can only be placed at the front,node name %s", baseExec.Name) return nil, fmt.Errorf("the exec port can only be placed at the front,node name %s", baseExec.Name)
} }
// 设置执行端口
if portType == Config_PortType_Exec { if portType == Config_PortType_Exec {
baseExec.SetOutPortById(output.PortId, NewPortExec()) baseExec.SetOutPortById(output.PortId, NewPortExec())
continue continue
} }
// 根据类型设置数据端口
hasData = true hasData = true
port, err := em.createPortByDataType(baseExec.Name, output.Name, output.DataType) port, err := em.createPortByDataType(baseExec.Name, output.Name, output.DataType)
if err != nil { if err != nil {
@@ -196,7 +209,7 @@ func (em *ExecPool) createExecFromJSON(baseExecConfig BaseExecConfig) (IInnerExe
return &baseExec, nil return &baseExec, nil
} }
func (em *ExecPool) loadBaseExec(exec IInnerExecNode) bool { func (em *ExecPool) addInnerExec(exec IInnerExecNode) bool {
if _, ok := em.innerExecNodeMap[exec.GetName()]; ok { if _, ok := em.innerExecNodeMap[exec.GetName()]; ok {
return false return false
} }
@@ -224,10 +237,14 @@ func (em *ExecPool) Register(exec IExecNode) bool {
return false return false
} }
// 设置实际执行结点中innerExecNode变量,BaseExecNode.innerExecNode = innerNode
baseExecNode.initInnerExecNode(innerNode.(*innerExecNode)) baseExecNode.initInnerExecNode(innerNode.(*innerExecNode))
// innerNode设置实际的exec变量,innerExecNode.IExecNode = exec
innerNode.SetExec(exec) innerNode.SetExec(exec)
em.execNodeMap[baseExec.GetName()] = baseExec // 将实际的执行结点保存到execNodeMap中
em.execNodeMap[baseExec.GetName()] = exec
return true return true
} }
@@ -238,7 +255,8 @@ func (em *ExecPool) GetExec(name string) IInnerExecNode {
return nil return nil
} }
func (em *ExecPool) loadSysExec() error { // regVariablesNode 注册变量结点GetVar_类型、SetVar_类型
func (em *ExecPool) regVariablesNode() error {
var err error var err error
if err = em.regGetVariables(Config_DataType_Int); err != nil { if err = em.regGetVariables(Config_DataType_Int); err != nil {
return err return err
@@ -300,7 +318,7 @@ func (em *ExecPool) regGetVariables(typ string) error {
var getVariablesNode GetVariablesNode var getVariablesNode GetVariablesNode
getVariablesNode.nodeName = baseExec.GetName() getVariablesNode.nodeName = baseExec.GetName()
if !em.loadBaseExec(&baseExec) { if !em.addInnerExec(&baseExec) {
return fmt.Errorf("exec %s already registered", baseExec.GetName()) return fmt.Errorf("exec %s already registered", baseExec.GetName())
} }
if !em.Register(&getVariablesNode) { if !em.Register(&getVariablesNode) {
@@ -336,7 +354,7 @@ func (em *ExecPool) regSetVariables(typ string) error {
baseExec.SetOutPortById(1, outPort) baseExec.SetOutPortById(1, outPort)
baseExec.IExecNode = &SetVariablesNode{nodeName: baseExec.GetName()} baseExec.IExecNode = &SetVariablesNode{nodeName: baseExec.GetName()}
if !em.loadBaseExec(&baseExec) { if !em.addInnerExec(&baseExec) {
return fmt.Errorf("exec %s already registered", baseExec.GetName()) return fmt.Errorf("exec %s already registered", baseExec.GetName())
} }
if !em.Register(baseExec.IExecNode) { if !em.Register(baseExec.IExecNode) {

View File

@@ -4,15 +4,20 @@ import (
"fmt" "fmt"
"time" "time"
"github.com/duanhf2012/origin/v2/log"
"github.com/duanhf2012/origin/v2/service" "github.com/duanhf2012/origin/v2/service"
"github.com/goccy/go-json" "github.com/goccy/go-json"
) )
const ReturnVarial = "g_Return" const ReturnVarial = "g_Return"
var IsDebug = false
type IGraph interface { type IGraph interface {
Do(entranceID int64, args ...any) (Port_Array, error) Do(entranceID int64, args ...any) (Port_Array, error)
Release() Release()
GetGraphFileName() string
HotReload(newBaseGraph *baseGraph)
} }
type IBlueprintModule interface { type IBlueprintModule interface {
@@ -28,7 +33,8 @@ type baseGraph struct {
} }
type Graph struct { type Graph struct {
graphID int64 graphFileName string
graphID int64
*baseGraph *baseGraph
graphContext graphContext
IBlueprintModule IBlueprintModule
@@ -137,12 +143,18 @@ func (gc *graphConfig) GetNodeByID(nodeID string) *nodeConfig {
} }
func (gr *Graph) Do(entranceID int64, args ...any) (Port_Array, error) { func (gr *Graph) Do(entranceID int64, args ...any) (Port_Array, error) {
if IsDebug {
log.Debug("Graph Do", log.String("graphName", gr.graphFileName), log.Int64("graphID", gr.graphID), log.Int64("entranceID", entranceID))
}
entranceNode := gr.entrance[entranceID] entranceNode := gr.entrance[entranceID]
if entranceNode == nil { if entranceNode == nil {
return nil, fmt.Errorf("entranceID:%d not found", entranceID) return nil, fmt.Errorf("entranceID:%d not found", entranceID)
} }
gr.variables = map[string]IPort{} gr.variables = map[string]IPort{}
gr.context = map[string]*ExecContext{}
if gr.globalVariables == nil { if gr.globalVariables == nil {
gr.globalVariables = map[string]IPort{} gr.globalVariables = map[string]IPort{}
} }
@@ -196,3 +208,11 @@ func (gr *Graph) Release() {
// 清理掉所有数据 // 清理掉所有数据
*gr = Graph{} *gr = Graph{}
} }
func (gr *Graph) HotReload(newBaseGraph *baseGraph) {
gr.baseGraph = newBaseGraph
}
func (gr *Graph) GetGraphFileName() string{
return gr.graphFileName
}

View File

@@ -52,6 +52,15 @@ func (gp *GraphPool) Load(execPool *ExecPool, graphFilePath string, blueprintMod
}) })
} }
func (gp *GraphPool) GetBaseGraph(graphName string) *baseGraph {
gr, ok := gp.mapGraphs[graphName]
if !ok {
return nil
}
return gr
}
func (gp *GraphPool) Create(graphName string, graphID int64) IGraph { func (gp *GraphPool) Create(graphName string, graphID int64) IGraph {
gr, ok := gp.mapGraphs[graphName] gr, ok := gp.mapGraphs[graphName]
if !ok { if !ok {
@@ -61,6 +70,7 @@ func (gp *GraphPool) Create(graphName string, graphID int64) IGraph {
var graph Graph var graph Graph
graph.baseGraph = gr graph.baseGraph = gr
graph.graphID = graphID graph.graphID = graphID
graph.graphFileName = graphName
graph.context = make(map[string]*ExecContext, 4) graph.context = make(map[string]*ExecContext, 4)
graph.IBlueprintModule = gp.blueprintModule graph.IBlueprintModule = gp.blueprintModule
return &graph return &graph
@@ -72,21 +82,25 @@ func (gp *GraphPool) processJSONFile(filePath string) error {
if err != nil { if err != nil {
return fmt.Errorf("failed to open file %s: %v", filePath, err) return fmt.Errorf("failed to open file %s: %v", filePath, err)
} }
defer func() { defer func() {
if err := file.Close(); err != nil { if err = file.Close(); err != nil {
fmt.Printf("关闭文件 %s 时出错: %v\n", filePath, err) fmt.Printf("关闭文件 %s 时出错: %v\n", filePath, err)
} }
}() }()
fileName := filepath.Base(filePath) fileName := filepath.Base(filePath)
ext := filepath.Ext(fileName) // 获取".html" ext := filepath.Ext(fileName) // 获取".vgf"
name := strings.TrimSuffix(fileName, ext) // 获取"name" name := strings.TrimSuffix(fileName, ext) // 获取"name"
// 解析文件
var gConfig graphConfig var gConfig graphConfig
decoder := json.NewDecoder(file) decoder := json.NewDecoder(file)
if err := decoder.Decode(&gConfig); err != nil { if err = decoder.Decode(&gConfig); err != nil {
return fmt.Errorf("failed to decode JSON from file %s: %v", filePath, err) return fmt.Errorf("failed to decode JSON from file %s: %v", filePath, err)
} }
// 预处理蓝图
return gp.prepareGraph(name, &gConfig) return gp.prepareGraph(name, &gConfig)
} }
@@ -137,8 +151,6 @@ func (gp *GraphPool) genVarExec(nodeCfg *nodeConfig, graphConfig *graphConfig) (
} }
e := gp.execPool.GetExec(nodeName) e := gp.execPool.GetExec(nodeName)
e.(IExecNode).setVariableName(varName)
return e, varName return e, varName
} }
@@ -174,7 +186,7 @@ func (gp *GraphPool) genAllNode(graphConfig *graphConfig) (map[string]*execNode,
func (gp *GraphPool) prepareOneNode(mapNodeExec map[string]*execNode, nodeExec *execNode, graphConfig *graphConfig, recursion *int) error { func (gp *GraphPool) prepareOneNode(mapNodeExec map[string]*execNode, nodeExec *execNode, graphConfig *graphConfig, recursion *int) error {
*recursion++ *recursion++
if *recursion > 100 { if *recursion > 256 {
return fmt.Errorf("recursion too deep") return fmt.Errorf("recursion too deep")
} }

View File

@@ -2,6 +2,7 @@ package blueprint
import ( import (
"fmt" "fmt"
"github.com/duanhf2012/origin/v2/log"
) )
type prePortNode struct { type prePortNode struct {
@@ -102,20 +103,6 @@ func (en *execNode) exec(gr *Graph) (int, error) {
return -1, err return -1, err
} }
//defer func() {
inPort, outPort := node.GetPorts()
debugString := "inPort:"
for i := 0; i < len(inPort); i++ {
debugString += fmt.Sprintf("%+v,", inPort[i])
}
debugString += " outPort:"
for i := 0; i < len(outPort); i++ {
debugString += fmt.Sprintf("%+v,", outPort[i])
}
fmt.Printf("exec node %s,%s\n", en.execNode.GetName(), debugString)
//}()
return e.Exec() return e.Exec()
} }
@@ -155,6 +142,10 @@ func (en *execNode) doSetInPort(gr *Graph, index int, inPort IPort) error {
} }
func (en *execNode) Do(gr *Graph, outPortArgs ...any) error { func (en *execNode) Do(gr *Graph, outPortArgs ...any) error {
if IsDebug {
log.Debug("Start ExecNode", log.String("Name",en.execNode.GetName()))
}
// 重新初始化上下文 // 重新初始化上下文
inPorts, outPorts := en.execNode.CloneInOutPort() inPorts, outPorts := en.execNode.CloneInOutPort()
gr.context[en.Id] = &ExecContext{ gr.context[en.Id] = &ExecContext{
@@ -164,7 +155,7 @@ func (en *execNode) Do(gr *Graph, outPortArgs ...any) error {
startOutIdx := en.execNode.GetOutPortParamStartIndex() startOutIdx := en.execNode.GetOutPortParamStartIndex()
for i := 0; i < len(outPortArgs); i++ { for i := 0; i < len(outPortArgs); i++ {
if i >= len(outPorts) { if i+startOutIdx >= len(outPorts) {
return fmt.Errorf("args %d not found in node %s", i, en.execNode.GetName()) return fmt.Errorf("args %d not found in node %s", i, en.execNode.GetName())
} }
@@ -194,6 +185,10 @@ func (en *execNode) Do(gr *Graph, outPortArgs ...any) error {
return err return err
} }
if IsDebug {
log.Debug("End ExecNode", log.String("Name",en.execNode.GetName()),log.Any("InPort",inPorts ),log.Any("OutPort",outPorts))
}
if nextIndex == -1 || en.nextNode == nil { if nextIndex == -1 || en.nextNode == nil {
return nil return nil
} }

View File

@@ -213,7 +213,7 @@ func (em *Port[T]) convertInt64(v any) (int64, bool) {
func (em *Port[T]) setAnyVale(v any) error { func (em *Port[T]) setAnyVale(v any) error {
switch v.(type) { switch v.(type) {
case int, int64: case int8,int16,int32,int, int64,uint8,uint16,uint32,uint, uint64:
val, ok := em.convertInt64(v) val, ok := em.convertInt64(v)
if !ok { if !ok {
return fmt.Errorf("port type is %T, but value is %v", em.PortVal, v) return fmt.Errorf("port type is %T, but value is %v", em.PortVal, v)

View File

@@ -1,4 +0,0 @@
package blueprint

View File

@@ -15,8 +15,6 @@ const (
EntranceID_Timer = 3 EntranceID_Timer = 3
) )
type Entrance_ArrayParam struct { type Entrance_ArrayParam struct {
BaseExecNode BaseExecNode
} }
@@ -53,15 +51,15 @@ func (em *Entrance_Timer) Exec() (int, error) {
return 0, nil return 0, nil
} }
type Output struct { type DebugOutput struct {
BaseExecNode BaseExecNode
} }
func (em *Output) GetName() string { func (em *DebugOutput) GetName() string {
return "Output" return "DebugOutput"
} }
func (em *Output) Exec() (int, error) { func (em *DebugOutput) Exec() (int, error) {
val, ok := em.GetInPortInt(1) val, ok := em.GetInPortInt(1)
if !ok { if !ok {
return 0, fmt.Errorf("output Exec inParam not found") return 0, fmt.Errorf("output Exec inParam not found")
@@ -77,7 +75,7 @@ func (em *Output) Exec() (int, error) {
return 0, fmt.Errorf("output Exec inParam not found") return 0, fmt.Errorf("output Exec inParam not found")
} }
fmt.Printf("output Exec inParam [%d] [%s] [%v]\n", val, valStr, valArray) log.Debug("DebugOutput Exec", log.Any("param1", val), log.Any("param2", valStr), log.Any("param3", valArray))
return 0, nil return 0, nil
} }
@@ -119,7 +117,8 @@ func (em *ForeachIntArray) Exec() (int, error) {
} }
for i := range array { for i := range array {
em.ExecContext.OutputPorts[2].SetInt(array[i].IntVal) em.ExecContext.OutputPorts[2].SetInt(Port_Int(i))
em.ExecContext.OutputPorts[3].SetInt(array[i].IntVal)
err := em.DoNext(0) err := em.DoNext(0)
if err != nil { if err != nil {
return -1, err return -1, err
@@ -471,6 +470,40 @@ func (em *RangeCompare) Exec() (int, error) {
return 0, nil return 0, nil
} }
// EqualSwitch 等于分支==
type EqualSwitch struct {
BaseExecNode
}
func (em *EqualSwitch) GetName() string {
return "EqualSwitch"
}
func (em *EqualSwitch) Exec() (int, error) {
inPortA := em.GetInPort(1)
if inPortA == nil {
return -1, fmt.Errorf("GreaterThanInteger inParam 1 not found")
}
ret, ok := inPortA.GetInt()
if !ok {
return -1, fmt.Errorf("GreaterThanInteger inParam 1 error")
}
intArray := em.execNode.GetInPortDefaultIntArrayValue(2)
if intArray == nil {
return 0, nil
}
for i := 0; i < len(intArray) && i < em.GetOutPortCount()-2; i++ {
if ret == intArray[i] {
return i + 2, nil
}
}
return 0, nil
}
// Probability 概率判断(万分比) // Probability 概率判断(万分比)
type Probability struct { type Probability struct {
BaseExecNode BaseExecNode

View File

@@ -12,13 +12,11 @@ const globalVariablesPrefix = "g_"
type GetVariablesNode struct { type GetVariablesNode struct {
BaseExecNode BaseExecNode
nodeName string nodeName string
varName string
} }
type SetVariablesNode struct { type SetVariablesNode struct {
BaseExecNode BaseExecNode
nodeName string nodeName string
varName string
} }
func (g *GetVariablesNode) GetName() string { func (g *GetVariablesNode) GetName() string {
@@ -27,14 +25,14 @@ func (g *GetVariablesNode) GetName() string {
func (g *GetVariablesNode) Exec() (int, error) { func (g *GetVariablesNode) Exec() (int, error) {
var port IPort var port IPort
if strings.HasPrefix(g.varName, globalVariablesPrefix) { if strings.HasPrefix(g.GetVariableName(), globalVariablesPrefix) {
port = g.gr.globalVariables[g.varName] port = g.gr.globalVariables[g.GetVariableName()]
} else { } else {
port = g.gr.variables[g.varName] port = g.gr.variables[g.GetVariableName()]
} }
if port == nil { if port == nil {
return -1, fmt.Errorf("variable %s not found,node name %s", g.varName, g.nodeName) return -1, fmt.Errorf("variable %s not found,node name %s", g.GetVariableName(), g.nodeName)
} }
if !g.SetOutPort(0, port) { if !g.SetOutPort(0, port) {
@@ -44,10 +42,7 @@ func (g *GetVariablesNode) Exec() (int, error) {
return 0, nil return 0, nil
} }
func (g *GetVariablesNode) setVariableName(name string) bool {
g.varName = name
return true
}
func (g *SetVariablesNode) GetName() string { func (g *SetVariablesNode) GetName() string {
return g.nodeName return g.nodeName
@@ -60,10 +55,10 @@ func (g *SetVariablesNode) Exec() (int, error) {
} }
varPort := port.Clone() varPort := port.Clone()
if strings.HasPrefix(g.varName, globalVariablesPrefix) { if strings.HasPrefix(g.GetVariableName(), globalVariablesPrefix) {
g.gr.globalVariables[g.varName] = varPort g.gr.globalVariables[g.GetVariableName()] = varPort
} else { } else {
g.gr.variables[g.varName] = varPort g.gr.variables[g.GetVariableName()] = varPort
} }
if !g.SetOutPort(1, varPort) { if !g.SetOutPort(1, varPort) {
@@ -73,7 +68,3 @@ func (g *SetVariablesNode) Exec() (int, error) {
return 0, nil return 0, nil
} }
func (g *SetVariablesNode) setVariableName(name string) bool {
g.varName = name
return true
}