kmodule.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. // package kmodule -- модуль на основе ядра
  2. package kmodule
  3. import (
  4. "fmt"
  5. "time"
  6. "gitp78su.ipnodns.ru/svi/kern/v4/kc/safe_int"
  7. "gitp78su.ipnodns.ru/svi/kern/v4/kc/safe_string"
  8. "gitp78su.ipnodns.ru/svi/kern/v4/krn/kbus/kbus_local"
  9. "gitp78su.ipnodns.ru/svi/kern/v4/krn/kctx"
  10. "gitp78su.ipnodns.ru/svi/kern/v4/krn/kmodule/mod_stat"
  11. . "gitp78su.ipnodns.ru/svi/kern/v4/lev0/kalias"
  12. . "gitp78su.ipnodns.ru/svi/kern/v4/lev0/ktypes"
  13. . "gitp78su.ipnodns.ru/svi/kern/v4/lev1/helpers"
  14. "gitp78su.ipnodns.ru/svi/kern/v4/lev1/local_ctx"
  15. . "gitp78su.ipnodns.ru/svi/kern/v4/lev1/result"
  16. )
  17. // kModule -- модуль на основе ядра
  18. type kModule struct {
  19. kCtx IKernelCtx
  20. ctx ILocalCtx
  21. name AModuleName
  22. bus IKernelBus
  23. timePhase ISafeInt
  24. strLive ISafeString
  25. stat IModuleStat
  26. }
  27. // NewKernelModule -- возвращает новый модуль на основе ядра
  28. func NewKernelModule(name AModuleName) IResult[IKernelModule] {
  29. if name == "" {
  30. err := fmt.Errorf("NewKernelModule(): name is empty")
  31. return NewErr[IKernelModule](err)
  32. }
  33. kCtx := kctx.GetKernelCtx()
  34. sf := &kModule{
  35. kCtx: kCtx,
  36. ctx: local_ctx.NewLocalCtx(kCtx.Ctx()),
  37. name: name,
  38. bus: kbus_local.GetKernelBusLocal(),
  39. timePhase: safe_int.NewSafeInt(1000), // 1000 msec
  40. strLive: safe_string.NewSafeString(),
  41. stat: mod_stat.NewModStat(name),
  42. }
  43. go sf.sigLive()
  44. return NewRes(IKernelModule(sf))
  45. }
  46. // Stat -- возвращает статистику модуля
  47. func (sf *kModule) Stat() IModuleStat {
  48. return sf.stat
  49. }
  50. // Log -- возвращает буферный лог
  51. func (sf *kModule) Log() ILogBuf {
  52. return sf.ctx.Log()
  53. }
  54. // Ctx -- возвращает контекст модуля
  55. func (sf *kModule) Ctx() ILocalCtx {
  56. return sf.ctx
  57. }
  58. // Run -- запускает модуль в работу
  59. func (sf *kModule) Run() {
  60. Hassert(false, "kModule.Run(): module='%v', parent not realised this method", sf.name)
  61. }
  62. // Name -- возвращает уникальное имя модуля
  63. func (sf *kModule) Name() AModuleName {
  64. return sf.name
  65. }
  66. // IsWork -- возвращает признак состояния работы
  67. func (sf *kModule) IsWork() bool {
  68. Hassert(false, "kModule.IsWork(): module='%v', parent not realised this method", sf.name)
  69. return false
  70. }
  71. // Live -- возвращает индикатор жизни модуля
  72. func (sf *kModule) Live() string {
  73. return sf.strLive.Get()
  74. }
  75. // Сигнал жизни, каждые 5 сек публикует в шину метку
  76. func (sf *kModule) sigLive() {
  77. var (
  78. topic = sf.name + "_live"
  79. iPhase = 0
  80. res IResult[bool]
  81. )
  82. fnPhase := func() {
  83. time.Sleep(time.Millisecond * time.Duration(sf.timePhase.Get()))
  84. select {
  85. case <-sf.kCtx.Ctx().Done():
  86. return
  87. default:
  88. switch iPhase {
  89. case 0:
  90. sf.strLive.Set("|")
  91. res = sf.bus.Publish(ATopic(topic), sf.strLive.Byte())
  92. case 1:
  93. sf.strLive.Set("/")
  94. res = sf.bus.Publish(ATopic(topic), sf.strLive.Byte())
  95. case 2:
  96. sf.strLive.Set("-")
  97. res = sf.bus.Publish(ATopic(topic), sf.strLive.Byte())
  98. case 3:
  99. sf.strLive.Set("\\")
  100. res = sf.bus.Publish(ATopic(topic), sf.strLive.Byte())
  101. iPhase = -1
  102. }
  103. res.Hassert("kModule.sigLive(): name=%v, in publish live", sf.Name())
  104. iPhase++
  105. sf.stat.Add(1)
  106. }
  107. }
  108. for {
  109. select {
  110. case <-sf.kCtx.Ctx().Done():
  111. return
  112. default:
  113. fnPhase()
  114. }
  115. }
  116. }