Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Index selectors #96

Merged
merged 45 commits into from
Apr 11, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
3771e81
Query.Intersect API & test
marino39 Mar 20, 2023
fa6c2ef
Naive implementation of Query.Intersects
marino39 Mar 20, 2023
a220743
Improve Query.Intersect benchmark
marino39 Mar 20, 2023
8968be7
Query.Intersect add sort / filter / offset / limit
marino39 Mar 20, 2023
c14e8ea
Index.Iter implementation
marino39 Mar 21, 2023
4dac79b
Use Index.Iter in Table Scan functions
marino39 Mar 21, 2023
490f432
Extract index update functions from Table to Index
marino39 Mar 21, 2023
e5db7e4
Extract index update functions from Table to Index: p2
marino39 Mar 21, 2023
13e42f5
KeySuccessor args change
marino39 Mar 21, 2023
a4bd17a
Fix: allocations is Table.PrimaryKey
marino39 Mar 21, 2023
ac937b9
Optimize Index.OnUpdate, so it requires less writes to database
marino39 Mar 22, 2023
3a4ee78
Use new Index callbacks in Table.reindex func
marino39 Mar 22, 2023
7a50e94
Add Intersect function to Index and use it in Query
marino39 Mar 22, 2023
80b2d1d
Benchmark Query.Intersect
marino39 Mar 23, 2023
e16bdc8
Move Index.Intersects to be map based because of poor performance
marino39 Mar 23, 2023
d8aa922
Reduce number of allocations in Index.Intersect
marino39 Mar 23, 2023
11e01d1
Add Query.Intersect().Order() test
marino39 Mar 23, 2023
fd5ce51
Query.Intersect().After() implementation
marino39 Mar 23, 2023
365ad1c
Query.After now works with Query.Order and Query.Intersect
marino39 Mar 24, 2023
323c907
Make sure that intersect queries do not have any modifiers like limit…
marino39 Mar 27, 2023
d39c691
Index selectors API proposition
marino39 Mar 27, 2023
d98773b
Merge branch 'master' into new_index_selectors
marino39 Mar 29, 2023
6f43140
Selector struct impl
marino39 Mar 29, 2023
8aa787f
Selector interface based approach
marino39 Mar 30, 2023
2790aa8
Table / Index.Intersect ported to use Selector
marino39 Mar 30, 2023
6afb4a8
Add table utils unit test
marino39 Mar 30, 2023
58fe15e
Use Selector in Query
marino39 Mar 30, 2023
863df97
Add files to .gitignore
marino39 Mar 30, 2023
cda655e
Fix: Example app
marino39 Mar 30, 2023
80b0f91
Cleanup
marino39 Mar 30, 2023
f63efaf
Table.Get SelectorRange implementation
marino39 Mar 31, 2023
0ca9793
Table.Get SelectorPoints implementation
marino39 Mar 31, 2023
a002c93
Table.Get SelectorPoints implementation -- add one more benchmark
marino39 Mar 31, 2023
9abce7e
Table.Get SelectorRanges implementation
marino39 Apr 4, 2023
0781cc7
Index.Iter unit tests
marino39 Apr 5, 2023
cdcb795
Index.Iter unit tests - PrimaryIndex
marino39 Apr 5, 2023
f38b78a
Index.Iter unit tests - new concept of Point Iter for primary index &…
marino39 Apr 5, 2023
a319514
Index.Iter restore SelectorPoints impl & unit tests
marino39 Apr 6, 2023
3f29335
Index OnInsert/OnUpdate/OnDelete unit tests
marino39 Apr 6, 2023
e8b19d2
Fix: Index.OnUpdate not properly updating indexes
marino39 Apr 6, 2023
40d552c
Update README.md
marino39 Apr 7, 2023
4e25ee3
Update comment in selector.go for SelectorRange and SelectorRanges
marino39 Apr 7, 2023
699851e
Fix: Issue with Query, SelectorRanges and Query.After
marino39 Apr 7, 2023
3bde6d3
Remove no longer valid test case
marino39 Apr 7, 2023
f8a2ab1
Query tests with different selectors and Query.After
marino39 Apr 11, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Merge branch 'master' into new_index_selectors
  • Loading branch information
marino39 committed Mar 29, 2023
commit d98773b8c8d270d040d34d92f39e6d4f8c6f5d22
146 changes: 121 additions & 25 deletions _benchmarks/suites/table_query_intersect_suite.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,11 +50,13 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
})

