-
Notifications
You must be signed in to change notification settings - Fork 3.7k
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
feat(indexer/base): schema and value validation #20665
Merged
Merged
Changes from 17 commits
Commits
Show all changes
52 commits
Select commit
Hold shift + click to select a range
76f6f32
feat: indexer base types
aaronc 63aeb85
WIP on tests
aaronc 216e8f8
update listener
aaronc b9fb6c9
Merge branch 'main' of github.com:cosmos/cosmos-sdk into aaronc/index…
aaronc 663ed17
rename column to field
aaronc 4311357
delete code, simplify
aaronc c52655a
add error return
aaronc 46669d3
remove ability to filter subscribed modules - this is a bit dangerous
aaronc 0a47c39
add docs about fields
aaronc 7fd604f
update table and entity language to object
aaronc 4a00094
rename to type
aaronc 0c7f529
add CHANGELOG.md
aaronc 408ddc4
add DecodableModule interface
aaronc 599e7cf
Merge branch 'main' of github.com:cosmos/cosmos-sdk into aaronc/index…
aaronc bc98756
make compatible with go 1.12
aaronc 68d0afc
remove CommitCatchupSync - catch-up design in flux, may be premature …
aaronc 50a8c37
restore validation code
aaronc dab02f7
Merge branch 'main' of github.com:cosmos/cosmos-sdk into aaronc/index…
aaronc 3606a04
update validation
aaronc 7cf9678
string tests
aaronc df3cde1
TestKindForGoValue
aaronc 7c7ff79
update validation
aaronc 2eb3ed2
WIP on TestField_Validate
aaronc 8e2db24
simplifications
aaronc 29d5a29
WIP updates
aaronc 36aa92d
updates
aaronc aa73bd2
updates
aaronc 160c186
update kind tests
aaronc 00dceff
good kind test coverage
aaronc ccb9ca4
field tests and docs
aaronc df727e2
enum tests
aaronc 842d420
object type tests
aaronc 4b18658
object type docs
aaronc 40b5d25
object type docs
aaronc 6ed4d2b
revert comment changes
aaronc 3ac7b0a
revert comment changes
aaronc 8f3391b
simplify tests
aaronc e94e34a
Merge branch 'main' into aaronc/indexer-base-validation
aaronc 1e1ffb7
refactor into standalone []Field validation methods
aaronc 68771be
Merge branch 'main' into aaronc/indexer-base-validation
aaronc d116e23
rename
aaronc a5520b8
Merge remote-tracking branch 'origin/aaronc/indexer-base-validation' …
aaronc ad3f07a
revert
aaronc 93fdacc
Merge branch 'main' of github.com:cosmos/cosmos-sdk into aaronc/index…
aaronc 0b6dfbd
add missing validation logic
aaronc 85e5114
add name validation
aaronc 450723c
add Decimal and Integer restrictions
aaronc 2ca77aa
Merge branch 'main' into aaronc/indexer-base-validation
aaronc 693f8fb
gofumpt
aaronc 9f4c1d0
Merge remote-tracking branch 'origin/aaronc/indexer-base-validation' …
aaronc 2fa1b7e
Merge branch 'main' into aaronc/indexer-base-validation
aaronc 6eaeaf9
Merge branch 'main' into aaronc/indexer-base-validation
aaronc File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
<!-- | ||
Guiding Principles: | ||
|
||
Changelogs are for humans, not machines. | ||
There should be an entry for every single version. | ||
The same types of changes should be grouped. | ||
Versions and sections should be linkable. | ||
The latest version comes first. | ||
The release date of each version is displayed. | ||
Mention whether you follow Semantic Versioning. | ||
|
||
Usage: | ||
|
||
Change log entries are to be added to the Unreleased section under the | ||
appropriate stanza (see below). Each entry should ideally include a tag and | ||
the Github issue reference in the following format: | ||
|
||
* (<tag>) \#<issue-number> message | ||
|
||
The issue numbers will later be link-ified during the release process so you do | ||
not have to worry about including a link manually, but you can if you wish. | ||
|
||
Types of changes (Stanzas): | ||
|
||
"Features" for new features. | ||
"Improvements" for changes in existing functionality. | ||
"Deprecated" for soon-to-be removed features. | ||
"Bug Fixes" for any bug fixes. | ||
"Client Breaking" for breaking Protobuf, gRPC and REST routes used by end-users. | ||
"CLI Breaking" for breaking CLI commands. | ||
"API Breaking" for breaking exported APIs used by developers building on SDK. | ||
Ref: https://keepachangelog.com/en/1.0.0/ | ||
--> | ||
|
||
# Changelog | ||
|
||
## [Unreleased] |
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,5 @@ | ||
# Indexer Base | ||
|
||
The indexer base module is designed to provide a stable, zero-dependency base layer for the built-in indexer functionality. Packages that integrate with the indexer should feel free to depend on this package without fear of any external dependencies being pulled in. | ||
|
||
The basic types for specifying index sources, targets and decoders are provided here along with a basic engine that ties these together. A package wishing to be an indexing source could accept an instance of `Engine` directly to be compatible with indexing. A package wishing to be a decoder can use the `Entity` and `Table` types. A package defining an indexing target should implement the `Indexer` interface. |
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,27 @@ | ||
package indexerbase | ||
|
||
// DecodableModule is an interface that modules can implement to provide a ModuleDecoder. | ||
// Usually these modules would also implement appmodule.AppModule, but that is not included | ||
// to keep this package free of any dependencies. | ||
type DecodableModule interface { | ||
|
||
// ModuleDecoder returns a ModuleDecoder for the module. | ||
ModuleDecoder() (ModuleDecoder, error) | ||
} | ||
|
||
// ModuleDecoder is a struct that contains the schema and a KVDecoder for a module. | ||
type ModuleDecoder struct { | ||
// Schema is the schema for the module. | ||
Schema ModuleSchema | ||
|
||
// KVDecoder is a function that decodes a key-value pair into an ObjectUpdate. | ||
// If modules pass logical updates directly to the engine and don't require logical decoding of raw bytes, | ||
// then this function should be nil. | ||
KVDecoder KVDecoder | ||
} | ||
|
||
// KVDecoder is a function that decodes a key-value pair into an ObjectUpdate. | ||
// If the KV-pair doesn't represent an object update, the function should return false | ||
// as the second return value. Error should only be non-nil when the decoder expected | ||
// to parse a valid update and was unable to. | ||
type KVDecoder = func(key, value []byte) (ObjectUpdate, bool, 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
package indexerbase | ||
|
||
// EnumDefinition represents the definition of an enum type. | ||
type EnumDefinition struct { | ||
// Name is the name of the enum type. | ||
Name string | ||
|
||
// Values is a list of distinct values that are part of the enum type. | ||
Values []string | ||
} |
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,142 @@ | ||
package indexerbase | ||
|
||
import "fmt" | ||
|
||
// Field represents a field in an object type. | ||
type Field struct { | ||
// Name is the name of the field. | ||
Name string | ||
|
||
// Kind is the basic type of the field. | ||
Kind Kind | ||
|
||
// Nullable indicates whether null values are accepted for the field. | ||
Nullable bool | ||
|
||
// AddressPrefix is the address prefix of the field's kind, currently only used for Bech32AddressKind. | ||
AddressPrefix string | ||
|
||
// EnumDefinition is the definition of the enum type and is only valid when Kind is EnumKind. | ||
EnumDefinition EnumDefinition | ||
} | ||
|
||
// Validate validates the field. | ||
func (c Field) Validate() error { | ||
// non-empty name | ||
if c.Name == "" { | ||
return fmt.Errorf("field name cannot be empty") | ||
} | ||
|
||
// valid kind | ||
if err := c.Kind.Validate(); err != nil { | ||
return fmt.Errorf("invalid field type for %q: %w", c.Name, err) | ||
} | ||
|
||
// address prefix only valid with Bech32AddressKind | ||
if c.Kind == Bech32AddressKind && c.AddressPrefix == "" { | ||
return fmt.Errorf("missing address prefix for field %q", c.Name) | ||
} else if c.Kind != Bech32AddressKind && c.AddressPrefix != "" { | ||
Check warning Code scanning / CodeQL Directly using the bech32 constants Warning
Directly using the bech32 constants instead of the configuration values
|
||
return fmt.Errorf("address prefix is only valid for field %q with type Bech32AddressKind", c.Name) | ||
} | ||
|
||
// enum definition only valid with EnumKind | ||
if c.Kind == EnumKind { | ||
if err := c.EnumDefinition.Validate(); err != nil { | ||
return fmt.Errorf("invalid enum definition for field %q: %w", c.Name, err) | ||
} | ||
} else if c.Kind != EnumKind && c.EnumDefinition.Name != "" && c.EnumDefinition.Values != nil { | ||
return fmt.Errorf("enum definition is only valid for field %q with type EnumKind", c.Name) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
// Validate validates the enum definition. | ||
func (e EnumDefinition) Validate() error { | ||
if e.Name == "" { | ||
return fmt.Errorf("enum definition name cannot be empty") | ||
} | ||
if len(e.Values) == 0 { | ||
return fmt.Errorf("enum definition values cannot be empty") | ||
} | ||
seen := make(map[string]bool, len(e.Values)) | ||
for i, v := range e.Values { | ||
if v == "" { | ||
return fmt.Errorf("enum definition value at index %d cannot be empty for enum %s", i, e.Name) | ||
} | ||
if seen[v] { | ||
return fmt.Errorf("duplicate enum definition value %q for enum %s", v, e.Name) | ||
} | ||
seen[v] = true | ||
} | ||
return nil | ||
} | ||
|
||
// ValidateValue validates that the value conforms to the field's kind and nullability. | ||
// It currently does not do any validation that IntegerKind, DecimalKind, Bech32AddressKind, or EnumKind | ||
// values are valid for their respective types behind conforming to the correct go type. | ||
func (c Field) ValidateValue(value any) error { | ||
if value == nil { | ||
if !c.Nullable { | ||
return fmt.Errorf("field %q cannot be null", c.Name) | ||
} | ||
return nil | ||
} | ||
return c.Kind.ValidateValueType(value) | ||
} | ||
|
||
// ValidateKey validates that the value conforms to the set of fields as a Key in an EntityUpdate. | ||
// See EntityUpdate.Key for documentation on the requirements of such values. | ||
func ValidateKey(fields []Field, value any) error { | ||
if len(fields) == 0 { | ||
return nil | ||
} | ||
|
||
if len(fields) == 1 { | ||
return fields[0].ValidateValue(value) | ||
} | ||
|
||
values, ok := value.([]any) | ||
if !ok { | ||
return fmt.Errorf("expected slice of values for key fields, got %T", value) | ||
} | ||
|
||
if len(fields) != len(values) { | ||
return fmt.Errorf("expected %d key fields, got %d values", len(fields), len(value.([]any))) | ||
} | ||
for i, field := range fields { | ||
if err := field.ValidateValue(values[i]); err != nil { | ||
return fmt.Errorf("invalid value for key field %q: %w", field.Name, err) | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
// ValidateValue validates that the value conforms to the set of fields as a Value in an EntityUpdate. | ||
// See EntityUpdate.Value for documentation on the requirements of such values. | ||
func ValidateValue(fields []Field, value any) error { | ||
valueUpdates, ok := value.(ValueUpdates) | ||
if ok { | ||
fieldMap := map[string]Field{} | ||
for _, field := range fields { | ||
fieldMap[field.Name] = field | ||
} | ||
var errs []error | ||
valueUpdates.Iterate(func(fieldName string, value any) bool { | ||
field, ok := fieldMap[fieldName] | ||
if !ok { | ||
errs = append(errs, fmt.Errorf("unknown field %q in value updates", fieldName)) | ||
} | ||
if err := field.ValidateValue(value); err != nil { | ||
errs = append(errs, fmt.Errorf("invalid value for field %q: %w", fieldName, err)) | ||
} | ||
return true | ||
}) | ||
|
||
if len(errs) > 0 { | ||
return fmt.Errorf("validation errors: %v", errs) | ||
} | ||
return nil | ||
} else { | ||
return ValidateKey(fields, value) | ||
} | ||
} |
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,7 @@ | ||
module cosmossdk.io/indexer/base | ||
|
||
// NOTE: this go.mod should have zero dependencies and remain on go 1.12 to stay compatible | ||
// with all known production releases of the Cosmos SDK. This is to ensure that all historical | ||
// apps could be patched to support indexing if desired. | ||
|
||
go 1.12 |
Empty file.
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Check warning
Code scanning / CodeQL
Directly using the bech32 constants Warning