call.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. package rules
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "fw/cp/constant"
  6. "fw/cp/node"
  7. "fw/cp/object"
  8. "fw/rt2"
  9. "fw/rt2/context"
  10. "fw/rt2/frame"
  11. rt_mod "fw/rt2/module"
  12. "fw/rt2/nodeframe"
  13. "fw/rt2/scope"
  14. "ypk/assert"
  15. )
  16. /**
  17. Для CallNode
  18. .Left() указывает на процедуру или на переменную процедурного типа
  19. .Left().Object() указывает на список внутренних объектов, в т.ч. переменных
  20. .Object() указывает первый элемент из списка входных параметров/переменных,
  21. то же что и.Left().Object().Link(), далее .Link() указывает на следующие за ним входные параметры
  22. .Right() указывает на узлы, которые передаются в параметры
  23. */
  24. var sys map[string]func(f frame.Frame, par node.Node)
  25. type Msg struct {
  26. Type string
  27. Data string
  28. }
  29. func callHandler(f frame.Frame, obj object.Object, data interface{}) {
  30. //n := rt2.Utils.NodeOf(f)
  31. //fmt.Println("call handler", obj)
  32. if obj == nil {
  33. return
  34. }
  35. m := rt_mod.DomainModule(f.Domain())
  36. cn := node.New(constant.CALL)
  37. cn.SetLeft(m.NodeByObject(obj))
  38. cc := node.New(constant.CONSTANT).(node.ConstantNode)
  39. cc.SetData(data)
  40. cn.SetRight(cc)
  41. rt2.Push(rt2.New(cn), f)
  42. }
  43. func process(f frame.Frame, par node.Node) {
  44. assert.For(par != nil, 20)
  45. sm := scope.This(f.Domain().Discover(context.SCOPE))
  46. switch par.(type) {
  47. case node.ConstantNode:
  48. msg := &Msg{}
  49. val := par.(node.ConstantNode).Data().(string)
  50. if err := json.Unmarshal([]byte(val), msg); err == nil {
  51. switch msg.Type {
  52. case "log":
  53. fmt.Println(msg.Data)
  54. callHandler(f, scope.FindObjByName(sm, "go_handler"), `{"type":"log"}`)
  55. default:
  56. panic(40)
  57. }
  58. }
  59. default:
  60. panic(fmt.Sprintln("unsupported param"))
  61. }
  62. }
  63. func init() {
  64. sys = make(map[string]func(f frame.Frame, par node.Node))
  65. sys["go_process"] = process
  66. }
  67. func syscall(f frame.Frame) {
  68. n := rt2.NodeOf(f)
  69. name := n.Left().Object().Name()
  70. sys[name](f, n.Right())
  71. }
  72. func callSeq(f frame.Frame) (seq frame.Sequence, ret frame.WAIT) {
  73. var fu nodeframe.FrameUtils
  74. n := fu.NodeOf(f)
  75. call := func(proc node.Node) {
  76. nf := fu.New(proc)
  77. fu.Push(nf, f)
  78. //передаем ссылку на цепочку значений параметров в данные фрейма входа в процедуру
  79. if (n.Right() != nil) && (proc.Object() != nil) {
  80. fu.DataOf(nf)[proc.Object()] = n.Right()
  81. } else {
  82. //fmt.Println("no data for call")
  83. }
  84. seq = func(f frame.Frame) (seq frame.Sequence, ret frame.WAIT) {
  85. var fu nodeframe.FrameUtils
  86. fu.DataOf(f.Parent())[n] = fu.DataOf(f)[n.Left().Object()]
  87. return frame.End()
  88. }
  89. ret = frame.LATER
  90. }
  91. switch n.Left().(type) {
  92. case node.ProcedureNode:
  93. m := rt_mod.DomainModule(f.Domain())
  94. proc := m.NodeByObject(n.Left().Object())
  95. call(proc)
  96. case node.VariableNode:
  97. m := rt_mod.DomainModule(f.Domain())
  98. sc := f.Domain().Discover(context.SCOPE).(scope.Manager)
  99. obj := sc.Select(scope.Designator(n.Left()))
  100. if obj, ok := obj.(object.Object); ok {
  101. proc := m.NodeByObject(obj)
  102. call(proc)
  103. } else {
  104. name := n.Left().Object().Name()
  105. switch {
  106. case name == "go_process":
  107. syscall(f)
  108. return frame.Tail(frame.STOP), frame.LATER
  109. default:
  110. panic(fmt.Sprintln("unknown sysproc variable", name))
  111. }
  112. }
  113. default:
  114. panic("unknown call left")
  115. }
  116. return seq, ret
  117. }