dict_topic.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. // package dict_topic -- потокобезопасный словарь топиков
  2. package dict_topic
  3. import (
  4. "log"
  5. "strings"
  6. "sync"
  7. "p78git.ddns.net/svi/gobus/internal/dict_topic/topic"
  8. "p78git.ddns.net/svi/gobus/pkg/alias"
  9. "p78git.ddns.net/svi/gobus/pkg/types"
  10. )
  11. // DictTopic -- потокобезопасный словарь топиков
  12. type DictTopic struct {
  13. dict map[alias.TopicName]types.ITopic // Словарь тпоиков
  14. dictProxy map[alias.ClientName]types.IClientProxy // Словарь прокси для клиентов
  15. block sync.RWMutex
  16. }
  17. // NewDictTopic -- возвращает новый потокобезопасный словарь топиков
  18. func NewDictTopic() (*DictTopic, error) {
  19. sf := &DictTopic{
  20. dict: make(map[alias.TopicName]types.ITopic),
  21. dictProxy: make(map[alias.ClientName]types.IClientProxy),
  22. }
  23. _ = types.IDictTopic(sf)
  24. return sf, nil
  25. }
  26. // Unsubscribe -- отписывает клиента от топиков
  27. func (sf *DictTopic) Unsubscribe(clientProxy types.IClientProxy) {
  28. sf.block.RLock()
  29. defer sf.block.RUnlock()
  30. sf.unsubscribe(clientProxy)
  31. }
  32. // Отписывает под блокировкой клиента от топиков
  33. func (sf *DictTopic) unsubscribe(clientProxy types.IClientProxy) {
  34. sample := clientProxy.SampleTopic()
  35. for name, topic := range sf.dict {
  36. if !strings.Contains(string(name), string(sample)) {
  37. continue
  38. }
  39. topic.Unsubscribe(clientProxy.ClientName())
  40. }
  41. }
  42. // Subscribe -- подписывает клиента на топики
  43. func (sf *DictTopic) Subscribe(clientProxy types.IClientProxy) {
  44. sf.block.RLock()
  45. defer sf.block.RUnlock()
  46. sf.subscribe(clientProxy)
  47. }
  48. // Подписывает по образцу топика клиента на события (внутри блокировки)
  49. func (sf *DictTopic) subscribe(clientProxy types.IClientProxy) {
  50. { // Сначала проверить наличие такого клиента и отписать его
  51. _, isOk := sf.dictProxy[clientProxy.ClientName()]
  52. if isOk {
  53. sf.unsubscribe(clientProxy)
  54. }
  55. }
  56. sample := clientProxy.SampleTopic()
  57. for name, topic := range sf.dict {
  58. if !strings.Contains(string(name), string(sample)) {
  59. continue
  60. }
  61. topic.Subscribe(clientProxy)
  62. }
  63. sf.dictProxy[clientProxy.ClientName()] = clientProxy
  64. }
  65. // Add -- добавляет новый топик
  66. func (sf *DictTopic) Add(name alias.TopicName) {
  67. sf.block.Lock()
  68. defer sf.block.Unlock()
  69. topic, err := topic.NewTopic(name)
  70. if err != nil {
  71. log.Printf("DictTopic.Add(): in create ITopic, err=\n\t%v\n", err)
  72. return
  73. }
  74. sf.dict[name] = topic
  75. { // Теперь пройтись по все прокси-клиентам и подписать их на новый топик
  76. for _, clientProxy := range sf.dictProxy {
  77. topic.Subscribe(clientProxy)
  78. }
  79. }
  80. }
  81. // Get -- возвращает топик по имени
  82. func (sf *DictTopic) Get(name alias.TopicName) types.ITopic {
  83. sf.block.RLock()
  84. defer sf.block.RUnlock()
  85. topic := sf.dict[name]
  86. return topic
  87. }
  88. // Len -- возвращает число всех топиков
  89. func (sf *DictTopic) Len() int {
  90. sf.block.RLock()
  91. defer sf.block.RUnlock()
  92. return len(sf.dict)
  93. }
  94. // Size -- возвращает размер всех топиков
  95. func (sf *DictTopic) Size() int {
  96. sf.block.RLock()
  97. defer sf.block.RUnlock()
  98. res := 0
  99. for _, topic := range sf.dict {
  100. res += topic.Size()
  101. }
  102. return res
  103. }