store.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. // package store -- главный тип плагина хранилища
  2. package store
  3. import (
  4. "fmt"
  5. "log"
  6. "sync"
  7. "github.com/syndtr/goleveldb/leveldb"
  8. "github.com/syndtr/goleveldb/leveldb/util"
  9. "wartank/pkg/components/safe_bool"
  10. "wartank/pkg/types"
  11. )
  12. // Store -- главный тип плагина хранилища
  13. type Store struct {
  14. ядро types.ИЯдро
  15. path string // Путь к хранилищу
  16. isWork *safe_bool.БезопБул // Признак открытой базы
  17. db *leveldb.DB // Объект хранилища
  18. block sync.Mutex // Блокировка для безопасности потоков
  19. }
  20. // NewStore -- возвращает новое хранилище
  21. func NewStore(ядро types.ИЯдро, path string) (*Store, error) {
  22. { // Прдусловия
  23. if ядро == nil {
  24. return nil, fmt.Errorf("NewStore: ядро == nil")
  25. }
  26. if path == "" {
  27. return nil, fmt.Errorf("NewStore: path == nil")
  28. }
  29. }
  30. sf := &Store{
  31. ядро: ядро,
  32. path: path,
  33. isWork: safe_bool.НовБезопБул(),
  34. }
  35. if err := sf.open(); err != nil {
  36. return nil, fmt.Errorf("NewStore: in open db, err=\n\t%w", err)
  37. }
  38. ядро.Wg().Add("ядро")
  39. return sf, nil
  40. }
  41. // Put -- добавляет в хранилище значение по ключу
  42. func (sf *Store) Put(key string, value []byte) error {
  43. sf.block.Lock()
  44. defer sf.block.Unlock()
  45. if key == "" {
  46. return fmt.Errorf("Store.Put(): key is empty")
  47. }
  48. if value == nil {
  49. return fmt.Errorf("Store.Put(): value == nil")
  50. }
  51. if err := sf.db.Put([]byte(key), value, nil); err != nil {
  52. return fmt.Errorf("Store.Put(): in put value, err=\n\t%w", err)
  53. }
  54. return nil
  55. }
  56. // Get -- возвращает значение по ключу
  57. func (sf *Store) Get(key string) ([]byte, error) {
  58. sf.block.Lock()
  59. defer sf.block.Unlock()
  60. value, err := sf.db.Get([]byte(key), nil)
  61. if err != nil {
  62. return nil, fmt.Errorf("Store.Get(): in get value, err=\n\t%w", err)
  63. }
  64. return value, nil
  65. }
  66. // Find -- возвращает список ключей по префиксу
  67. func (sf *Store) Find(prefix string) ([]string, error) {
  68. sf.block.Lock()
  69. defer sf.block.Unlock()
  70. var keys []string
  71. slice := &util.Range{Start: []byte(prefix)}
  72. iter := sf.db.NewIterator(slice, nil)
  73. for iter.Next() {
  74. key := string(iter.Key())
  75. keys = append(keys, key)
  76. }
  77. iter.Release()
  78. if err := iter.Error(); err != nil {
  79. return nil, fmt.Errorf("Store.Find(): in find keys, err=\n\t%w", err)
  80. }
  81. return keys, nil
  82. }
  83. // Del -- удаляет значение по ключу
  84. func (sf *Store) Del(key string) error {
  85. sf.block.Lock()
  86. defer sf.block.Unlock()
  87. if err := sf.db.Delete([]byte(key), nil); err != nil {
  88. return fmt.Errorf("Store.Del(): in delete key, err=\n\t%w", err)
  89. }
  90. return nil
  91. }
  92. // open -- открывает хранилище
  93. func (sf *Store) open() error {
  94. sf.block.Lock()
  95. defer sf.block.Unlock()
  96. db, err := leveldb.OpenFile(sf.path, nil)
  97. fnOpen := func() {
  98. sf.db = db
  99. sf.isWork.Уст()
  100. go sf.close()
  101. }
  102. if err == nil {
  103. fnOpen()
  104. return nil
  105. }
  106. log.Printf("Store.open(): try restore db, in open err=\n\t%v", err)
  107. db, err = leveldb.RecoverFile(sf.path, nil)
  108. if err != nil {
  109. return fmt.Errorf("Store.open(): in recover db, err=\n\t%w", err)
  110. }
  111. fnOpen()
  112. return nil
  113. }
  114. // close -- закрывает хранилище по требованию
  115. func (sf *Store) close() {
  116. <-sf.ядро.CtxApp().Done()
  117. sf.block.Lock()
  118. defer sf.block.Unlock()
  119. sf.isWork.Сброс()
  120. if err := sf.db.Close(); err != nil {
  121. log.Printf("Store.close(): in close db, err=\n\t%v", err)
  122. return
  123. }
  124. sf.ядро.Wg().Done("ядро")
  125. }