const (
_ = bond.PrimaryIndexID
TokenBalanceAccountAddressIndexID = iota
TokenBalanceContractAddressIndexID = iota
TokenBalanceAccountAddressOrderBalanceDESCIndexID = iota
TokenBalanceAccountAndContractAddressIndexID = iota
_ = bond.PrimaryIndexID
TokenBalanceAccountAddressIndexID = iota
TokenBalanceContractAddressIndexID = iota
TokenBalanceContractAddressAccountAddressTokenIDIndexID = iota
TokenBalanceContractAddressTokenIDIndexID = iota
TokenBalanceAccountAddressOrderBalanceDESCIndexID = iota
TokenBalanceAccountAndContractAddressIndexID = iota
)

var (
Expand All @@ -70,7 +72,29 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
IndexID: TokenBalanceContractAddressIndexID,
IndexName: "contract_address_idx",
IndexKeyFunc: func(builder bond.KeyBuilder, tb *TokenBalance) []byte {
return builder.AddStringField(tb.ContractAddress).Bytes()
return builder.AddStringField(tb.ContractAddress).
AddUint32Field(tb.TokenID).
Bytes()
},
IndexOrderFunc: bond.IndexOrderDefault[*TokenBalance],
})
TokenBalanceContractAddressTokenIDIndex = bond.NewIndex[*TokenBalance](bond.IndexOptions[*TokenBalance]{
IndexID: TokenBalanceContractAddressTokenIDIndexID,
IndexName: "contract_address_token_id_idx",
IndexKeyFunc: func(builder bond.KeyBuilder, tb *TokenBalance) []byte {
return builder.AddStringField(tb.ContractAddress).
AddUint32Field(tb.TokenID).
Bytes()
},
IndexOrderFunc: bond.IndexOrderDefault[*TokenBalance],
})
TokenBalanceContractAddressAccountAddressTokenIDIndex = bond.NewIndex[*TokenBalance](bond.IndexOptions[*TokenBalance]{
IndexID: TokenBalanceContractAddressAccountAddressTokenIDIndexID,
IndexName: "contract_address_account_address_token_id_idx",
IndexKeyFunc: func(builder bond.KeyBuilder, tb *TokenBalance) []byte {
return builder.AddStringField(tb.ContractAddress).
AddUint32Field(tb.TokenID).
Bytes()
},
IndexOrderFunc: bond.IndexOrderDefault[*TokenBalance],
})
Expand Down Expand Up @@ -100,6 +124,8 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
err := tokenBalanceTable.AddIndex([]*bond.Index[*TokenBalance]{
TokenBalanceAccountAddressIndex,
TokenBalanceContractAddressIndex,
TokenBalanceContractAddressTokenIDIndex,
TokenBalanceContractAddressAccountAddressTokenIDIndex,
TokenBalanceAccountAddressOrderBalanceDESCIndex,
TokenBalanceAccountAndContractAddressIndex,
})
Expand All @@ -112,9 +138,9 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
tokenBalances = append(tokenBalances, &TokenBalance{
ID: uint64(i + 1),
AccountID: 0,
ContractAddress: fmt.Sprintf("0xtestContract%d", i%10),
AccountAddress: fmt.Sprintf("0xtestAccount%d", i%1000),
TokenID: uint32(i),
ContractAddress: fmt.Sprintf("0xtestContract%d", i%1),
AccountAddress: fmt.Sprintf("0xtestAccount%d", i%10000),
TokenID: uint32(i % 10),
Balance: uint64((i % 100) * 10),
})
}
Expand All @@ -126,23 +152,25 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma

tokenBalances = nil

filter := func(tb *TokenBalance) bool {
return tb.AccountAddress == "0xtestAccount0" && tb.TokenID == 0
}

