store.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. // package store -- хранилище на диске
  2. package store
  3. import (
  4. "fmt"
  5. "log"
  6. "os"
  7. "strings"
  8. "sync"
  9. "github.com/syndtr/goleveldb/leveldb"
  10. "github.com/syndtr/goleveldb/leveldb/util"
  11. "git.p78su.freemyip.com/svi/gostore/pkg/types"
  12. )
  13. // StoreDisk -- хранилище на диске
  14. type Store struct {
  15. serv types.IServCtx
  16. db *leveldb.DB
  17. isWork bool
  18. block sync.RWMutex
  19. }
  20. var IsBad_ = false // Для тестов
  21. // NewStore -- возвращает новое хранилище на диске
  22. func NewStore(serv types.IServCtx) (types.IStore, error) {
  23. if IsBad_ {
  24. return nil, fmt.Errorf("NewStore(): IsBad_==true")
  25. }
  26. if serv == nil {
  27. return nil, fmt.Errorf("NewStore(): IServCtx==nil")
  28. }
  29. _ = os.MkdirAll("./store/db", 0750)
  30. db, err := leveldb.OpenFile("./store/db", nil)
  31. if err != nil {
  32. if !strings.Contains(err.Error(), "leveldb: manifest corrupted") {
  33. return nil, fmt.Errorf("NewStore(): in create IStoreDisk, err=\n\t%w", err)
  34. }
  35. db, err = leveldb.RecoverFile("./stor/db", nil)
  36. if err != nil {
  37. return nil, fmt.Errorf("NewStore(): in recovery DB, err=\n\t%w", err)
  38. }
  39. }
  40. sf := &Store{
  41. serv: serv,
  42. db: db,
  43. }
  44. sf.serv.Wg().Add(1)
  45. sf.isWork = true
  46. go sf.close()
  47. return sf, nil
  48. }
  49. // Проверка на нерабочую схему
  50. func (sf *Store) isClosed() bool {
  51. sf.block.RLock()
  52. defer sf.block.RUnlock()
  53. return !sf.isWork
  54. }
  55. // Put -- размещает в хранилище ключ и значение
  56. func (sf *Store) Put(key string, val []byte) error {
  57. if sf.isClosed() {
  58. return fmt.Errorf("Store.Put(): store is closed")
  59. }
  60. if key == "" {
  61. return fmt.Errorf("Store.Put(): key is empty")
  62. }
  63. err := sf.db.Put([]byte(key), val, nil)
  64. if err != nil {
  65. return fmt.Errorf("Store.Put(): key=%q\terr=\n\t%w", key, err)
  66. }
  67. return nil
  68. }
  69. // Get -- возвращает значение ключа
  70. func (sf *Store) Get(key string) ([]byte, error) {
  71. if sf.isClosed() {
  72. return nil, fmt.Errorf("Store.Get(): store is closed")
  73. }
  74. val, err := sf.db.Get([]byte(key), nil)
  75. if err != nil {
  76. return nil, fmt.Errorf("Store.Get(): key=%q\terr=\n\t%w", key, err)
  77. }
  78. return val, nil
  79. }
  80. // Find -- ищет ключи по префиксу
  81. func (sf *Store) Find(prefixKey string) ([]string, error) {
  82. if sf.isClosed() {
  83. return nil, fmt.Errorf("Store.Find(): store is closed")
  84. }
  85. lstKey := []string{}
  86. iter := sf.db.NewIterator(util.BytesPrefix([]byte(prefixKey)), nil)
  87. for iter.Next() {
  88. key := iter.Key()
  89. lstKey = append(lstKey, string(key))
  90. }
  91. iter.Release()
  92. err := iter.Error()
  93. if err != nil {
  94. return nil, fmt.Errorf("Store.Find(): preefixKey=%q\terr=\n\t%w", prefixKey, err)
  95. }
  96. return lstKey, nil
  97. }
  98. // Del -- удаление ключа из базы
  99. func (sf *Store) Del(key string) error {
  100. if sf.isClosed() {
  101. return fmt.Errorf("Store.Del(): store is closed")
  102. }
  103. err := sf.db.Delete([]byte(key), nil)
  104. if err != nil {
  105. return fmt.Errorf("Store.Del(): key=%q\terr=\n\t%w", key, err)
  106. }
  107. return nil
  108. }
  109. // Ожидание закрытия приложения
  110. func (sf *Store) close() {
  111. <-sf.serv.Ctx().Done()
  112. sf.block.Lock()
  113. defer sf.block.Unlock()
  114. if !sf.isWork {
  115. return
  116. }
  117. sf.isWork = false
  118. err := sf.db.Close()
  119. if err != nil {
  120. log.Printf("Store.close(): err=\n\t%v\n", err)
  121. }
  122. sf.serv.Wg().Done()
  123. }