package storage import ( "context" "fmt" "github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/aclrecordproto" "github.com/anytypeio/go-anytype-infrastructure-experiments/pkg/acl/treechangeproto" "sync" ) type inMemoryACLListStorage struct { records []*aclrecordproto.RawACLRecordWithId id string sync.RWMutex } func NewInMemoryACLListStorage( id string, records []*aclrecordproto.RawACLRecordWithId) (ListStorage, error) { return &inMemoryACLListStorage{ id: id, records: records, RWMutex: sync.RWMutex{}, }, nil } func (i *inMemoryACLListStorage) Root() (*aclrecordproto.RawACLRecordWithId, error) { i.RLock() defer i.RUnlock() return i.records[0], nil } func (i *inMemoryACLListStorage) Head() (*aclrecordproto.RawACLRecordWithId, error) { i.RLock() defer i.RUnlock() return i.records[len(i.records)-1], nil } func (i *inMemoryACLListStorage) GetRawRecord(ctx context.Context, id string) (*aclrecordproto.RawACLRecordWithId, error) { i.RLock() defer i.RUnlock() for _, rec := range i.records { if rec.Id == id { return rec, nil } } return nil, fmt.Errorf("no such record") } func (i *inMemoryACLListStorage) AddRawRecord(ctx context.Context, rec *aclrecordproto.RawACLRecordWithId) error { panic("implement me") } func (i *inMemoryACLListStorage) ID() (string, error) { i.RLock() defer i.RUnlock() return i.id, nil } type inMemoryTreeStorage struct { id string root *treechangeproto.RawTreeChangeWithId heads []string changes map[string]*treechangeproto.RawTreeChangeWithId sync.RWMutex } func NewInMemoryTreeStorage( treeId string, root *treechangeproto.RawTreeChangeWithId, heads []string, changes []*treechangeproto.RawTreeChangeWithId) (TreeStorage, error) { allChanges := make(map[string]*treechangeproto.RawTreeChangeWithId) for _, ch := range changes { allChanges[ch.Id] = ch } allChanges[treeId] = root return &inMemoryTreeStorage{ id: treeId, root: root, heads: heads, changes: allChanges, RWMutex: sync.RWMutex{}, }, nil } func (t *inMemoryTreeStorage) ID() (string, error) { t.RLock() defer t.RUnlock() return t.id, nil } func (t *inMemoryTreeStorage) Root() (*treechangeproto.RawTreeChangeWithId, error) { t.RLock() defer t.RUnlock() return t.root, nil } func (t *inMemoryTreeStorage) Heads() ([]string, error) { t.RLock() defer t.RUnlock() return t.heads, nil } func (t *inMemoryTreeStorage) SetHeads(heads []string) error { t.Lock() defer t.Unlock() t.heads = t.heads[:0] for _, h := range heads { t.heads = append(t.heads, h) } return nil } func (t *inMemoryTreeStorage) AddRawChange(change *treechangeproto.RawTreeChangeWithId) error { t.Lock() defer t.Unlock() // TODO: better to do deep copy t.changes[change.Id] = change return nil } func (t *inMemoryTreeStorage) GetRawChange(ctx context.Context, changeId string) (*treechangeproto.RawTreeChangeWithId, error) { t.RLock() defer t.RUnlock() if res, exists := t.changes[changeId]; exists { return res, nil } return nil, fmt.Errorf("could not get change with id: %s", changeId) } type inMemoryStorageProvider struct { objects map[string]TreeStorage sync.RWMutex } func (i *inMemoryStorageProvider) TreeStorage(id string) (TreeStorage, error) { i.RLock() defer i.RUnlock() if tree, exists := i.objects[id]; exists { return tree, nil } return nil, ErrUnknownTreeId } func (i *inMemoryStorageProvider) CreateTreeStorage(payload TreeStorageCreatePayload) (TreeStorage, error) { i.Lock() defer i.Unlock() res, err := NewInMemoryTreeStorage(payload.TreeId, payload.RootRawChange, payload.Heads, payload.Changes) if err != nil { return nil, err } i.objects[payload.TreeId] = res return res, nil } func NewInMemoryTreeStorageProvider() Provider { return &inMemoryStorageProvider{ objects: make(map[string]TreeStorage), } }