var queryInputs = []struct {
index *bond.Index[*TokenBalance]
index2 *bond.Index[*TokenBalance]
indexName string
selector *TokenBalance
selector2 *TokenBalance
indexName string
filter func(tb *TokenBalance) bool
order func(tb *TokenBalance, tb2 *TokenBalance) bool
offset int
limit int
}{
// AccountAddress Index
{index: TokenBalanceAccountAndContractAddressIndex, indexName: "AccountAndContractAddress", selector: &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}, offset: 0, limit: 0},
{index: TokenBalanceAccountAndContractAddressIndex, indexName: "AccountAndContractAddress", selector: &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}, offset: 0, limit: 500},
{index: TokenBalanceAccountAndContractAddressIndex, indexName: "AccountAndContractAddress", selector: &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}, offset: 0, limit: 1000},
{index: TokenBalanceAccountAndContractAddressIndex, indexName: "AccountAndContractAddress", selector: &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}, offset: 0, limit: 5000},
{index: TokenBalanceAccountAndContractAddressIndex, indexName: "AccountAndContractAddress", selector: &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}, offset: 0, limit: 10000},
{index: TokenBalanceContractAddressIndex, indexName: "TokenBalanceContractAddressIndex", filter: filter, selector: &TokenBalance{ContractAddress: "0xtestContract0"}, offset: 0, limit: 0},
{index: TokenBalanceContractAddressIndex, indexName: "TokenBalanceContractAddressIndex", filter: filter, selector: &TokenBalance{ContractAddress: "0xtestContract0"}, offset: 0, limit: 500},
{index: TokenBalanceContractAddressIndex, indexName: "TokenBalanceContractAddressIndex", filter: filter, selector: &TokenBalance{ContractAddress: "0xtestContract0"}, offset: 0, limit: 1000},
{index: TokenBalanceContractAddressIndex, indexName: "TokenBalanceContractAddressIndex", filter: filter, selector: &TokenBalance{ContractAddress: "0xtestContract0"}, offset: 0, limit: 5000},
{index: TokenBalanceContractAddressIndex, indexName: "TokenBalanceContractAddressIndex", filter: filter, selector: &TokenBalance{ContractAddress: "0xtestContract0"}, offset: 0, limit: 10000},
}

for _, v := range queryInputs {
Expand All @@ -153,15 +181,15 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma

results = append(results,
bs.Benchmark(bench.Benchmark{
Name: fmt.Sprintf("%s/%s/Query_Index_No_Intersect_%s_Sel_%d_Offset_%d_Limit_%d",
Name: fmt.Sprintf("%s/%s/Query_Index_Filter_%s_Sel_%d_Offset_%d_Limit_%d",
bs.Name, serializer.Name, v.indexName, selectorID, v.offset, v.limit),
Inputs: v,
BenchmarkFunc: QueryIntersectWithOpts(tokenBalanceTable, v.index, v.selector, v.index2, v.selector2, v.offset, v.limit),
BenchmarkFunc: QueryFilterWithOpts(tokenBalanceTable, v.index, v.selector, v.filter, v.offset, v.limit),
}),
)
}

selector := &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0"}
selector := &TokenBalance{AccountAddress: "0xtestAccount0", ContractAddress: "0xtestContract0", TokenID: 0}

var queryInputs2 = []struct {
index *bond.Index[*TokenBalance]
Expand All @@ -175,11 +203,11 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
limit int
}{
// AccountAddress Index
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressIndex, indexName: "AccountAddress_And_ContractAddress", selector: selector, selector2: selector, offset: 0, limit: 0},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressIndex, indexName: "AccountAddress_And_ContractAddress", selector: selector, selector2: selector, offset: 0, limit: 500},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressIndex, indexName: "AccountAddress_And_ContractAddress", selector: selector, selector2: selector, offset: 0, limit: 1000},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressIndex, indexName: "AccountAddress_And_ContractAddress", selector: selector, selector2: selector, offset: 0, limit: 5000},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressIndex, indexName: "AccountAddress_And_ContractAddress", selector: selector, selector2: selector, offset: 0, limit: 10000},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressTokenIDIndex, indexName: "AccountAddress_And_ContractAddress_TokenID", selector: selector, selector2: selector, offset: 0, limit: 0},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressTokenIDIndex, indexName: "AccountAddress_And_ContractAddress_TokenID", selector: selector, selector2: selector, offset: 0, limit: 500},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressTokenIDIndex, indexName: "AccountAddress_And_ContractAddress_TokenID", selector: selector, selector2: selector, offset: 0, limit: 1000},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressTokenIDIndex, indexName: "AccountAddress_And_ContractAddress_TokenID", selector: selector, selector2: selector, offset: 0, limit: 5000},
{index: TokenBalanceAccountAddressIndex, index2: TokenBalanceContractAddressTokenIDIndex, indexName: "AccountAddress_And_ContractAddress_TokenID", selector: selector, selector2: selector, offset: 0, limit: 10000},
}

for _, v := range queryInputs2 {
Expand All @@ -198,6 +226,41 @@ func BenchmarkTableQueryIntersectSuite(bs *bench.BenchmarkSuite) []bench.Benchma
)
}

