-
Notifications
You must be signed in to change notification settings - Fork 103
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'master' into ryan/ecocredit-audit
- Loading branch information
Showing
37 changed files
with
736 additions
and
548 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
package orm_test | ||
|
||
import ( | ||
"pgregory.net/rapid" | ||
|
||
"github.com/regen-network/regen-ledger/orm/testdata" | ||
) | ||
|
||
// genGroupMember generates a new group member. At the moment it doesn't | ||
// generate empty strings for Group or Member. | ||
var genGroupMember = rapid.Custom(func(t *rapid.T) *testdata.GroupMember { | ||
return &testdata.GroupMember{ | ||
Group: []byte(rapid.StringN(1, 100, 150).Draw(t, "group").(string)), | ||
Member: []byte(rapid.StringN(1, 100, 150).Draw(t, "member").(string)), | ||
Weight: rapid.Uint64().Draw(t, "weight").(uint64), | ||
} | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,88 +1,15 @@ | ||
package orm | ||
|
||
import ( | ||
"reflect" | ||
|
||
"github.com/cosmos/cosmos-sdk/store/prefix" | ||
"github.com/cosmos/cosmos-sdk/types/errors" | ||
) | ||
|
||
// TableExportable | ||
type TableExportable interface { | ||
// Table returns the table to export | ||
Table() table | ||
} | ||
|
||
// SequenceExportable | ||
type SequenceExportable interface { | ||
// Sequence returns the sequence to export | ||
Sequence() Sequence | ||
} | ||
|
||
// ExportTableData iterates over the given table entries and stores them at the passed ModelSlicePtr. | ||
// When the given table implements the `SequenceExportable` interface then it's current value | ||
// is returned as well or otherwise defaults to 0. | ||
func ExportTableData(ctx HasKVStore, t TableExportable, dest ModelSlicePtr) (uint64, error) { | ||
it, err := t.Table().PrefixScan(ctx, nil, nil) | ||
if err != nil { | ||
return 0, errors.Wrap(err, "table PrefixScan failure when exporting table data") | ||
} | ||
_, err = ReadAll(it, dest) | ||
if err != nil { | ||
return 0, err | ||
} | ||
var seqValue uint64 | ||
if st, ok := t.(SequenceExportable); ok { | ||
seqValue = st.Sequence().CurVal(ctx) | ||
} | ||
return seqValue, err | ||
} | ||
|
||
// ImportTableData initializes a table and attaches indexers from the given data interface{}. | ||
// data should be a slice of structs that implement PrimaryKeyed (eg []*GroupInfo). | ||
// The seqValue is optional and only used with tables that implement the `SequenceExportable` interface. | ||
func ImportTableData(ctx HasKVStore, t TableExportable, data interface{}, seqValue uint64) error { | ||
table := t.Table() | ||
if err := clearAllInTable(ctx, table); err != nil { | ||
return errors.Wrap(err, "clear old entries") | ||
} | ||
|
||
if st, ok := t.(SequenceExportable); ok { | ||
if err := st.Sequence().InitVal(ctx, seqValue); err != nil { | ||
return errors.Wrap(err, "sequence") | ||
} | ||
} | ||
|
||
// Provided data must be a slice | ||
modelSlice := reflect.ValueOf(data) | ||
if modelSlice.Kind() != reflect.Slice { | ||
return errors.Wrap(ErrArgument, "data must be a slice") | ||
} | ||
|
||
// Create table entries | ||
for i := 0; i < modelSlice.Len(); i++ { | ||
obj, ok := modelSlice.Index(i).Interface().(PrimaryKeyed) | ||
if !ok { | ||
return errors.Wrapf(ErrArgument, "unsupported type :%s", reflect.TypeOf(data).Elem().Elem()) | ||
} | ||
err := table.Create(ctx, PrimaryKey(obj), obj) | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
|
||
return nil | ||
} | ||
|
||
// clearAllInTable deletes all entries in a table with delete interceptors called | ||
func clearAllInTable(ctx HasKVStore, table table) error { | ||
store := prefix.NewStore(ctx.KVStore(table.storeKey), []byte{table.prefix}) | ||
it := store.Iterator(nil, nil) | ||
defer it.Close() | ||
for ; it.Valid(); it.Next() { | ||
if err := table.Delete(ctx, it.Key()); err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
// Export stores all the values in the table in the passed | ||
// ModelSlicePtr. If the table has an associated sequence, then its | ||
// current value is returned, otherwise 0 is returned by default. | ||
Export(HasKVStore, ModelSlicePtr) (uint64, error) | ||
|
||
// Import clears the table and initializes it from the given data | ||
// interface{}. data should be a slice of structs that implement | ||
// PrimaryKeyed (eg []*GroupInfo). The seqValue is optional and only | ||
// used with tables that have an associated sequence. | ||
Import(HasKVStore, interface{}, uint64) error | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
package orm_test | ||
|
||
import ( | ||
"testing" | ||
|
||
"pgregory.net/rapid" | ||
|
||
"github.com/regen-network/regen-ledger/orm" | ||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestPrefixRangeProperty(t *testing.T) { | ||
t.Run("TestPrefixRange", rapid.MakeCheck(func(t *rapid.T) { | ||
prefix := rapid.SliceOf(rapid.Byte()).Draw(t, "prefix").([]byte) | ||
|
||
start, end := orm.PrefixRange(prefix) | ||
|
||
// len(prefix) == 0 => start == nil && end == nil | ||
if len(prefix) == 0 { | ||
require.Nil(t, start) | ||
require.Nil(t, end) | ||
} else { | ||
// start == prefix | ||
require.Equal(t, prefix, start) | ||
|
||
// Would overflow if all bytes are 255 | ||
wouldOverflow := true | ||
for _, b := range prefix { | ||
if b != 255 { | ||
wouldOverflow = false | ||
} | ||
} | ||
|
||
// Overflow => end == nil | ||
if wouldOverflow { | ||
require.Nil(t, end) | ||
} else { | ||
require.Equal(t, len(start), len(end)) | ||
|
||
// Scan back and find last value that isn't 255 | ||
overflowIndex := len(start) - 1 | ||
for overflowIndex > 0 && prefix[overflowIndex] == 255 { | ||
overflowIndex-- | ||
} | ||
|
||
// bytes should be the same up to overflow | ||
// index, one greater at overflow and 0 from | ||
// then on | ||
for i, b := range start { | ||
if i < overflowIndex { | ||
require.Equal(t, b, end[i]) | ||
} else if i == overflowIndex { | ||
require.Equal(t, b+1, end[i]) | ||
} else { | ||
require.Equal(t, uint8(0), end[i]) | ||
} | ||
} | ||
} | ||
} | ||
})) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.