mirror of
https://github.com/anyproto/any-sync.git
synced 2025-06-11 18:20:28 +09:00
337 lines
8 KiB
Go
337 lines
8 KiB
Go
//go:generate mockgen -destination mock_list/mock_list.go github.com/anyproto/any-sync/commonspace/object/acl/list AclList
|
|
package list
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"sync"
|
|
|
|
"github.com/anyproto/any-sync/commonspace/object/accountdata"
|
|
"github.com/anyproto/any-sync/commonspace/object/acl/liststorage"
|
|
"github.com/anyproto/any-sync/consensus/consensusproto"
|
|
"github.com/anyproto/any-sync/util/cidutil"
|
|
"github.com/anyproto/any-sync/util/crypto"
|
|
)
|
|
|
|
type IterFunc = func(record *AclRecord) (IsContinue bool)
|
|
|
|
var (
|
|
ErrIncorrectCID = errors.New("incorrect CID")
|
|
ErrRecordAlreadyExists = errors.New("record already exists")
|
|
)
|
|
|
|
type RWLocker interface {
|
|
sync.Locker
|
|
RLock()
|
|
RUnlock()
|
|
}
|
|
|
|
type AcceptorVerifier interface {
|
|
VerifyAcceptor(rec *consensusproto.RawRecord) (err error)
|
|
}
|
|
|
|
type NoOpAcceptorVerifier struct {
|
|
}
|
|
|
|
func (n NoOpAcceptorVerifier) VerifyAcceptor(rec *consensusproto.RawRecord) (err error) {
|
|
return nil
|
|
}
|
|
|
|
type AclList interface {
|
|
RWLocker
|
|
Id() string
|
|
Root() *consensusproto.RawRecordWithId
|
|
Records() []*AclRecord
|
|
AclState() *AclState
|
|
IsAfter(first string, second string) (bool, error)
|
|
HasHead(head string) bool
|
|
Head() *AclRecord
|
|
|
|
RecordsAfter(ctx context.Context, id string) (records []*consensusproto.RawRecordWithId, err error)
|
|
Get(id string) (*AclRecord, error)
|
|
GetIndex(idx int) (*AclRecord, error)
|
|
Iterate(iterFunc IterFunc)
|
|
IterateFrom(startId string, iterFunc IterFunc)
|
|
|
|
KeyStorage() crypto.KeyStorage
|
|
RecordBuilder() AclRecordBuilder
|
|
|
|
ValidateRawRecord(record *consensusproto.RawRecord) (err error)
|
|
AddRawRecord(rawRec *consensusproto.RawRecordWithId) (err error)
|
|
AddRawRecords(rawRecords []*consensusproto.RawRecordWithId) (err error)
|
|
|
|
Close(ctx context.Context) (err error)
|
|
}
|
|
|
|
type aclList struct {
|
|
root *consensusproto.RawRecordWithId
|
|
records []*AclRecord
|
|
indexes map[string]int
|
|
id string
|
|
|
|
stateBuilder *aclStateBuilder
|
|
recordBuilder AclRecordBuilder
|
|
keyStorage crypto.KeyStorage
|
|
aclState *AclState
|
|
storage liststorage.ListStorage
|
|
|
|
sync.RWMutex
|
|
}
|
|
|
|
type internalDeps struct {
|
|
storage liststorage.ListStorage
|
|
keyStorage crypto.KeyStorage
|
|
stateBuilder *aclStateBuilder
|
|
recordBuilder AclRecordBuilder
|
|
acceptorVerifier AcceptorVerifier
|
|
}
|
|
|
|
func BuildAclListWithIdentity(acc *accountdata.AccountKeys, storage liststorage.ListStorage, verifier AcceptorVerifier) (AclList, error) {
|
|
keyStorage := crypto.NewKeyStorage()
|
|
deps := internalDeps{
|
|
storage: storage,
|
|
keyStorage: keyStorage,
|
|
stateBuilder: newAclStateBuilderWithIdentity(acc),
|
|
recordBuilder: NewAclRecordBuilder(storage.Id(), keyStorage, acc, verifier),
|
|
acceptorVerifier: verifier,
|
|
}
|
|
return build(deps)
|
|
}
|
|
|
|
func BuildAclList(storage liststorage.ListStorage, verifier AcceptorVerifier) (AclList, error) {
|
|
keyStorage := crypto.NewKeyStorage()
|
|
deps := internalDeps{
|
|
storage: storage,
|
|
keyStorage: keyStorage,
|
|
stateBuilder: newAclStateBuilder(),
|
|
recordBuilder: NewAclRecordBuilder(storage.Id(), keyStorage, nil, verifier),
|
|
acceptorVerifier: verifier,
|
|
}
|
|
return build(deps)
|
|
}
|
|
|
|
func build(deps internalDeps) (list AclList, err error) {
|
|
var (
|
|
storage = deps.storage
|
|
id = deps.storage.Id()
|
|
recBuilder = deps.recordBuilder
|
|
stateBuilder = deps.stateBuilder
|
|
)
|
|
head, err := storage.Head()
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
rawRecordWithId, err := storage.GetRawRecord(context.Background(), head)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
record, err := recBuilder.UnmarshallWithId(rawRecordWithId)
|
|
if err != nil {
|
|
return
|
|
}
|
|
records := []*AclRecord{record}
|
|
|
|
for record.PrevId != "" {
|
|
rawRecordWithId, err = storage.GetRawRecord(context.Background(), record.PrevId)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
record, err = recBuilder.UnmarshallWithId(rawRecordWithId)
|
|
if err != nil {
|
|
return
|
|
}
|
|
records = append(records, record)
|
|
}
|
|
|
|
indexes := make(map[string]int)
|
|
for i, j := 0, len(records)-1; i < j; i, j = i+1, j-1 {
|
|
records[i], records[j] = records[j], records[i]
|
|
indexes[records[i].Id] = i
|
|
indexes[records[j].Id] = j
|
|
}
|
|
// adding missed index if needed
|
|
if len(records)%2 != 0 {
|
|
indexes[records[len(records)/2].Id] = len(records) / 2
|
|
}
|
|
|
|
stateBuilder.Init(id)
|
|
state, err := stateBuilder.Build(records)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// TODO: check if this is correct (raw model instead of unmarshalled)
|
|
rootWithId, err := storage.Root()
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
recBuilder.(*aclRecordBuilder).state = state
|
|
list = &aclList{
|
|
root: rootWithId,
|
|
records: records,
|
|
indexes: indexes,
|
|
stateBuilder: stateBuilder,
|
|
recordBuilder: recBuilder,
|
|
aclState: state,
|
|
storage: storage,
|
|
id: id,
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *aclList) RecordBuilder() AclRecordBuilder {
|
|
return a.recordBuilder
|
|
}
|
|
|
|
func (a *aclList) Records() []*AclRecord {
|
|
return a.records
|
|
}
|
|
|
|
func (a *aclList) ValidateRawRecord(rawRec *consensusproto.RawRecord) (err error) {
|
|
record, err := a.recordBuilder.Unmarshall(rawRec)
|
|
if err != nil {
|
|
return
|
|
}
|
|
return a.aclState.Validator().ValidateAclRecordContents(record)
|
|
}
|
|
|
|
func (a *aclList) AddRawRecords(rawRecords []*consensusproto.RawRecordWithId) (err error) {
|
|
for _, rec := range rawRecords {
|
|
err = a.AddRawRecord(rec)
|
|
if err != nil && err != ErrRecordAlreadyExists {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *aclList) AddRawRecord(rawRec *consensusproto.RawRecordWithId) (err error) {
|
|
if _, ok := a.indexes[rawRec.Id]; ok {
|
|
return ErrRecordAlreadyExists
|
|
}
|
|
record, err := a.recordBuilder.UnmarshallWithId(rawRec)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if err = a.aclState.applyRecord(record); err != nil {
|
|
return
|
|
}
|
|
a.records = append(a.records, record)
|
|
a.indexes[record.Id] = len(a.records) - 1
|
|
if err = a.storage.AddRawRecord(context.Background(), rawRec); err != nil {
|
|
return
|
|
}
|
|
if err = a.storage.SetHead(rawRec.Id); err != nil {
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *aclList) Id() string {
|
|
return a.id
|
|
}
|
|
|
|
func (a *aclList) Root() *consensusproto.RawRecordWithId {
|
|
return a.root
|
|
}
|
|
|
|
func (a *aclList) AclState() *AclState {
|
|
return a.aclState
|
|
}
|
|
|
|
func (a *aclList) KeyStorage() crypto.KeyStorage {
|
|
return a.keyStorage
|
|
}
|
|
|
|
func (a *aclList) IsAfter(first string, second string) (bool, error) {
|
|
firstRec, okFirst := a.indexes[first]
|
|
secondRec, okSecond := a.indexes[second]
|
|
if !okFirst || !okSecond {
|
|
return false, fmt.Errorf("not all entries are there: first (%t), second (%t)", okFirst, okSecond)
|
|
}
|
|
return firstRec >= secondRec, nil
|
|
}
|
|
|
|
func (a *aclList) Head() *AclRecord {
|
|
return a.records[len(a.records)-1]
|
|
}
|
|
|
|
func (a *aclList) HasHead(head string) bool {
|
|
_, exists := a.indexes[head]
|
|
return exists
|
|
}
|
|
|
|
func (a *aclList) Get(id string) (*AclRecord, error) {
|
|
recIdx, ok := a.indexes[id]
|
|
if !ok {
|
|
return nil, ErrNoSuchRecord
|
|
}
|
|
return a.records[recIdx], nil
|
|
}
|
|
|
|
func (a *aclList) GetIndex(idx int) (*AclRecord, error) {
|
|
// TODO: when we add snapshots we will have to monitor record num in snapshots
|
|
if idx < 0 || idx >= len(a.records) {
|
|
return nil, ErrNoSuchRecord
|
|
}
|
|
return a.records[idx], nil
|
|
}
|
|
|
|
func (a *aclList) Iterate(iterFunc IterFunc) {
|
|
for _, rec := range a.records {
|
|
if !iterFunc(rec) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (a *aclList) RecordsAfter(ctx context.Context, id string) (records []*consensusproto.RawRecordWithId, err error) {
|
|
recIdx, ok := a.indexes[id]
|
|
if !ok {
|
|
return nil, ErrNoSuchRecord
|
|
}
|
|
for i := recIdx + 1; i < len(a.records); i++ {
|
|
rawRec, err := a.storage.GetRawRecord(ctx, a.records[i].Id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
records = append(records, rawRec)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *aclList) IterateFrom(startId string, iterFunc IterFunc) {
|
|
recIdx, ok := a.indexes[startId]
|
|
if !ok {
|
|
return
|
|
}
|
|
for i := recIdx; i < len(a.records); i++ {
|
|
if !iterFunc(a.records[i]) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (a *aclList) Close(ctx context.Context) (err error) {
|
|
return nil
|
|
}
|
|
|
|
func WrapAclRecord(rawRec *consensusproto.RawRecord) *consensusproto.RawRecordWithId {
|
|
payload, err := rawRec.Marshal()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
id, err := cidutil.NewCidFromBytes(payload)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return &consensusproto.RawRecordWithId{
|
|
Payload: payload,
|
|
Id: id,
|
|
}
|
|
}
|