var queryInputs3 = []struct {
index *bond.Index[*TokenBalance]
index2 *bond.Index[*TokenBalance]
indexName string
selector *TokenBalance
selector2 *TokenBalance
filter func(tb *TokenBalance) bool
order func(tb *TokenBalance, tb2 *TokenBalance) bool
offset int
limit int
}{
// AccountAddress Index
{index: TokenBalanceContractAddressAccountAddressTokenIDIndex, indexName: "TokenBalanceContractAddressAccountAddressTokenID", selector: selector, offset: 0, limit: 0},
{index: TokenBalanceContractAddressAccountAddressTokenIDIndex, indexName: "TokenBalanceContractAddressAccountAddressTokenID", selector: selector, offset: 0, limit: 500},
{index: TokenBalanceContractAddressAccountAddressTokenIDIndex, indexName: "TokenBalanceContractAddressAccountAddressTokenID", selector: selector, offset: 0, limit: 1000},
{index: TokenBalanceContractAddressAccountAddressTokenIDIndex, indexName: "TokenBalanceContractAddressAccountAddressTokenID", selector: selector, offset: 0, limit: 5000},
{index: TokenBalanceContractAddressAccountAddressTokenIDIndex, indexName: "TokenBalanceContractAddressAccountAddressTokenID", selector: selector, offset: 0, limit: 10000},
}

for _, v := range queryInputs3 {
var selectorID = uint64(0)
if v.selector != nil {
selectorID = v.selector.ID
}

results = append(results,
bs.Benchmark(bench.Benchmark{
Name: fmt.Sprintf("%s/%s/Query_Index_Direct_%s_Sel_%d_Offset_%d_Limit_%d",
bs.Name, serializer.Name, v.indexName, selectorID, v.offset, v.limit),
Inputs: v,
BenchmarkFunc: QueryIntersectWithOpts(tokenBalanceTable, v.index, v.selector, v.index2, v.selector2, v.offset, v.limit),
}),
)
}

tearDownDatabase(db)
}

Expand Down Expand Up @@ -239,3 +302,36 @@ func QueryIntersectWithOpts(tbt bond.Table[*TokenBalance], idx *bond.Index[*Toke
}
}
}

func QueryFilterWithOpts(tbt bond.Table[*TokenBalance], idx *bond.Index[*TokenBalance], sel *TokenBalance, filter func(tb *TokenBalance) bool, offset int, limit int) func(b *testing.B) {
return func(b *testing.B) {
b.ReportAllocs()

for i := 0; i < b.N; i++ {
var tokenBalances []*TokenBalance

q := tbt.Query().Filter(filter)
if idx != nil && sel != nil {
q = q.With(idx, sel)
}

if offset != 0 {
q = q.Offset(uint64(offset))
}

if limit != 0 {
q = q.Limit(uint64(limit))
}

err := q.Execute(context.Background(), &tokenBalances)
if err != nil {
panic(err)
}

if len(tokenBalances) == 0 {
fmt.Printf("failed with patams %d %d\n", offset, limit)
panic("no results")
}
}
}
}
11 changes: 9 additions & 2 deletions index.go
Original file line number Diff line number Diff line change
Expand Up @@ -194,22 +194,29 @@ func (idx *Index[T]) Name() string {
}

func (idx *Index[T]) Iter(table Table[T], sel T, optBatch ...Batch) Iterator {
lowerBound := encodeIndexKey(table, sel, idx, make([]byte, 0, 1024))
upperBound := keySuccessor(lowerBound[0:_KeyPrefixSplitIndex(lowerBound)], make([]byte, 0, 1024))
lowerBound := encodeIndexKey(table, sel, idx, table.DB().getKeyBufferPool().Get()[:0])
upperBound := keySuccessor(lowerBound[0:_KeyPrefixSplitIndex(lowerBound)], table.DB().getKeyBufferPool().Get()[:0])

releaseBuffers := func() {
table.DB().getKeyBufferPool().Put(lowerBound[:0])
table.DB().getKeyBufferPool().Put(upperBound[:0])
}

if len(optBatch) > 0 {
return optBatch[0].Iter(&IterOptions{
IterOptions: pebble.IterOptions{
LowerBound: lowerBound,
UpperBound: upperBound,
},
releaseBufferOnClose: releaseBuffers,
})
} else {
return table.DB().Iter(&IterOptions{
IterOptions: pebble.IterOptions{
LowerBound: lowerBound,
UpperBound: upperBound,
},
releaseBufferOnClose: releaseBuffers,
})
}
}
Expand Down
You are viewing a condensed version of this merge commit. You can view the full changes here.