From 37fc03860549c3f122c3acffabbb2aaa4e7dc034 Mon Sep 17 00:00:00 2001 From: Ian Norden Date: Wed, 13 Mar 2019 11:14:35 -0500 Subject: [PATCH] fixes for review comments --- README.md | 54 ++- cmd/compose.go | 7 +- cmd/composeAndExecute.go | 19 +- cmd/contractWatcher.go | 2 +- cmd/execute.go | 20 +- environments/example.toml | 10 +- .../contract_watcher_full_transformer_test.go | 47 +- ...contract_watcher_light_transformer_test.go | 56 ++- ...transformer.go => contract_transformer.go} | 4 +- ...generic_watcher.go => contract_watcher.go} | 24 +- pkg/config/contract.go | 163 ++++--- pkg/config/plugin.go | 28 +- .../full/converter/converter.go | 14 +- .../full/converter/converter_test.go | 12 +- .../full/transformer/transformer.go | 22 +- .../full/transformer/transformer_test.go | 333 +++---------- .../light/converter/converter.go | 16 +- .../light/converter/converter_test.go | 15 +- .../repository/header_repository_test.go | 110 ++--- .../light/retriever/retriever_suite_test.go | 2 +- .../light/transformer/transformer.go | 48 +- .../light/transformer/transformer_test.go | 448 ++---------------- .../shared/contract/contract.go | 2 +- .../helpers/test_helpers/mocks/entities.go | 58 +-- .../shared/helpers/test_helpers/test_data.go | 109 +++++ .../repository/event_repository_test.go | 6 +- .../retriever/address_retriever_test.go | 3 +- pkg/contract_watcher/shared/types/event.go | 2 +- pkg/contract_watcher/shared/types/mode.go | 4 +- pkg/fakes/mock_parser.go | 5 +- pkg/plugin/writer/writer.go | 8 +- 31 files changed, 634 insertions(+), 1017 deletions(-) rename libraries/shared/transformer/{generic_transformer.go => contract_transformer.go} (87%) rename libraries/shared/watcher/{generic_watcher.go => contract_watcher.go} (71%) create mode 100644 pkg/contract_watcher/shared/helpers/test_helpers/test_data.go diff --git a/README.md b/README.md index b2546848..e9787b6c 100644 --- a/README.md +++ b/README.md @@ -156,9 +156,17 @@ and a transformer is an interface for filtering through that raw Ethereum data t ### contractWatcher The `contractWatcher` command is a built-in generic contract watcher. It can watch any and all events for a given contract provided the contract's ABI is available. -It also provides some state variable coverage by automating polling of public methods, with some restrictions. +It also provides some state variable coverage by automating polling of public methods, with some restrictions: +1. The method must have 2 or less arguments +2. The method's arguments must all be of type address or bytes32 (hash) +3. The method must return a single value -This command requires a pre-synced (full or light) vulcanizeDB (see above sections) and currently requires the contract ABI be available on etherscan or provided by the user. +This command operates in two modes- `light` and `full`- which require a light or full-synced vulcanizeDB, respectively. + +This command requires the contract ABI be available on Etherscan if it is not provided in the config file by the user. + +If method polling is turned on we require an archival node at the ETH ipc endpoint in our config, whether or not we are operating in `light` or `full` mode. +Otherwise, when operating in `light` mode, we only need to connect to a full node to fetch event logs. This command takes a config of the form: @@ -240,10 +248,26 @@ The 'method' and 'event' identifiers are tacked onto the end of the table names Example: -Running `./vulcanizedb contractWatcher --config --starting-block-number=5197514 --contract-address=0x8dd5fbce2f6a956c3022ba3663759011dd51e73e --events=Transfer --events=Mint --methods=balanceOf` -watches Transfer and Mint events of the TrueUSD contract and polls its balanceOf method using the addresses we find emitted from those events +Running `./vulcanizedb contractWatcher --config=./environments/example.toml --mode=light` -It produces and populates a schema with three tables: +Runs our contract watcher in light mode, configured to watch the contracts specified in the config file. Note that +by default we operate in `light` mode but the flag is included here to demonstrate its use. + +The example config we link to in this example watches two contracts, the ENS Registry (0x314159265dD8dbb310642f98f50C066173C1259b) and TrueUSD (0x8dd5fbCe2F6a956C3022bA3663759011Dd51e73E). + +Because the ENS Registry is configured with only an ABI and a starting block, we will watch all events for this contract and poll none of its methods. Note that the ENS Registry is an example +of a contract which does not have its ABI available over Etherscan and must have it included in the config file. + +The TrueUSD contract is configured with two events (`Transfer` and `Mint`) and a single method (`balanceOf`), as such it will watch these two events and use any addresses it collects emitted from them +to poll the `balanceOf` method with those addresses at every block. Note that we do not provide an ABI for TrueUSD as its ABI can be fetched from Etherscan. + +For the ENS contract, it produces and populates a schema with four tables" +`light_0x314159265dd8dbb310642f98f50c066173c1259b.newowner_event` +`light_0x314159265dd8dbb310642f98f50c066173c1259b.newresolver_event` +`light_0x314159265dd8dbb310642f98f50c066173c1259b.newttl_event` +`light_0x314159265dd8dbb310642f98f50c066173c1259b.transfer_event` + +For the TrusUSD contract, it produces and populates a schema with three tables: `light_0x8dd5fbce2f6a956c3022ba3663759011dd51e73e.transfer_event` `light_0x8dd5fbce2f6a956c3022ba3663759011dd51e73e.mint_event` @@ -276,7 +300,9 @@ Table "light_0x8dd5fbce2f6a956c3022ba3663759011dd51e73e.balanceof_method" | who_ | character varying(66) | | not null | | extended | | | | returned | numeric | | not null | | main | | | -The addition of '_' after table names is to prevent collisions with reserved Postgres words +The addition of '_' after table names is to prevent collisions with reserved Postgres words. + +Also notice that the contract address used for the schema name has been down-cased. ### composeAndExecute The `composeAndExecute` command is used to compose and execute over an arbitrary set of custom transformers. @@ -359,9 +385,9 @@ The config provides information for composing a set of transformers: that fetches state and storage diffs from an ETH node (instead of, for example, from IPFS) - `eth_event` indicates the transformer works with the [event watcher](https://github.com/vulcanize/maker-vulcanizedb/blob/staging/libraries/shared/watcher/event_watcher.go) that fetches event logs from an ETH node - - `eth_generic` indicates the transformer works with the [generic watcher](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/watcher/generic_watcher.go) - that is made to work with [omni pkg](https://github.com/vulcanize/maker-vulcanizedb/tree/staging/pkg/omni) - based transformers which work with either a light or full sync vDB to watch events and poll public methods + - `eth_contract` indicates the transformer works with the [contract watcher](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/watcher/generic_watcher.go) + that is made to work with [contract_watcher pkg](https://github.com/vulcanize/maker-vulcanizedb/tree/staging/pkg/omni) + based transformers which work with either a light or full sync vDB to watch events and poll public methods ([example](https://github.com/vulcanize/ens_transformers/blob/working/transformers/domain_records/transformer.go)) - `migrations` is the relative path from `repository` to the db migrations directory for the transformer - `rank` determines the order that migrations are ran, with lower ranked migrations running first - this is to help isolate any potential conflicts between transformer migrations @@ -393,13 +419,13 @@ type exporter string var Exporter exporter -func (e exporter) Export() []interface1.EventTransformerInitializer, []interface1.StorageTransformerInitializer, []interface1.GenericTransformerInitializer { +func (e exporter) Export() []interface1.EventTransformerInitializer, []interface1.StorageTransformerInitializer, []interface1.ContractTransformerInitializer { return []interface1.TransformerInitializer{ transformer1.TransformerInitializer, transformer3.TransformerInitializer, }, []interface1.StorageTransformerInitializer{ transformer4.StorageTransformerInitializer, - }, []interface1.GenericTransformerInitializer{ + }, []interface1.ContractTransformerInitializer{ transformer2.TransformerInitializer, } } @@ -409,12 +435,12 @@ func (e exporter) Export() []interface1.EventTransformerInitializer, []interface To plug in an external transformer we need to: * Create a [package](https://github.com/vulcanize/ens_transformers/blob/working/transformers/registry/new_owner/initializer/initializer.go) -that exports a variable `TransformerInitializer`, `StorageTransformerInitializer`, or `GenericTransformerInitializer` that are of type [TransformerInitializer](https://github.com/vulcanize/maker-vulcanizedb/blob/compose_and_execute/libraries/shared/transformer/event_transformer.go#L33) +that exports a variable `TransformerInitializer`, `StorageTransformerInitializer`, or `ContractTransformerInitializer` that are of type [TransformerInitializer](https://github.com/vulcanize/maker-vulcanizedb/blob/compose_and_execute/libraries/shared/transformer/event_transformer.go#L33) or [StorageTransformerInitializer](https://github.com/vulcanize/maker-vulcanizedb/blob/compose_and_execute/libraries/shared/transformer/storage_transformer.go#L31), -or [GenericTransformerInitializer](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/transformer/generic_transformer.go#L31), respectively +or [ContractTransformerInitializer](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/transformer/contract_transformer.go#L31), respectively * Design the transformers to work in the context of their [event](https://github.com/vulcanize/maker-vulcanizedb/blob/compose_and_execute/libraries/shared/watcher/event_watcher.go#L83), [storage](https://github.com/vulcanize/maker-vulcanizedb/blob/compose_and_execute/libraries/shared/watcher/storage_watcher.go#L53), -or [generic](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/watcher/generic_watcher.go#L68) watcher execution modes +or [contract](https://github.com/vulcanize/maker-vulcanizedb/blob/omni_update/libraries/shared/watcher/contract_watcher.go#L68) watcher execution modes * Create db migrations to run against vulcanizeDB so that we can store the transformer output * Do not `goose fix` the transformer migrations * Specify migration locations for each transformer in the config with the `exporter.transformer.migrations` fields diff --git a/cmd/compose.go b/cmd/compose.go index 93b51af9..28de4af1 100644 --- a/cmd/compose.go +++ b/cmd/compose.go @@ -62,7 +62,7 @@ var composeCmd = &cobra.Command{ rank = "0" [exporter.transformer2] path = "path/to/transformer2" - type = "eth_generic" + type = "eth_contract" repository = "github.com/account/repo" migrations = "db/migrations" rank = "0" @@ -91,8 +91,9 @@ from it and loaded into and executed over by the appropriate watcher. The type of watcher that the transformer works with is specified using the type variable for each transformer in the config. Currently there are watchers of event data from an eth node (eth_event) and storage data from an eth node -(eth_storage), and a more generic interface for accepting omni pkg based transformers -which can perform both event watching and public method polling. +(eth_storage), and a more generic interface for accepting contract_watcher pkg +based transformers which can perform both event watching and public method +polling (eth_contract). Transformers of different types can be ran together in the same command using a single config file or in separate command instances using different config files diff --git a/cmd/composeAndExecute.go b/cmd/composeAndExecute.go index b74cf280..c6aa318b 100644 --- a/cmd/composeAndExecute.go +++ b/cmd/composeAndExecute.go @@ -62,7 +62,7 @@ var composeAndExecuteCmd = &cobra.Command{ rank = "0" [exporter.transformer2] path = "path/to/transformer2" - type = "eth_generic" + type = "eth_contract" repository = "github.com/account/repo" migrations = "db/migrations" rank = "2" @@ -91,8 +91,9 @@ from it and loaded into and executed over by the appropriate watcher. The type of watcher that the transformer works with is specified using the type variable for each transformer in the config. Currently there are watchers of event data from an eth node (eth_event) and storage data from an eth node -(eth_storage), and a more generic interface for accepting omni pkg based transformers -which can perform both event watching and public method polling. +(eth_storage), and a more generic interface for accepting contract_watcher pkg +based transformers which can perform both event watching and public method +polling (eth_contract). Transformers of different types can be ran together in the same command using a single config file or in separate command instances using different config files @@ -150,8 +151,8 @@ func composeAndExecute() { os.Exit(1) } - // Use the Exporters export method to load the EventTransformerInitializer and StorageTransformerInitializer sets - ethEventInitializers, ethStorageInitializers, genericInitializers := exporter.Export() + // Use the Exporters export method to load the EventTransformerInitializer, StorageTransformerInitializer, and ContractTransformerInitializer sets + ethEventInitializers, ethStorageInitializers, ethContractInitializers := exporter.Export() // Setup bc and db objects blockChain := getBlockChain() @@ -175,11 +176,11 @@ func composeAndExecute() { go watchEthStorage(&sw, &wg) } - if len(genericInitializers) > 0 { - gw := watcher.NewGenericWatcher(&db, blockChain) - gw.AddTransformers(genericInitializers) + if len(ethContractInitializers) > 0 { + gw := watcher.NewContractWatcher(&db, blockChain) + gw.AddTransformers(ethContractInitializers) wg.Add(1) - go genericWatching(&gw, &wg) + go contractWatching(&gw, &wg) } wg.Wait() } diff --git a/cmd/contractWatcher.go b/cmd/contractWatcher.go index ae1c0ca7..9305249f 100644 --- a/cmd/contractWatcher.go +++ b/cmd/contractWatcher.go @@ -94,7 +94,7 @@ func contractWatcher() { blockChain := getBlockChain() db := utils.LoadPostgres(databaseConfig, blockChain.Node()) - var t st.GenericTransformer + var t st.ContractTransformer con := config.ContractConfig{} con.PrepConfig() switch mode { diff --git a/cmd/execute.go b/cmd/execute.go index b76bbb59..3010db59 100644 --- a/cmd/execute.go +++ b/cmd/execute.go @@ -99,8 +99,8 @@ func execute() { os.Exit(1) } - // Use the Exporters export method to load the EventTransformerInitializer and StorageTransformerInitializer sets - ethEventInitializers, ethStorageInitializers, genericInitializers := exporter.Export() + // Use the Exporters export method to load the EventTransformerInitializer, StorageTransformerInitializer, and ContractTransformerInitializer sets + ethEventInitializers, ethStorageInitializers, ethContractInitializers := exporter.Export() // Setup bc and db objects blockChain := getBlockChain() @@ -124,11 +124,11 @@ func execute() { go watchEthStorage(&sw, &wg) } - if len(genericInitializers) > 0 { - gw := watcher.NewGenericWatcher(&db, blockChain) - gw.AddTransformers(genericInitializers) + if len(ethContractInitializers) > 0 { + gw := watcher.NewContractWatcher(&db, blockChain) + gw.AddTransformers(ethContractInitializers) wg.Add(1) - go genericWatching(&gw, &wg) + go contractWatching(&gw, &wg) } wg.Wait() } @@ -139,7 +139,7 @@ func init() { } type Exporter interface { - Export() ([]transformer.EventTransformerInitializer, []transformer.StorageTransformerInitializer, []transformer.GenericTransformerInitializer) + Export() ([]transformer.EventTransformerInitializer, []transformer.StorageTransformerInitializer, []transformer.ContractTransformerInitializer) } func watchEthEvents(w *watcher.EventWatcher, wg *syn.WaitGroup) { @@ -176,10 +176,10 @@ func watchEthStorage(w *watcher.StorageWatcher, wg *syn.WaitGroup) { } } -func genericWatching(w *watcher.GenericWatcher, wg *syn.WaitGroup) { +func contractWatching(w *watcher.ContractWatcher, wg *syn.WaitGroup) { defer wg.Done() - // Execute over the GenericTransformerInitializer set using the generic watcher - log.Info("executing generic transformers") + // Execute over the ContractTransformerInitializer set using the contract watcher + log.Info("executing contract_watcher transformers") ticker := time.NewTicker(pollingInterval) defer ticker.Stop() for range ticker.C { diff --git a/environments/example.toml b/environments/example.toml index 2da5f317..66522f05 100644 --- a/environments/example.toml +++ b/environments/example.toml @@ -1,5 +1,5 @@ [database] - name = "vulcanize_public" + name = "vulcanize_infura" hostname = "localhost" port = 5432 @@ -10,13 +10,12 @@ network = "" addresses = [ "0x314159265dD8dbb310642f98f50C066173C1259b", - "0xdAC17F958D2ee523a2206206994597C13D831ec7" + "0x8dd5fbCe2F6a956C3022bA3663759011Dd51e73E" ] [contract.0x314159265dD8dbb310642f98f50C066173C1259b] abi = '[{"constant":true,"inputs":[{"name":"node","type":"bytes32"}],"name":"resolver","outputs":[{"name":"","type":"address"}],"payable":false,"type":"function"},{"constant":true,"inputs":[{"name":"node","type":"bytes32"}],"name":"owner","outputs":[{"name":"","type":"address"}],"payable":false,"type":"function"},{"constant":false,"inputs":[{"name":"node","type":"bytes32"},{"name":"label","type":"bytes32"},{"name":"owner","type":"address"}],"name":"setSubnodeOwner","outputs":[],"payable":false,"type":"function"},{"constant":false,"inputs":[{"name":"node","type":"bytes32"},{"name":"ttl","type":"uint64"}],"name":"setTTL","outputs":[],"payable":false,"type":"function"},{"constant":true,"inputs":[{"name":"node","type":"bytes32"}],"name":"ttl","outputs":[{"name":"","type":"uint64"}],"payable":false,"type":"function"},{"constant":false,"inputs":[{"name":"node","type":"bytes32"},{"name":"resolver","type":"address"}],"name":"setResolver","outputs":[],"payable":false,"type":"function"},{"constant":false,"inputs":[{"name":"node","type":"bytes32"},{"name":"owner","type":"address"}],"name":"setOwner","outputs":[],"payable":false,"type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"name":"node","type":"bytes32"},{"indexed":false,"name":"owner","type":"address"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"node","type":"bytes32"},{"indexed":true,"name":"label","type":"bytes32"},{"indexed":false,"name":"owner","type":"address"}],"name":"NewOwner","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"node","type":"bytes32"},{"indexed":false,"name":"resolver","type":"address"}],"name":"NewResolver","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"node","type":"bytes32"},{"indexed":false,"name":"ttl","type":"uint64"}],"name":"NewTTL","type":"event"}]' startingBlock = 3327417 - [contract.0xdAC17F958D2ee523a2206206994597C13D831ec7] - abi = '[{"constant":true,"inputs":[],"name":"name","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_upgradedAddress","type":"address"}],"name":"deprecate","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_spender","type":"address"},{"name":"_value","type":"uint256"}],"name":"approve","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"deprecated","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_evilUser","type":"address"}],"name":"addBlackList","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_from","type":"address"},{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"name":"transferFrom","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"upgradedAddress","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"address"}],"name":"balances","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"decimals","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"maximumFee","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"_totalSupply","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"unpause","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_maker","type":"address"}],"name":"getBlackListStatus","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"address"},{"name":"","type":"address"}],"name":"allowed","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"paused","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"who","type":"address"}],"name":"balanceOf","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"pause","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"getOwner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"owner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"symbol","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"name":"transfer","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"newBasisPoints","type":"uint256"},{"name":"newMaxFee","type":"uint256"}],"name":"setParams","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"amount","type":"uint256"}],"name":"issue","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"amount","type":"uint256"}],"name":"redeem","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_owner","type":"address"},{"name":"_spender","type":"address"}],"name":"allowance","outputs":[{"name":"remaining","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"basisPointsRate","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"address"}],"name":"isBlackListed","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_clearedUser","type":"address"}],"name":"removeBlackList","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"MAX_UINT","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_blackListedUser","type":"address"}],"name":"destroyBlackFunds","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"inputs":[{"name":"_initialSupply","type":"uint256"},{"name":"_name","type":"string"},{"name":"_symbol","type":"string"},{"name":"_decimals","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"name":"amount","type":"uint256"}],"name":"Issue","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"amount","type":"uint256"}],"name":"Redeem","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"newAddress","type":"address"}],"name":"Deprecate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"feeBasisPoints","type":"uint256"},{"indexed":false,"name":"maxFee","type":"uint256"}],"name":"Params","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"_blackListedUser","type":"address"},{"indexed":false,"name":"_balance","type":"uint256"}],"name":"DestroyedBlackFunds","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"_user","type":"address"}],"name":"AddedBlackList","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"_user","type":"address"}],"name":"RemovedBlackList","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"owner","type":"address"},{"indexed":true,"name":"spender","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"from","type":"address"},{"indexed":true,"name":"to","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[],"name":"Pause","type":"event"},{"anonymous":false,"inputs":[],"name":"Unpause","type":"event"}]' + [contract.0x8dd5fbCe2F6a956C3022bA3663759011Dd51e73E] events = [ "Transfer", "Issue" @@ -24,5 +23,4 @@ methods = [ "balanceOf" ] - startingBlock = 4634748 - piping = false \ No newline at end of file + startingBlock = 5197514 \ No newline at end of file diff --git a/integration_test/contract_watcher_full_transformer_test.go b/integration_test/contract_watcher_full_transformer_test.go index cecf11eb..03e174b9 100644 --- a/integration_test/contract_watcher_full_transformer_test.go +++ b/integration_test/contract_watcher_full_transformer_test.go @@ -42,7 +42,7 @@ var _ = Describe("contractWatcher full transformer", func() { It("Initializes transformer's contract objects", func() { blockRepository.CreateOrUpdateBlock(mocks.TransferBlock1) blockRepository.CreateOrUpdateBlock(mocks.TransferBlock2) - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) @@ -55,6 +55,28 @@ var _ = Describe("contractWatcher full transformer", func() { Expect(c.Name).To(Equal("TrueUSD")) Expect(c.Address).To(Equal(tusdAddr)) }) + + It("Fails to initialize if first and most recent blocks cannot be fetched from vDB", func() { + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) + err = t.Init() + Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) + }) + + It("Does nothing if watched events are unset", func() { + blockRepository.CreateOrUpdateBlock(mocks.TransferBlock1) + blockRepository.CreateOrUpdateBlock(mocks.TransferBlock2) + var testConf config.ContractConfig + testConf = test_helpers.TusdConfig + testConf.Events = nil + t := transformer.NewTransformer(testConf, blockChain, db) + err = t.Init() + Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no filters created")) + + _, ok := t.Contracts[tusdAddr] + Expect(ok).To(Equal(false)) + }) }) Describe("Execute", func() { @@ -64,7 +86,7 @@ var _ = Describe("contractWatcher full transformer", func() { }) It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) @@ -84,7 +106,7 @@ var _ = Describe("contractWatcher full transformer", func() { It("Keeps track of contract-related addresses while transforming event data if they need to be used for later method polling", func() { var testConf config.ContractConfig - testConf = mocks.TusdConfig + testConf = test_helpers.TusdConfig testConf.Methods = map[string][]string{ tusdAddr: {"balanceOf"}, } @@ -121,7 +143,7 @@ var _ = Describe("contractWatcher full transformer", func() { It("Polls given methods using generated token holder address", func() { var testConf config.ContractConfig - testConf = mocks.TusdConfig + testConf = test_helpers.TusdConfig testConf.Methods = map[string][]string{ tusdAddr: {"balanceOf"}, } @@ -146,13 +168,15 @@ var _ = Describe("contractWatcher full transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.balanceof_method WHERE who_ = '0xfE9e8709d3215310075d67E3ed32A380CCf451C8' AND block = '6194634'", tusdAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) It("Fails if initialization has not been done", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Execute() Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("transformer has no initialized contracts to work with")) }) }) @@ -163,7 +187,7 @@ var _ = Describe("contractWatcher full transformer", func() { }) It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.ENSConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) @@ -184,7 +208,7 @@ var _ = Describe("contractWatcher full transformer", func() { It("Keeps track of contract-related hashes while transforming event data if they need to be used for later method polling", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, } @@ -214,7 +238,7 @@ var _ = Describe("contractWatcher full transformer", func() { It("Polls given methods using generated token holder address", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, } @@ -238,11 +262,12 @@ var _ = Describe("contractWatcher full transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x9THIS110dcc444fIS242510c09bbAbe21aFAKEcacNODE82f7b843HASH61ba391' AND block = '6194636'", ensAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) It("It does not perist events if they do not pass the emitted arg filter", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.EventArgs = map[string][]string{ ensAddr: {"fake_filter_value"}, } @@ -257,11 +282,12 @@ var _ = Describe("contractWatcher full transformer", func() { log := test_helpers.LightNewOwnerLog{} err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.newowner_event", ensAddr)).StructScan(&log) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("does not exist")) }) It("If a method arg filter is applied, only those arguments are used in polling", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.MethodArgs = map[string][]string{ ensAddr: {"0x0000000000000000000000000000000000000000000000000000c02aaa39b223"}, } @@ -283,6 +309,7 @@ var _ = Describe("contractWatcher full transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x9dd48110dcc444fdc242510c09bbbbe21a5975cac061d82f7b843bce061ba391' AND block = '6194636'", ensAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) }) }) diff --git a/integration_test/contract_watcher_light_transformer_test.go b/integration_test/contract_watcher_light_transformer_test.go index 3ca326c3..45323c10 100644 --- a/integration_test/contract_watcher_light_transformer_test.go +++ b/integration_test/contract_watcher_light_transformer_test.go @@ -2,13 +2,13 @@ package integration import ( "fmt" - "github.com/vulcanize/vulcanizedb/pkg/config" "strings" "github.com/ethereum/go-ethereum/common" . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" + "github.com/vulcanize/vulcanizedb/pkg/config" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/light/transformer" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/constants" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers" @@ -40,7 +40,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Initializes transformer's contract objects", func() { headerRepository.CreateOrUpdateHeader(mocks.MockHeader1) headerRepository.CreateOrUpdateHeader(mocks.MockHeader3) - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) @@ -53,6 +53,27 @@ var _ = Describe("contractWatcher light transformer", func() { Expect(c.Name).To(Equal("TrueUSD")) Expect(c.Address).To(Equal(tusdAddr)) }) + + It("Fails to initialize if first and block cannot be fetched from vDB headers table", func() { + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) + err = t.Init() + Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) + }) + + It("Does nothing if nothing if no addresses are configured", func() { + headerRepository.CreateOrUpdateHeader(mocks.MockHeader1) + headerRepository.CreateOrUpdateHeader(mocks.MockHeader3) + var testConf config.ContractConfig + testConf = test_helpers.TusdConfig + testConf.Addresses = nil + t := transformer.NewTransformer(testConf, blockChain, db) + err = t.Init() + Expect(err).ToNot(HaveOccurred()) + + _, ok := t.Contracts[tusdAddr] + Expect(ok).To(Equal(false)) + }) }) Describe("Execute- against TrueUSD contract", func() { @@ -70,7 +91,7 @@ var _ = Describe("contractWatcher light transformer", func() { }) It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) err = t.Execute() @@ -88,7 +109,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Keeps track of contract-related addresses while transforming event data if they need to be used for later method polling", func() { var testConf config.ContractConfig - testConf = mocks.TusdConfig + testConf = test_helpers.TusdConfig testConf.Methods = map[string][]string{ tusdAddr: {"balanceOf"}, } @@ -133,7 +154,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Polls given methods using generated token holder address", func() { var testConf config.ContractConfig - testConf = mocks.TusdConfig + testConf = test_helpers.TusdConfig testConf.Methods = map[string][]string{ tusdAddr: {"balanceOf"}, } @@ -151,12 +172,14 @@ var _ = Describe("contractWatcher light transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843b1234567890' AND block = '6791669'", tusdAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) It("Fails if initialization has not been done", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.TusdConfig, blockChain, db) err = t.Execute() Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("transformer has no initialized contracts")) }) }) @@ -175,7 +198,7 @@ var _ = Describe("contractWatcher light transformer", func() { }) It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.ENSConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) err = t.Execute() @@ -193,7 +216,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Keeps track of contract-related hashes while transforming event data if they need to be used for later method polling", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, } @@ -222,7 +245,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Polls given method using list of collected hashes", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, } @@ -245,11 +268,12 @@ var _ = Describe("contractWatcher light transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x9THIS110dcc444fIS242510c09bbAbe21aFAKEcacNODE82f7b843HASH61ba391' AND block = '6885696'", ensAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) It("It does not persist events if they do not pass the emitted arg filter", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.EventArgs = map[string][]string{ ensAddr: {"fake_filter_value"}, } @@ -262,11 +286,12 @@ var _ = Describe("contractWatcher light transformer", func() { log := test_helpers.LightNewOwnerLog{} err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.newowner_event", ensAddr)).StructScan(&log) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("does not exist")) }) It("If a method arg filter is applied, only those arguments are used in polling", func() { var testConf config.ContractConfig - testConf = mocks.ENSConfig + testConf = test_helpers.ENSConfig testConf.MethodArgs = map[string][]string{ ensAddr: {"0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae"}, } @@ -287,6 +312,7 @@ var _ = Describe("contractWatcher light transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047' AND block = '6885696'", ensAddr)).StructScan(&res) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) }) @@ -301,7 +327,7 @@ var _ = Describe("contractWatcher light transformer", func() { }) It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSandTusdConfig, blockChain, db) + t := transformer.NewTransformer(test_helpers.ENSandTusdConfig, blockChain, db) err = t.Init() Expect(err).ToNot(HaveOccurred()) err = t.Execute() @@ -326,7 +352,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Keeps track of contract-related hashes and addresses while transforming event data if they need to be used for later method polling", func() { var testConf config.ContractConfig - testConf = mocks.ENSandTusdConfig + testConf = test_helpers.ENSandTusdConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, tusdAddr: {"balanceOf"}, @@ -367,7 +393,7 @@ var _ = Describe("contractWatcher light transformer", func() { It("Polls given methods for each contract, using list of collected values", func() { var testConf config.ContractConfig - testConf = mocks.ENSandTusdConfig + testConf = test_helpers.ENSandTusdConfig testConf.Methods = map[string][]string{ ensAddr: {"owner"}, tusdAddr: {"balanceOf"}, @@ -391,6 +417,7 @@ var _ = Describe("contractWatcher light transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ceMADEUPaaf4HASHc186badTHItransformers.8IS625bFAKE' AND block = '6885696'", ensAddr)).StructScan(&owner) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) bal := test_helpers.BalanceOf{} err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x8cA465764873E71CEa525F5EB6AE973d650c22C2' AND block = '6885701'", tusdAddr)).StructScan(&bal) @@ -400,6 +427,7 @@ var _ = Describe("contractWatcher light transformer", func() { err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843b1234567890' AND block = '6885701'", tusdAddr)).StructScan(&bal) Expect(err).To(HaveOccurred()) + Expect(err.Error()).To(ContainSubstring("no rows in result set")) }) }) }) diff --git a/libraries/shared/transformer/generic_transformer.go b/libraries/shared/transformer/contract_transformer.go similarity index 87% rename from libraries/shared/transformer/generic_transformer.go rename to libraries/shared/transformer/contract_transformer.go index fc133b17..14b8ab97 100644 --- a/libraries/shared/transformer/generic_transformer.go +++ b/libraries/shared/transformer/contract_transformer.go @@ -22,10 +22,10 @@ import ( "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres" ) -type GenericTransformer interface { +type ContractTransformer interface { Init() error Execute() error GetConfig() config.ContractConfig } -type GenericTransformerInitializer func(db *postgres.DB, bc core.BlockChain) GenericTransformer +type ContractTransformerInitializer func(db *postgres.DB, bc core.BlockChain) ContractTransformer diff --git a/libraries/shared/watcher/generic_watcher.go b/libraries/shared/watcher/contract_watcher.go similarity index 71% rename from libraries/shared/watcher/generic_watcher.go rename to libraries/shared/watcher/contract_watcher.go index 77b609c2..880dfb5b 100644 --- a/libraries/shared/watcher/generic_watcher.go +++ b/libraries/shared/watcher/contract_watcher.go @@ -14,10 +14,10 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . -// Dynamic watcher is built with a more generic interface -// that allows offloading more of the operatinal logic to +// Contract watcher is built with a more generic interface +// that allows offloading more of the operational logic to // the transformers, allowing them to act more dynamically -// Built to work primarily with the omni pkging +// Built to work primarily with the contract_watcher packaging package watcher import ( @@ -30,26 +30,26 @@ import ( "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres" ) -type GenericWatcher struct { - Transformers []transformer.GenericTransformer +type ContractWatcher struct { + Transformers []transformer.ContractTransformer DB *postgres.DB BlockChain core.BlockChain } -func NewGenericWatcher(db *postgres.DB, bc core.BlockChain) GenericWatcher { - return GenericWatcher{ +func NewContractWatcher(db *postgres.DB, bc core.BlockChain) ContractWatcher { + return ContractWatcher{ DB: db, BlockChain: bc, } } -func (watcher *GenericWatcher) AddTransformers(inits interface{}) error { - initializers, ok := inits.([]transformer.GenericTransformerInitializer) +func (watcher *ContractWatcher) AddTransformers(inits interface{}) error { + initializers, ok := inits.([]transformer.ContractTransformerInitializer) if !ok { - return fmt.Errorf("initializers of type %T, not %T", inits, []transformer.GenericTransformerInitializer{}) + return fmt.Errorf("initializers of type %T, not %T", inits, []transformer.ContractTransformerInitializer{}) } - watcher.Transformers = make([]transformer.GenericTransformer, 0, len(initializers)) + watcher.Transformers = make([]transformer.ContractTransformer, 0, len(initializers)) for _, initializer := range initializers { t := initializer(watcher.DB, watcher.BlockChain) watcher.Transformers = append(watcher.Transformers, t) @@ -65,7 +65,7 @@ func (watcher *GenericWatcher) AddTransformers(inits interface{}) error { return nil } -func (watcher *GenericWatcher) Execute(interface{}) error { +func (watcher *ContractWatcher) Execute(interface{}) error { for _, transformer := range watcher.Transformers { err := transformer.Execute() if err != nil { diff --git a/pkg/config/contract.go b/pkg/config/contract.go index a9dfefa7..a2a09d2f 100644 --- a/pkg/config/contract.go +++ b/pkg/config/contract.go @@ -71,7 +71,7 @@ func (oc *ContractConfig) PrepConfig() { oc.Addresses = make(map[string]bool, len(addrs)) oc.Abis = make(map[string]string, len(addrs)) oc.Methods = make(map[string][]string, len(addrs)) - oc.EventArgs = make(map[string][]string, len(addrs)) + oc.Events = make(map[string][]string, len(addrs)) oc.MethodArgs = make(map[string][]string, len(addrs)) oc.EventArgs = make(map[string][]string, len(addrs)) oc.StartingBlocks = make(map[string]int64, len(addrs)) @@ -86,92 +86,131 @@ func (oc *ContractConfig) PrepConfig() { transformer := viper.GetStringMap("contract." + addr) // Get and check abi - abi, abiOK := transformer["abi"] - if !abiOK || abi == nil { - log.Fatal(addr, "transformer config is missing `abi` value") - } - abiRef, abiOK := abi.(string) + var abi string + _, abiOK := transformer["abi"] if !abiOK { - log.Fatal(addr, "transformer `events` not of type []string") + log.Warnf("contract %s not configured with an ABI, will attempt to fetch it from Etherscan\r\n", addr) + } else { + abiInterface := transformer["abi"] + abi, abiOK = abiInterface.(string) + if !abiOK { + log.Fatal(addr, "transformer `abi` not of type []string") + } } - oc.Abis[strings.ToLower(addr)] = abiRef + oc.Abis[strings.ToLower(addr)] = abi // Get and check events - events, eventsOK := transformer["events"] - if !eventsOK || events == nil { - log.Fatal(addr, "transformer config is missing `events` value") - } - eventsRef, eventsOK := events.([]string) + events := make([]string, 0) + _, eventsOK := transformer["events"] if !eventsOK { - log.Fatal(addr, "transformer `events` not of type []string") + log.Warnf("contract %s not configured with a list of events to watch, will watch all events\r\n", addr) + events = []string{} + } else { + eventsInterface := transformer["events"] + eventsI, eventsOK := eventsInterface.([]interface{}) + if !eventsOK { + log.Fatal(addr, "transformer `events` not of type []string\r\n") + } + for _, strI := range eventsI { + str, strOK := strI.(string) + if !strOK { + log.Fatal(addr, "transformer `events` not of type []string\r\n") + } + events = append(events, str) + } } - if eventsRef == nil { - eventsRef = []string{} - } - oc.Events[strings.ToLower(addr)] = eventsRef + oc.Events[strings.ToLower(addr)] = events // Get and check methods - methods, methodsOK := transformer["methods"] - if !methodsOK || methods == nil { - log.Fatal(addr, "transformer config is missing `methods` value") - } - methodsRef, methodsOK := methods.([]string) + methods := make([]string, 0) + _, methodsOK := transformer["methods"] if !methodsOK { - log.Fatal(addr, "transformer `methods` not of type []string") + log.Warnf("contract %s not configured with a list of methods to poll, will not poll any methods\r\n", addr) + methods = []string{} + } else { + methodsInterface := transformer["methods"] + methodsI, methodsOK := methodsInterface.([]interface{}) + if !methodsOK { + log.Fatal(addr, "transformer `methods` not of type []string\r\n") + } + for _, strI := range methodsI { + str, strOK := strI.(string) + if !strOK { + log.Fatal(addr, "transformer `methods` not of type []string\r\n") + } + methods = append(methods, str) + } } - if methodsRef == nil { - methodsRef = []string{} - } - oc.Methods[strings.ToLower(addr)] = methodsRef + oc.Methods[strings.ToLower(addr)] = methods // Get and check eventArgs - eventArgs, eventArgsOK := transformer["eventArgs"] - if !eventArgsOK || eventArgs == nil { - log.Fatal(addr, "transformer config is missing `eventArgs` value") - } - eventArgsRef, eventArgsOK := eventArgs.([]string) + eventArgs := make([]string, 0) + _, eventArgsOK := transformer["eventArgs"] if !eventArgsOK { - log.Fatal(addr, "transformer `eventArgs` not of type []string") + log.Warnf("contract %s not configured with a list of event arguments to filter for, will not filter events for specific emitted values\r\n", addr) + eventArgs = []string{} + } else { + eventArgsInterface := transformer["eventArgs"] + eventArgsI, eventArgsOK := eventArgsInterface.([]interface{}) + if !eventArgsOK { + log.Fatal(addr, "transformer `eventArgs` not of type []string\r\n") + } + for _, strI := range eventArgsI { + str, strOK := strI.(string) + if !strOK { + log.Fatal(addr, "transformer `eventArgs` not of type []string\r\n") + } + eventArgs = append(eventArgs, str) + } } - if eventArgsRef == nil { - eventArgsRef = []string{} - } - oc.EventArgs[strings.ToLower(addr)] = eventArgsRef + oc.EventArgs[strings.ToLower(addr)] = eventArgs // Get and check methodArgs - methodArgs, methodArgsOK := transformer["methodArgs"] - if !methodArgsOK || methodArgs == nil { - log.Fatal(addr, "transformer config is missing `methodArgs` value") - } - methodArgsRef, methodArgsOK := methodArgs.([]string) + methodArgs := make([]string, 0) + _, methodArgsOK := transformer["methodArgs"] if !methodArgsOK { - log.Fatal(addr, "transformer `methodArgs` not of type []string") + log.Warnf("contract %s not configured with a list of method argument values to poll with, will poll methods with all available arguments\r\n", addr) + methodArgs = []string{} + } else { + methodArgsInterface := transformer["methodArgs"] + methodArgsI, methodArgsOK := methodArgsInterface.([]interface{}) + if !methodArgsOK { + log.Fatal(addr, "transformer `methodArgs` not of type []string\r\n") + } + for _, strI := range methodArgsI { + str, strOK := strI.(string) + if !strOK { + log.Fatal(addr, "transformer `methodArgs` not of type []string\r\n") + } + methodArgs = append(methodArgs, str) + } } - if methodArgsRef == nil { - methodArgsRef = []string{} - } - oc.MethodArgs[strings.ToLower(addr)] = methodArgsRef + oc.MethodArgs[strings.ToLower(addr)] = methodArgs // Get and check startingBlock - start, startOK := transformer["startingBlock"] - if !startOK || start == nil { - log.Fatal(addr, "transformer config is missing `startingBlock` value") - } - startRef, startOK := start.(int64) + startInterface, startOK := transformer["startingblock"] if !startOK { - log.Fatal(addr, "transformer `startingBlock` not of type int") + log.Fatal(addr, "transformer config is missing `startingBlock` value\r\n") } - oc.StartingBlocks[strings.ToLower(addr)] = startRef + start, startOK := startInterface.(int64) + if !startOK { + log.Fatal(addr, "transformer `startingBlock` not of type int\r\n") + } + oc.StartingBlocks[strings.ToLower(addr)] = start // Get pipping - pipe, pipeOK := transformer["pipping"] - if !pipeOK || pipe == nil { - log.Fatal(addr, "transformer config is missing `pipping` value") - } - pipeRef, pipeOK := pipe.(bool) + var piping bool + _, pipeOK := transformer["piping"] if !pipeOK { - log.Fatal(addr, "transformer `piping` not of type bool") + log.Warnf("contract %s does not have its `piping` set, by default piping is turned off\r\n", addr) + piping = false + } else { + pipingInterface := transformer["piping"] + piping, pipeOK = pipingInterface.(bool) + if !pipeOK { + log.Fatal(addr, "transformer `piping` not of type bool\r\n") + } } - oc.Piping[strings.ToLower(addr)] = pipeRef + oc.Piping[strings.ToLower(addr)] = piping } } diff --git a/pkg/config/plugin.go b/pkg/config/plugin.go index f521cb43..fad07cbc 100644 --- a/pkg/config/plugin.go +++ b/pkg/config/plugin.go @@ -41,13 +41,13 @@ type Transformer struct { RepositoryPath string } -func (c *Plugin) GetPluginPaths() (string, string, error) { - path, err := helpers.CleanPath(c.FilePath) +func (pluginConfig *Plugin) GetPluginPaths() (string, string, error) { + path, err := helpers.CleanPath(pluginConfig.FilePath) if err != nil { return "", "", err } - name := strings.Split(c.FileName, ".")[0] + name := strings.Split(pluginConfig.FileName, ".")[0] goFile := filepath.Join(path, name+".go") soFile := filepath.Join(path, name+".so") @@ -55,13 +55,13 @@ func (c *Plugin) GetPluginPaths() (string, string, error) { } // Removes duplicate migration paths and returns them in ranked order -func (c *Plugin) GetMigrationsPaths() ([]string, error) { +func (pluginConfig *Plugin) GetMigrationsPaths() ([]string, error) { paths := make(map[uint64]string) highestRank := -1 - for name, transformer := range c.Transformers { + for name, transformer := range pluginConfig.Transformers { repo := transformer.RepositoryPath mig := transformer.MigrationPath - path := filepath.Join("$GOPATH/src", c.Home, "vendor", repo, mig) + path := filepath.Join("$GOPATH/src", pluginConfig.Home, "vendor", repo, mig) cleanPath, err := helpers.CleanPath(path) if err != nil { return nil, err @@ -96,9 +96,9 @@ func (c *Plugin) GetMigrationsPaths() ([]string, error) { } // Removes duplicate repo paths before returning them -func (c *Plugin) GetRepoPaths() map[string]bool { +func (pluginConfig *Plugin) GetRepoPaths() map[string]bool { paths := make(map[string]bool) - for _, transformer := range c.Transformers { + for _, transformer := range pluginConfig.Transformers { paths[transformer.RepositoryPath] = true } @@ -111,29 +111,29 @@ const ( UnknownTransformerType TransformerType = iota EthEvent EthStorage - EthGeneric + EthContract ) -func (pt TransformerType) String() string { +func (transformerType TransformerType) String() string { names := [...]string{ "Unknown", "eth_event", "eth_storage", - "eth_generic", + "eth_contract", } - if pt > EthGeneric || pt < EthEvent { + if transformerType > EthContract || transformerType < EthEvent { return "Unknown" } - return names[pt] + return names[transformerType] } func GetTransformerType(str string) TransformerType { types := [...]TransformerType{ EthEvent, EthStorage, - EthGeneric, + EthContract, } for _, ty := range types { diff --git a/pkg/contract_watcher/full/converter/converter.go b/pkg/contract_watcher/full/converter/converter.go index 47f3fa42..39341145 100644 --- a/pkg/contract_watcher/full/converter/converter.go +++ b/pkg/contract_watcher/full/converter/converter.go @@ -34,27 +34,21 @@ import ( // Converter is used to convert watched event logs to // custom logs containing event input name => value maps -type Converter interface { +type ConverterInterface interface { Convert(watchedEvent core.WatchedEvent, event types.Event) (*types.Log, error) Update(info *contract.Contract) } -type converter struct { +type Converter struct { ContractInfo *contract.Contract } -func NewConverter(info *contract.Contract) *converter { - return &converter{ - ContractInfo: info, - } -} - -func (c *converter) Update(info *contract.Contract) { +func (c *Converter) Update(info *contract.Contract) { c.ContractInfo = info } // Convert the given watched event log into a types.Log for the given event -func (c *converter) Convert(watchedEvent core.WatchedEvent, event types.Event) (*types.Log, error) { +func (c *Converter) Convert(watchedEvent core.WatchedEvent, event types.Event) (*types.Log, error) { contract := bind.NewBoundContract(common.HexToAddress(c.ContractInfo.Address), c.ContractInfo.ParsedAbi, nil, nil, nil) values := make(map[string]interface{}) log := helpers.ConvertToLog(watchedEvent) diff --git a/pkg/contract_watcher/full/converter/converter_test.go b/pkg/contract_watcher/full/converter/converter_test.go index 9b7a46cb..864a53a2 100644 --- a/pkg/contract_watcher/full/converter/converter_test.go +++ b/pkg/contract_watcher/full/converter/converter_test.go @@ -39,7 +39,8 @@ var _ = Describe("Converter", func() { Describe("Update", func() { It("Updates contract con held by the converter", func() { - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) Expect(c.ContractInfo).To(Equal(con)) con := test_helpers.SetupTusdContract([]string{}, []string{}) @@ -58,7 +59,8 @@ var _ = Describe("Converter", func() { err = con.GenerateFilters() Expect(err).ToNot(HaveOccurred()) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) log, err := c.Convert(mocks.MockTranferEvent, event) Expect(err).ToNot(HaveOccurred()) @@ -77,7 +79,8 @@ var _ = Describe("Converter", func() { event, ok := con.Events["Transfer"] Expect(ok).To(Equal(true)) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) _, err := c.Convert(mocks.MockTranferEvent, event) Expect(err).ToNot(HaveOccurred()) @@ -101,7 +104,8 @@ var _ = Describe("Converter", func() { It("Fails with an empty contract", func() { event := con.Events["Transfer"] - c := converter.NewConverter(&contract.Contract{}) + c := converter.Converter{} + c.Update(&contract.Contract{}) _, err = c.Convert(mocks.MockTranferEvent, event) Expect(err).To(HaveOccurred()) }) diff --git a/pkg/contract_watcher/full/transformer/transformer.go b/pkg/contract_watcher/full/transformer/transformer.go index 119aa6ec..132292ab 100644 --- a/pkg/contract_watcher/full/transformer/transformer.go +++ b/pkg/contract_watcher/full/transformer/transformer.go @@ -34,7 +34,7 @@ import ( ) // Requires a fully synced vDB and a running eth node (or infura) -type transformer struct { +type Transformer struct { // Database interfaces datastore.FilterRepository // Log filters repo; accepts filters generated by Contract.GenerateFilters() datastore.WatchedEventRepository // Watched event log views, created by the log filters @@ -45,8 +45,8 @@ type transformer struct { retriever.BlockRetriever // Retrieves first block for contract and current block height // Processing interfaces - converter.Converter // Converts watched event logs into custom log - poller.Poller // Polls methods using contract's token holder addresses and persists them using method datastore + converter.ConverterInterface // Converts watched event logs into custom log + poller.Poller // Polls methods using contract's token holder addresses and persists them using method datastore // Store contract configuration information Config config.ContractConfig @@ -56,12 +56,12 @@ type transformer struct { } // Transformer takes in config for blockchain, database, and network id -func NewTransformer(con config.ContractConfig, BC core.BlockChain, DB *postgres.DB) *transformer { - return &transformer{ +func NewTransformer(con config.ContractConfig, BC core.BlockChain, DB *postgres.DB) *Transformer { + return &Transformer{ Poller: poller.NewPoller(BC, DB, types.FullSync), Parser: parser.NewParser(con.Network), BlockRetriever: retriever.NewBlockRetriever(DB), - Converter: converter.NewConverter(&contract.Contract{}), + ConverterInterface: &converter.Converter{}, Contracts: map[string]*contract.Contract{}, WatchedEventRepository: repositories.WatchedEventRepository{DB: DB}, FilterRepository: repositories.FilterRepository{DB: DB}, @@ -74,7 +74,7 @@ func NewTransformer(con config.ContractConfig, BC core.BlockChain, DB *postgres. // Loops over all of the addr => filter sets // Uses parser to pull event info from abi // Use this info to generate event filters -func (tr *transformer) Init() error { +func (tr *Transformer) Init() error { for contractAddr := range tr.Config.Addresses { // Configure Abi if tr.Config.Abis[contractAddr] == "" { @@ -108,7 +108,7 @@ func (tr *transformer) Init() error { // Get contract name if it has one var name = new(string) - tr.FetchContractData(tr.Abi(), contractAddr, "name", nil, &name, lastBlock) + tr.FetchContractData(tr.Abi(), contractAddr, "name", nil, name, lastBlock) // Remove any potential accidental duplicate inputs in arg filter values eventArgs := map[string]bool{} @@ -162,7 +162,7 @@ func (tr *transformer) Init() error { // Uses converter to convert logs into custom log type // Persists converted logs into custuom postgres tables // Calls selected methods, using token holder address generated during event log conversion -func (tr *transformer) Execute() error { +func (tr *Transformer) Execute() error { if len(tr.Contracts) == 0 { return errors.New("error: transformer has no initialized contracts to work with") } @@ -181,7 +181,7 @@ func (tr *transformer) Execute() error { // Iterate over watched event logs for _, we := range watchedEvents { // Convert them to our custom log type - cstm, err := tr.Converter.Convert(*we, con.Events[eventSig]) + cstm, err := tr.ConverterInterface.Convert(*we, con.Events[eventSig]) if err != nil { return err } @@ -210,6 +210,6 @@ func (tr *transformer) Execute() error { return nil } -func (tr *transformer) GetConfig() config.ContractConfig { +func (tr *Transformer) GetConfig() config.ContractConfig { return tr.Config } diff --git a/pkg/contract_watcher/full/transformer/transformer_test.go b/pkg/contract_watcher/full/transformer/transformer_test.go index ee6d465e..0cb0fb31 100644 --- a/pkg/contract_watcher/full/transformer/transformer_test.go +++ b/pkg/contract_watcher/full/transformer/transformer_test.go @@ -17,307 +17,82 @@ package transformer_test import ( - "fmt" - "github.com/vulcanize/vulcanizedb/pkg/config" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers/mocks" "math/rand" - "strings" "time" - "github.com/ethereum/go-ethereum/common" . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/full/retriever" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/full/transformer" - "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/constants" - "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers" - "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers/mocks" - "github.com/vulcanize/vulcanizedb/pkg/core" - "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres" - "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres/repositories" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/contract" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/parser" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/poller" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/types" + "github.com/vulcanize/vulcanizedb/pkg/fakes" ) var _ = Describe("Transformer", func() { - var db *postgres.DB - var err error - var blockChain core.BlockChain - var blockRepository repositories.BlockRepository - var ensAddr = strings.ToLower(constants.EnsContractAddress) - var tusdAddr = strings.ToLower(constants.TusdContractAddress) + var fakeAddress = "0x1234567890abcdef" rand.Seed(time.Now().UnixNano()) - BeforeEach(func() { - db, blockChain = test_helpers.SetupDBandBC() - blockRepository = *repositories.NewBlockRepository(db) - }) - - AfterEach(func() { - test_helpers.TearDown(db) - }) - Describe("Init", func() { It("Initializes transformer's contract objects", func() { - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock1) - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock2) - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() + blockRetriever := &fakes.MockFullBlockRetriever{} + firstBlock := int64(1) + mostRecentBlock := int64(2) + blockRetriever.FirstBlock = firstBlock + blockRetriever.MostRecentBlock = mostRecentBlock + + parsr := &fakes.MockParser{} + fakeAbi := "fake_abi" + eventName := "Transfer" + event := types.Event{} + parsr.AbiToReturn = fakeAbi + parsr.EventName = eventName + parsr.Event = event + + pollr := &fakes.MockPoller{} + fakeContractName := "fake_contract_name" + pollr.ContractName = fakeContractName + + t := getTransformer(blockRetriever, parsr, pollr) + + err := t.Init() + Expect(err).ToNot(HaveOccurred()) - c, ok := t.Contracts[tusdAddr] + c, ok := t.Contracts[fakeAddress] Expect(ok).To(Equal(true)) - Expect(c.StartingBlock).To(Equal(int64(6194633))) - Expect(c.LastBlock).To(Equal(int64(6194634))) - Expect(c.Abi).To(Equal(constants.TusdAbiString)) - Expect(c.Name).To(Equal("TrueUSD")) - Expect(c.Address).To(Equal(tusdAddr)) + Expect(c.StartingBlock).To(Equal(firstBlock)) + Expect(c.LastBlock).To(Equal(mostRecentBlock)) + Expect(c.Abi).To(Equal(fakeAbi)) + Expect(c.Name).To(Equal(fakeContractName)) + Expect(c.Address).To(Equal(fakeAddress)) }) It("Fails to initialize if first and most recent blocks cannot be fetched from vDB", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() - Expect(err).To(HaveOccurred()) - }) - - It("Does nothing if watched events are unset", func() { - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock1) - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock2) - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Events = nil - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).To(HaveOccurred()) - - _, ok := t.Contracts[tusdAddr] - Expect(ok).To(Equal(false)) - }) - }) - - Describe("Execute", func() { - BeforeEach(func() { - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock1) - blockRepository.CreateOrUpdateBlock(mocks.TransferBlock2) - }) - - It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.TransferLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.transfer_event WHERE block = 6194634", tusdAddr)).StructScan(&log) - - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(log.Tx).To(Equal("0x135391a0962a63944e5908e6fedfff90fb4be3e3290a21017861099bad654eee")) - Expect(log.Block).To(Equal(int64(6194634))) - Expect(log.From).To(Equal("0x000000000000000000000000000000000000Af21")) - Expect(log.To).To(Equal("0x09BbBBE21a5975cAc061D82f7b843bCE061BA391")) - Expect(log.Value).To(Equal("1097077688018008265106216665536940668749033598146")) - }) - - It("Keeps track of contract-related addresses while transforming event data if they need to be used for later method polling", func() { - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Methods = map[string][]string{ - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - c, ok := t.Contracts[tusdAddr] - Expect(ok).To(Equal(true)) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - b, ok := c.EmittedAddrs[common.HexToAddress("0x000000000000000000000000000000000000Af21")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedAddrs[common.HexToAddress("0x09BbBBE21a5975cAc061D82f7b843bCE061BA391")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x09BbBBE21a5975cAc061D82f7b843b1234567890")] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x")] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[""] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x09THISE21a5IS5cFAKE1D82fAND43bCE06MADEUP")] - Expect(ok).To(Equal(false)) - }) - - It("Polls given methods using generated token holder address", func() { - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Methods = map[string][]string{ - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.BalanceOf{} - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.balanceof_method WHERE who_ = '0x000000000000000000000000000000000000Af21' AND block = '6194634'", tusdAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Balance).To(Equal("0")) - Expect(res.TokenName).To(Equal("TrueUSD")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843bCE061BA391' AND block = '6194634'", tusdAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Balance).To(Equal("0")) - Expect(res.TokenName).To(Equal("TrueUSD")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.balanceof_method WHERE who_ = '0xfE9e8709d3215310075d67E3ed32A380CCf451C8' AND block = '6194634'", tusdAddr)).StructScan(&res) - Expect(err).To(HaveOccurred()) - }) - - It("Fails if initialization has not been done", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - - err = t.Execute() - Expect(err).To(HaveOccurred()) - }) - }) - - Describe("Execute- against ENS registry contract", func() { - BeforeEach(func() { - blockRepository.CreateOrUpdateBlock(mocks.NewOwnerBlock1) - blockRepository.CreateOrUpdateBlock(mocks.NewOwnerBlock2) - }) - - It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSConfig, blockChain, db) - - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.NewOwnerLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.newowner_event", ensAddr)).StructScan(&log) - - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(log.Tx).To(Equal("0x135391a0962a63944e5908e6fedfff90fb4be3e3290a21017861099bad654bbb")) - Expect(log.Block).To(Equal(int64(6194635))) - Expect(log.Node).To(Equal("0x0000000000000000000000000000000000000000000000000000c02aaa39b223")) - Expect(log.Label).To(Equal("0x9dd48110dcc444fdc242510c09bbbbe21a5975cac061d82f7b843bce061ba391")) - Expect(log.Owner).To(Equal("0x000000000000000000000000000000000000Af21")) - }) - - It("Keeps track of contract-related hashes while transforming event data if they need to be used for later method polling", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - c, ok := t.Contracts[ensAddr] - Expect(ok).To(Equal(true)) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - Expect(len(c.EmittedHashes)).To(Equal(3)) - - b, ok := c.EmittedHashes[common.HexToHash("0x0000000000000000000000000000000000000000000000000000c02aaa39b223")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedHashes[common.HexToHash("0x9dd48110dcc444fdc242510c09bbbbe21a5975cac061d82f7b843bce061ba391")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - // Doesn't keep track of address since it wouldn't be used in calling the 'owner' method - _, ok = c.EmittedAddrs[common.HexToAddress("0x000000000000000000000000000000000000Af21")] - Expect(ok).To(Equal(false)) - }) - - It("Polls given methods using generated token holder address", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.Owner{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x0000000000000000000000000000000000000000000000000000c02aaa39b223' AND block = '6194636'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x0000000000000000000000000000000000000000")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x9dd48110dcc444fdc242510c09bbbbe21a5975cac061d82f7b843bce061ba391' AND block = '6194636'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x0000000000000000000000000000000000000000")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x9THIS110dcc444fIS242510c09bbAbe21aFAKEcacNODE82f7b843HASH61ba391' AND block = '6194636'", ensAddr)).StructScan(&res) - Expect(err).To(HaveOccurred()) - }) - - It("It does not perist events if they do not pass the emitted arg filter", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.EventArgs = map[string][]string{ - ensAddr: {"fake_filter_value"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.LightNewOwnerLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.newowner_event", ensAddr)).StructScan(&log) - Expect(err).To(HaveOccurred()) - }) - - It("If a method arg filter is applied, only those arguments are used in polling", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - testConf.MethodArgs = map[string][]string{ - ensAddr: {"0x0000000000000000000000000000000000000000000000000000c02aaa39b223"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.Owner{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x0000000000000000000000000000000000000000000000000000c02aaa39b223' AND block = '6194636'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x0000000000000000000000000000000000000000")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM full_%s.owner_method WHERE node_ = '0x9dd48110dcc444fdc242510c09bbbbe21a5975cac061d82f7b843bce061ba391' AND block = '6194636'", ensAddr)).StructScan(&res) + blockRetriever := &fakes.MockFullBlockRetriever{} + blockRetriever.FirstBlockErr = fakes.FakeError + t := getTransformer(blockRetriever, &fakes.MockParser{}, &fakes.MockPoller{}) + + err := t.Init() + Expect(err).To(HaveOccurred()) + Expect(err).To(MatchError(fakes.FakeError)) }) }) }) + +func getTransformer(blockRetriever retriever.BlockRetriever, parsr parser.Parser, pollr poller.Poller) transformer.Transformer { + return transformer.Transformer{ + FilterRepository: &fakes.MockFilterRepository{}, + Parser: parsr, + BlockRetriever: blockRetriever, + Poller: pollr, + Contracts: map[string]*contract.Contract{}, + Config: mocks.MockConfig, + } +} diff --git a/pkg/contract_watcher/light/converter/converter.go b/pkg/contract_watcher/light/converter/converter.go index b938c61e..a1b7e051 100644 --- a/pkg/contract_watcher/light/converter/converter.go +++ b/pkg/contract_watcher/light/converter/converter.go @@ -32,28 +32,22 @@ import ( "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/types" ) -type Converter interface { +type ConverterInterface interface { Convert(logs []gethTypes.Log, event types.Event, headerID int64) ([]types.Log, error) ConvertBatch(logs []gethTypes.Log, events map[string]types.Event, headerID int64) (map[string][]types.Log, error) Update(info *contract.Contract) } -type converter struct { +type Converter struct { ContractInfo *contract.Contract } -func NewConverter(info *contract.Contract) *converter { - return &converter{ - ContractInfo: info, - } -} - -func (c *converter) Update(info *contract.Contract) { +func (c *Converter) Update(info *contract.Contract) { c.ContractInfo = info } // Convert the given watched event log into a types.Log for the given event -func (c *converter) Convert(logs []gethTypes.Log, event types.Event, headerID int64) ([]types.Log, error) { +func (c *Converter) Convert(logs []gethTypes.Log, event types.Event, headerID int64) ([]types.Log, error) { contract := bind.NewBoundContract(common.HexToAddress(c.ContractInfo.Address), c.ContractInfo.ParsedAbi, nil, nil, nil) returnLogs := make([]types.Log, 0, len(logs)) for _, log := range logs { @@ -132,7 +126,7 @@ func (c *converter) Convert(logs []gethTypes.Log, event types.Event, headerID in } // Convert the given watched event logs into types.Logs; returns a map of event names to a slice of their converted logs -func (c *converter) ConvertBatch(logs []gethTypes.Log, events map[string]types.Event, headerID int64) (map[string][]types.Log, error) { +func (c *Converter) ConvertBatch(logs []gethTypes.Log, events map[string]types.Event, headerID int64) (map[string][]types.Log, error) { contract := bind.NewBoundContract(common.HexToAddress(c.ContractInfo.Address), c.ContractInfo.ParsedAbi, nil, nil, nil) eventsToLogs := make(map[string][]types.Log) for _, event := range events { diff --git a/pkg/contract_watcher/light/converter/converter_test.go b/pkg/contract_watcher/light/converter/converter_test.go index fb29c556..769fe057 100644 --- a/pkg/contract_watcher/light/converter/converter_test.go +++ b/pkg/contract_watcher/light/converter/converter_test.go @@ -38,7 +38,8 @@ var _ = Describe("Converter", func() { Describe("Update", func() { It("Updates contract info held by the converter", func() { con = test_helpers.SetupTusdContract(tusdWantedEvents, []string{}) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) Expect(c.ContractInfo).To(Equal(con)) info := test_helpers.SetupTusdContract([]string{}, []string{}) @@ -56,7 +57,8 @@ var _ = Describe("Converter", func() { event, ok := con.Events["Transfer"] Expect(ok).To(Equal(true)) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) logs, err := c.Convert([]types.Log{mocks.MockTransferLog1, mocks.MockTransferLog2}, event, 232) Expect(err).ToNot(HaveOccurred()) Expect(len(logs)).To(Equal(2)) @@ -80,7 +82,8 @@ var _ = Describe("Converter", func() { event, ok := con.Events["Transfer"] Expect(ok).To(Equal(true)) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) _, err := c.Convert([]types.Log{mocks.MockTransferLog1, mocks.MockTransferLog2}, event, 232) Expect(err).ToNot(HaveOccurred()) @@ -110,7 +113,8 @@ var _ = Describe("Converter", func() { event, ok := con.Events["NewOwner"] Expect(ok).To(Equal(true)) - c := converter.NewConverter(con) + c := converter.Converter{} + c.Update(con) _, err := c.Convert([]types.Log{mocks.MockNewOwnerLog1, mocks.MockNewOwnerLog2}, event, 232) Expect(err).ToNot(HaveOccurred()) Expect(len(con.EmittedHashes)).To(Equal(3)) @@ -143,7 +147,8 @@ var _ = Describe("Converter", func() { It("Fails with an empty contract", func() { event := con.Events["Transfer"] - c := converter.NewConverter(&contract.Contract{}) + c := converter.Converter{} + c.Update(&contract.Contract{}) _, err = c.Convert([]types.Log{mocks.MockTransferLog1}, event, 232) Expect(err).To(HaveOccurred()) }) diff --git a/pkg/contract_watcher/light/repository/header_repository_test.go b/pkg/contract_watcher/light/repository/header_repository_test.go index 1ca2e1ac..02a94bbf 100644 --- a/pkg/contract_watcher/light/repository/header_repository_test.go +++ b/pkg/contract_watcher/light/repository/header_repository_test.go @@ -33,8 +33,8 @@ import ( var _ = Describe("Repository", func() { var db *postgres.DB var bc core.BlockChain - var omniHeaderRepo repository.HeaderRepository // omni/light header repository - var coreHeaderRepo repositories.HeaderRepository // pkg/datastore header repository + var contractHeaderRepo repository.HeaderRepository // contract_watcher light header repository + var coreHeaderRepo repositories.HeaderRepository // pkg/datastore header repository var eventIDs = []string{ "eventName_contractAddr", "eventName_contractAddr2", @@ -48,7 +48,7 @@ var _ = Describe("Repository", func() { BeforeEach(func() { db, bc = test_helpers.SetupDBandBC() - omniHeaderRepo = repository.NewHeaderRepository(db) + contractHeaderRepo = repository.NewHeaderRepository(db) coreHeaderRepo = repositories.NewHeaderRepository(db) }) @@ -62,7 +62,7 @@ var _ = Describe("Repository", func() { _, err := db.Exec(query) Expect(err).To(HaveOccurred()) - err = omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err = contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) _, err = db.Exec(query) @@ -70,13 +70,13 @@ var _ = Describe("Repository", func() { }) It("Caches column it creates so that it does not need to repeatedly query the database to check for it's existence", func() { - _, ok := omniHeaderRepo.CheckCache(eventIDs[0]) + _, ok := contractHeaderRepo.CheckCache(eventIDs[0]) Expect(ok).To(Equal(false)) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - v, ok := omniHeaderRepo.CheckCache(eventIDs[0]) + v, ok := contractHeaderRepo.CheckCache(eventIDs[0]) Expect(ok).To(Equal(true)) Expect(v).To(Equal(true)) }) @@ -89,7 +89,7 @@ var _ = Describe("Repository", func() { Expect(err).To(HaveOccurred()) } - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) for _, id := range eventIDs { @@ -100,15 +100,15 @@ var _ = Describe("Repository", func() { It("Caches columns it creates so that it does not need to repeatedly query the database to check for it's existence", func() { for _, id := range eventIDs { - _, ok := omniHeaderRepo.CheckCache(id) + _, ok := contractHeaderRepo.CheckCache(id) Expect(ok).To(Equal(false)) } - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) for _, id := range eventIDs { - v, ok := omniHeaderRepo.CheckCache(id) + v, ok := contractHeaderRepo.CheckCache(id) Expect(ok).To(Equal(true)) Expect(v).To(Equal(true)) } @@ -118,20 +118,20 @@ var _ = Describe("Repository", func() { Describe("MissingHeaders", func() { It("Returns all unchecked headers for the given eventID", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) }) It("Returns unchecked headers in ascending order", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) @@ -145,10 +145,10 @@ var _ = Describe("Repository", func() { It("Returns only contiguous chunks of headers", func() { addDiscontinuousHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(2)) Expect(missingHeaders[0].BlockNumber).To(Equal(int64(6194632))) @@ -157,10 +157,10 @@ var _ = Describe("Repository", func() { It("Fails if eventID does not yet exist in check_headers table", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - _, err = omniHeaderRepo.MissingHeaders(6194632, 6194635, "notEventId") + _, err = contractHeaderRepo.MissingHeaders(6194632, 6194635, "notEventId") Expect(err).To(HaveOccurred()) }) }) @@ -168,36 +168,36 @@ var _ = Describe("Repository", func() { Describe("MissingHeadersForAll", func() { // HERE It("Returns all headers that have not been checked for all of the ids provided", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) + missingHeaders, err := contractHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) - err = omniHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[0]) + err = contractHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) + missingHeaders, err = contractHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) - err = omniHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[1]) + err = contractHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[1]) Expect(err).ToNot(HaveOccurred()) - err = omniHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[2]) + err = contractHeaderRepo.MarkHeaderChecked(missingHeaders[0].Id, eventIDs[2]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeadersForAll(6194633, 6194635, eventIDs) + missingHeaders, err = contractHeaderRepo.MissingHeadersForAll(6194633, 6194635, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(2)) }) It("Returns only contiguous chunks of headers", func() { addDiscontinuousHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) + missingHeaders, err := contractHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(2)) Expect(missingHeaders[0].BlockNumber).To(Equal(int64(6194632))) @@ -206,10 +206,10 @@ var _ = Describe("Repository", func() { It("Returns at most 100 headers", func() { add102Headers(coreHeaderRepo, bc) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeadersForAll(6194632, 6194733, eventIDs) + missingHeaders, err := contractHeaderRepo.MissingHeadersForAll(6194632, 6194733, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(100)) Expect(missingHeaders[0].BlockNumber).To(Equal(int64(6194632))) @@ -218,11 +218,11 @@ var _ = Describe("Repository", func() { It("Fails if one of the eventIDs does not yet exist in check_headers table", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) badEventIDs := append(eventIDs, "notEventId") - _, err = omniHeaderRepo.MissingHeadersForAll(6194632, 6194635, badEventIDs) + _, err = contractHeaderRepo.MissingHeadersForAll(6194632, 6194635, badEventIDs) Expect(err).To(HaveOccurred()) }) }) @@ -230,36 +230,36 @@ var _ = Describe("Repository", func() { Describe("MarkHeaderChecked", func() { It("Marks the header checked for the given eventID", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) headerID := missingHeaders[0].Id - err = omniHeaderRepo.MarkHeaderChecked(headerID, eventIDs[0]) + err = contractHeaderRepo.MarkHeaderChecked(headerID, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeaders(6194633, 6194635, eventIDs[0]) + missingHeaders, err = contractHeaderRepo.MissingHeaders(6194633, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(2)) }) It("Fails if eventID does not yet exist in check_headers table", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumn(eventIDs[0]) + err := contractHeaderRepo.AddCheckColumn(eventIDs[0]) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) headerID := missingHeaders[0].Id - err = omniHeaderRepo.MarkHeaderChecked(headerID, "notEventId") + err = contractHeaderRepo.MarkHeaderChecked(headerID, "notEventId") Expect(err).To(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err = contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) }) @@ -268,18 +268,18 @@ var _ = Describe("Repository", func() { Describe("MarkHeaderCheckedForAll", func() { It("Marks the header checked for all provided column ids", func() { addHeaders(coreHeaderRepo) - err := omniHeaderRepo.AddCheckColumns(eventIDs) + err := contractHeaderRepo.AddCheckColumns(eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err := omniHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) + missingHeaders, err := contractHeaderRepo.MissingHeadersForAll(6194632, 6194635, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) headerID := missingHeaders[0].Id - err = omniHeaderRepo.MarkHeaderCheckedForAll(headerID, eventIDs) + err = contractHeaderRepo.MarkHeaderCheckedForAll(headerID, eventIDs) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeaders(6194633, 6194635, eventIDs[0]) + missingHeaders, err = contractHeaderRepo.MissingHeaders(6194633, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(2)) }) @@ -296,17 +296,17 @@ var _ = Describe("Repository", func() { var missingHeaders []core.Header for _, id := range methodIDs { - err := omniHeaderRepo.AddCheckColumn(id) + err := contractHeaderRepo.AddCheckColumn(id) Expect(err).ToNot(HaveOccurred()) - missingHeaders, err = omniHeaderRepo.MissingHeaders(6194632, 6194635, id) + missingHeaders, err = contractHeaderRepo.MissingHeaders(6194632, 6194635, id) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) } - err := omniHeaderRepo.MarkHeadersCheckedForAll(missingHeaders, methodIDs) + err := contractHeaderRepo.MarkHeadersCheckedForAll(missingHeaders, methodIDs) Expect(err).ToNot(HaveOccurred()) for _, id := range methodIDs { - missingHeaders, err = omniHeaderRepo.MissingHeaders(6194632, 6194635, id) + missingHeaders, err = contractHeaderRepo.MissingHeaders(6194632, 6194635, id) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(0)) } @@ -317,28 +317,28 @@ var _ = Describe("Repository", func() { It("Returns headers that have been checked for all the provided events but have not been checked for all the provided methods", func() { addHeaders(coreHeaderRepo) for i, id := range eventIDs { - err := omniHeaderRepo.AddCheckColumn(id) + err := contractHeaderRepo.AddCheckColumn(id) Expect(err).ToNot(HaveOccurred()) - err = omniHeaderRepo.AddCheckColumn(methodIDs[i]) + err = contractHeaderRepo.AddCheckColumn(methodIDs[i]) Expect(err).ToNot(HaveOccurred()) } - missingHeaders, err := omniHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) + missingHeaders, err := contractHeaderRepo.MissingHeaders(6194632, 6194635, eventIDs[0]) Expect(err).ToNot(HaveOccurred()) Expect(len(missingHeaders)).To(Equal(3)) headerID := missingHeaders[0].Id headerID2 := missingHeaders[1].Id for i, id := range eventIDs { - err = omniHeaderRepo.MarkHeaderChecked(headerID, id) + err = contractHeaderRepo.MarkHeaderChecked(headerID, id) Expect(err).ToNot(HaveOccurred()) - err = omniHeaderRepo.MarkHeaderChecked(headerID2, id) + err = contractHeaderRepo.MarkHeaderChecked(headerID2, id) Expect(err).ToNot(HaveOccurred()) - err = omniHeaderRepo.MarkHeaderChecked(headerID, methodIDs[i]) + err = contractHeaderRepo.MarkHeaderChecked(headerID, methodIDs[i]) Expect(err).ToNot(HaveOccurred()) } - intersectionHeaders, err := omniHeaderRepo.MissingMethodsCheckedEventsIntersection(6194632, 6194635, methodIDs, eventIDs) + intersectionHeaders, err := contractHeaderRepo.MissingMethodsCheckedEventsIntersection(6194632, 6194635, methodIDs, eventIDs) Expect(err).ToNot(HaveOccurred()) Expect(len(intersectionHeaders)).To(Equal(1)) Expect(intersectionHeaders[0].Id).To(Equal(headerID2)) diff --git a/pkg/contract_watcher/light/retriever/retriever_suite_test.go b/pkg/contract_watcher/light/retriever/retriever_suite_test.go index 5a254ef2..83a16269 100644 --- a/pkg/contract_watcher/light/retriever/retriever_suite_test.go +++ b/pkg/contract_watcher/light/retriever/retriever_suite_test.go @@ -27,7 +27,7 @@ import ( func TestRetriever(t *testing.T) { RegisterFailHandler(Fail) - RunSpecs(t, "Light BLock Number Retriever Suite Test") + RunSpecs(t, "Light Block Number Retriever Suite Test") } var _ = BeforeSuite(func() { diff --git a/pkg/contract_watcher/light/transformer/transformer.go b/pkg/contract_watcher/light/transformer/transformer.go index 080ca040..7753e286 100644 --- a/pkg/contract_watcher/light/transformer/transformer.go +++ b/pkg/contract_watcher/light/transformer/transformer.go @@ -38,7 +38,7 @@ import ( ) // Requires a light synced vDB (headers) and a running eth node (or infura) -type transformer struct { +type Transformer struct { // Database interfaces srep.EventRepository // Holds transformed watched event log data repository.HeaderRepository // Interface for interaction with header repositories @@ -48,9 +48,9 @@ type transformer struct { retriever.BlockRetriever // Retrieves first block for contract and current block height // Processing interfaces - fetcher.Fetcher // Fetches event logs, using header hashes - converter.Converter // Converts watched event logs into custom log - poller.Poller // Polls methods using arguments collected from events and persists them using a method datastore + fetcher.Fetcher // Fetches event logs, using header hashes + converter.ConverterInterface // Converts watched event logs into custom log + poller.Poller // Polls methods using arguments collected from events and persists them using a method datastore // Store contract configuration information Config config.ContractConfig @@ -74,18 +74,18 @@ type transformer struct { // 4. Execute // Transformer takes in config for blockchain, database, and network id -func NewTransformer(con config.ContractConfig, bc core.BlockChain, db *postgres.DB) *transformer { +func NewTransformer(con config.ContractConfig, bc core.BlockChain, db *postgres.DB) *Transformer { - return &transformer{ - Poller: poller.NewPoller(bc, db, types.LightSync), - Fetcher: fetcher.NewFetcher(bc), - Parser: parser.NewParser(con.Network), - HeaderRepository: repository.NewHeaderRepository(db), - BlockRetriever: retriever.NewBlockRetriever(db), - Converter: converter.NewConverter(&contract.Contract{}), - Contracts: map[string]*contract.Contract{}, - EventRepository: srep.NewEventRepository(db, types.LightSync), - Config: con, + return &Transformer{ + Poller: poller.NewPoller(bc, db, types.LightSync), + Fetcher: fetcher.NewFetcher(bc), + Parser: parser.NewParser(con.Network), + HeaderRepository: repository.NewHeaderRepository(db), + BlockRetriever: retriever.NewBlockRetriever(db), + ConverterInterface: &converter.Converter{}, + Contracts: map[string]*contract.Contract{}, + EventRepository: srep.NewEventRepository(db, types.LightSync), + Config: con, } } @@ -93,7 +93,7 @@ func NewTransformer(con config.ContractConfig, bc core.BlockChain, db *postgres. // Loops over all of the addr => filter sets // Uses parser to pull event info from abi // Use this info to generate event filters -func (tr *transformer) Init() error { +func (tr *Transformer) Init() error { // Initialize internally configured transformer settings tr.contractAddresses = make([]string, 0) // Holds all contract addresses, for batch fetching of logs tr.sortedEventIds = make(map[string][]string) // Map to sort event column ids by contract, for post fetch processing and persisting of logs @@ -124,10 +124,6 @@ func (tr *transformer) Init() error { if err != nil { return err } - lastBlock, err := tr.BlockRetriever.RetrieveMostRecentBlock() - if err != nil { - return err - } // Set to specified range if it falls within the bounds if firstBlock < tr.Config.StartingBlocks[contractAddr] { @@ -136,7 +132,7 @@ func (tr *transformer) Init() error { // Get contract name if it has one var name = new(string) - tr.FetchContractData(tr.Abi(), contractAddr, "name", nil, &name, lastBlock) + tr.Poller.FetchContractData(tr.Abi(), contractAddr, "name", nil, name, -1) // Remove any potential accidental duplicate inputs eventArgs := map[string]bool{} @@ -201,7 +197,7 @@ func (tr *transformer) Init() error { return nil } -func (tr *transformer) Execute() error { +func (tr *Transformer) Execute() error { if len(tr.Contracts) == 0 { return errors.New("error: transformer has no initialized contracts") } @@ -253,10 +249,10 @@ func (tr *transformer) Execute() error { } // Configure converter with this contract con := tr.Contracts[conAddr] - tr.Converter.Update(con) + tr.ConverterInterface.Update(con) // Convert logs into batches of log mappings (eventName => []types.Logs - convertedLogs, err := tr.Converter.ConvertBatch(logs, con.Events, header.Id) + convertedLogs, err := tr.ConverterInterface.ConvertBatch(logs, con.Events, header.Id) if err != nil { return err } @@ -293,7 +289,7 @@ func (tr *transformer) Execute() error { } // Used to poll contract methods at a given header -func (tr *transformer) methodPolling(header core.Header, sortedMethodIds map[string][]string) error { +func (tr *Transformer) methodPolling(header core.Header, sortedMethodIds map[string][]string) error { for _, con := range tr.Contracts { // Skip method polling processes if no methods are specified // Also don't try to poll methods below this contract's specified starting block @@ -317,6 +313,6 @@ func (tr *transformer) methodPolling(header core.Header, sortedMethodIds map[str return nil } -func (tr *transformer) GetConfig() config.ContractConfig { +func (tr *Transformer) GetConfig() config.ContractConfig { return tr.Config } diff --git a/pkg/contract_watcher/light/transformer/transformer_test.go b/pkg/contract_watcher/light/transformer/transformer_test.go index 9fe3d3a1..98a31722 100644 --- a/pkg/contract_watcher/light/transformer/transformer_test.go +++ b/pkg/contract_watcher/light/transformer/transformer_test.go @@ -17,425 +17,69 @@ package transformer_test import ( - "fmt" - "github.com/vulcanize/vulcanizedb/pkg/config" - "strings" - - "github.com/ethereum/go-ethereum/common" . "github.com/onsi/ginkgo" . "github.com/onsi/gomega" - + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/light/retriever" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/light/transformer" - "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/constants" - "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/contract" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/helpers/test_helpers/mocks" - "github.com/vulcanize/vulcanizedb/pkg/core" - "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres" - "github.com/vulcanize/vulcanizedb/pkg/datastore/postgres/repositories" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/parser" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/poller" + "github.com/vulcanize/vulcanizedb/pkg/fakes" ) var _ = Describe("Transformer", func() { - var db *postgres.DB - var err error - var blockChain core.BlockChain - var headerRepository repositories.HeaderRepository - var headerID int64 - var ensAddr = strings.ToLower(constants.EnsContractAddress) - var tusdAddr = strings.ToLower(constants.TusdContractAddress) - - BeforeEach(func() { - db, blockChain = test_helpers.SetupDBandBC() - headerRepository = repositories.NewHeaderRepository(db) - }) - - AfterEach(func() { - test_helpers.TearDown(db) - }) - + var fakeAddress = "0x1234567890abcdef" Describe("Init", func() { It("Initializes transformer's contract objects", func() { - headerRepository.CreateOrUpdateHeader(mocks.MockHeader1) - headerRepository.CreateOrUpdateHeader(mocks.MockHeader3) - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() + blockRetriever := &fakes.MockLightBlockRetriever{} + firstBlock := int64(1) + blockRetriever.FirstBlock = firstBlock + + parsr := &fakes.MockParser{} + fakeAbi := "fake_abi" + parsr.AbiToReturn = fakeAbi + + pollr := &fakes.MockPoller{} + fakeContractName := "fake_contract_name" + pollr.ContractName = fakeContractName + + t := getFakeTransformer(blockRetriever, parsr, pollr) + + err := t.Init() + Expect(err).ToNot(HaveOccurred()) - c, ok := t.Contracts[tusdAddr] + c, ok := t.Contracts[fakeAddress] Expect(ok).To(Equal(true)) - Expect(c.StartingBlock).To(Equal(int64(6194632))) + Expect(c.StartingBlock).To(Equal(firstBlock)) Expect(c.LastBlock).To(Equal(int64(-1))) - Expect(c.Abi).To(Equal(constants.TusdAbiString)) - Expect(c.Name).To(Equal("TrueUSD")) - Expect(c.Address).To(Equal(tusdAddr)) + Expect(c.Abi).To(Equal(fakeAbi)) + Expect(c.Name).To(Equal(fakeContractName)) + Expect(c.Address).To(Equal(fakeAddress)) }) - It("Fails to initialize if first and most recent block numbers cannot be fetched from vDB headers table", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() - Expect(err).To(HaveOccurred()) - }) - - It("Does nothing if nothing if no addresses are configured", func() { - headerRepository.CreateOrUpdateHeader(mocks.MockHeader1) - headerRepository.CreateOrUpdateHeader(mocks.MockHeader3) - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Addresses = nil - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - - _, ok := t.Contracts[tusdAddr] - Expect(ok).To(Equal(false)) - }) - }) - - Describe("Execute- against TrueUSD contract", func() { - BeforeEach(func() { - header1, err := blockChain.GetHeaderByNumber(6791668) - Expect(err).ToNot(HaveOccurred()) - header2, err := blockChain.GetHeaderByNumber(6791669) - Expect(err).ToNot(HaveOccurred()) - header3, err := blockChain.GetHeaderByNumber(6791670) - Expect(err).ToNot(HaveOccurred()) - headerRepository.CreateOrUpdateHeader(header1) - headerID, err = headerRepository.CreateOrUpdateHeader(header2) - Expect(err).ToNot(HaveOccurred()) - headerRepository.CreateOrUpdateHeader(header3) - }) - - It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.LightTransferLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.transfer_event", tusdAddr)).StructScan(&log) - Expect(err).ToNot(HaveOccurred()) - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(log.HeaderID).To(Equal(headerID)) - Expect(log.From).To(Equal("0x1062a747393198f70F71ec65A582423Dba7E5Ab3")) - Expect(log.To).To(Equal("0x2930096dB16b4A44Ecd4084EA4bd26F7EeF1AEf0")) - Expect(log.Value).To(Equal("9998940000000000000000")) - }) - - It("Keeps track of contract-related addresses while transforming event data if they need to be used for later method polling", func() { - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Methods = map[string][]string{ - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - c, ok := t.Contracts[tusdAddr] - Expect(ok).To(Equal(true)) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - Expect(len(c.EmittedAddrs)).To(Equal(4)) - Expect(len(c.EmittedHashes)).To(Equal(0)) - - b, ok := c.EmittedAddrs[common.HexToAddress("0x1062a747393198f70F71ec65A582423Dba7E5Ab3")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedAddrs[common.HexToAddress("0x2930096dB16b4A44Ecd4084EA4bd26F7EeF1AEf0")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedAddrs[common.HexToAddress("0x571A326f5B15E16917dC17761c340c1ec5d06f6d")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedAddrs[common.HexToAddress("0xFBb1b73C4f0BDa4f67dcA266ce6Ef42f520fBB98")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x09BbBBE21a5975cAc061D82f7b843b1234567890")] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x")] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[""] - Expect(ok).To(Equal(false)) - - _, ok = c.EmittedAddrs[common.HexToAddress("0x09THISE21a5IS5cFAKE1D82fAND43bCE06MADEUP")] - Expect(ok).To(Equal(false)) - }) - - It("Polls given methods using generated token holder address", func() { - var testConf config.ContractConfig - testConf = mocks.TusdConfig - testConf.Methods = map[string][]string{ - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.BalanceOf{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x1062a747393198f70F71ec65A582423Dba7E5Ab3' AND block = '6791669'", tusdAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Balance).To(Equal("55849938025000000000000")) - Expect(res.TokenName).To(Equal("TrueUSD")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843b1234567890' AND block = '6791669'", tusdAddr)).StructScan(&res) - Expect(err).To(HaveOccurred()) - }) - - It("Fails if initialization has not been done", func() { - t := transformer.NewTransformer(mocks.TusdConfig, blockChain, db) - err = t.Execute() - Expect(err).To(HaveOccurred()) - }) - }) - - Describe("Execute- against ENS registry contract", func() { - BeforeEach(func() { - header1, err := blockChain.GetHeaderByNumber(6885695) - Expect(err).ToNot(HaveOccurred()) - header2, err := blockChain.GetHeaderByNumber(6885696) - Expect(err).ToNot(HaveOccurred()) - header3, err := blockChain.GetHeaderByNumber(6885697) - Expect(err).ToNot(HaveOccurred()) - headerRepository.CreateOrUpdateHeader(header1) - headerID, err = headerRepository.CreateOrUpdateHeader(header2) - Expect(err).ToNot(HaveOccurred()) - headerRepository.CreateOrUpdateHeader(header3) - }) - - It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSConfig, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.LightNewOwnerLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.newowner_event", ensAddr)).StructScan(&log) - Expect(err).ToNot(HaveOccurred()) - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(log.HeaderID).To(Equal(headerID)) - Expect(log.Node).To(Equal("0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae")) - Expect(log.Label).To(Equal("0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047")) - Expect(log.Owner).To(Equal("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")) - }) - - It("Keeps track of contract-related hashes while transforming event data if they need to be used for later method polling", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - c, ok := t.Contracts[ensAddr] - Expect(ok).To(Equal(true)) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - Expect(len(c.EmittedHashes)).To(Equal(2)) - Expect(len(c.EmittedAddrs)).To(Equal(0)) - - b, ok := c.EmittedHashes[common.HexToHash("0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = c.EmittedHashes[common.HexToHash("0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - // Doesn't keep track of address since it wouldn't be used in calling the 'owner' method - _, ok = c.EmittedAddrs[common.HexToAddress("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")] - Expect(ok).To(Equal(false)) - }) - - It("Polls given method using list of collected hashes", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.Owner{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae' AND block = '6885696'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047' AND block = '6885696'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x0000000000000000000000000000000000000000")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x9THIS110dcc444fIS242510c09bbAbe21aFAKEcacNODE82f7b843HASH61ba391' AND block = '6885696'", ensAddr)).StructScan(&res) - Expect(err).To(HaveOccurred()) - }) - - It("It does not persist events if they do not pass the emitted arg filter", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.EventArgs = map[string][]string{ - ensAddr: {"fake_filter_value"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - log := test_helpers.LightNewOwnerLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.newowner_event", ensAddr)).StructScan(&log) - Expect(err).To(HaveOccurred()) - }) - - It("If a method arg filter is applied, only those arguments are used in polling", func() { - var testConf config.ContractConfig - testConf = mocks.ENSConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - } - testConf.MethodArgs = map[string][]string{ - ensAddr: {"0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - res := test_helpers.Owner{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae' AND block = '6885696'", ensAddr)).StructScan(&res) - Expect(err).ToNot(HaveOccurred()) - Expect(res.Address).To(Equal("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")) - Expect(res.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047' AND block = '6885696'", ensAddr)).StructScan(&res) - Expect(err).To(HaveOccurred()) - }) - }) - - Describe("Execute- against both ENS and TrueUSD", func() { - BeforeEach(func() { - for i := 6885692; i < 6885702; i++ { - header, err := blockChain.GetHeaderByNumber(int64(i)) - Expect(err).ToNot(HaveOccurred()) - _, err = headerRepository.CreateOrUpdateHeader(header) - Expect(err).ToNot(HaveOccurred()) - } - }) - - It("Transforms watched contract data into custom repositories", func() { - t := transformer.NewTransformer(mocks.ENSandTusdConfig, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - newOwnerLog := test_helpers.LightNewOwnerLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.newowner_event", ensAddr)).StructScan(&newOwnerLog) - Expect(err).ToNot(HaveOccurred()) - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(newOwnerLog.Node).To(Equal("0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae")) - Expect(newOwnerLog.Label).To(Equal("0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047")) - Expect(newOwnerLog.Owner).To(Equal("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")) - - transferLog := test_helpers.LightTransferLog{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.transfer_event", tusdAddr)).StructScan(&transferLog) - Expect(err).ToNot(HaveOccurred()) - // We don't know vulcID, so compare individual fields instead of complete structures - Expect(transferLog.From).To(Equal("0x8cA465764873E71CEa525F5EB6AE973d650c22C2")) - Expect(transferLog.To).To(Equal("0xc338482360651E5D30BEd77b7c85358cbBFB2E0e")) - Expect(transferLog.Value).To(Equal("2800000000000000000000")) - }) - - It("Keeps track of contract-related hashes and addresses while transforming event data if they need to be used for later method polling", func() { - var testConf config.ContractConfig - testConf = mocks.ENSandTusdConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - ens, ok := t.Contracts[ensAddr] - Expect(ok).To(Equal(true)) - tusd, ok := t.Contracts[tusdAddr] - Expect(ok).To(Equal(true)) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - Expect(len(ens.EmittedHashes)).To(Equal(2)) - Expect(len(ens.EmittedAddrs)).To(Equal(0)) - Expect(len(tusd.EmittedAddrs)).To(Equal(2)) - Expect(len(tusd.EmittedHashes)).To(Equal(0)) - - b, ok := ens.EmittedHashes[common.HexToHash("0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = ens.EmittedHashes[common.HexToHash("0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = tusd.EmittedAddrs[common.HexToAddress("0x8cA465764873E71CEa525F5EB6AE973d650c22C2")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - b, ok = tusd.EmittedAddrs[common.HexToAddress("0xc338482360651E5D30BEd77b7c85358cbBFB2E0e")] - Expect(ok).To(Equal(true)) - Expect(b).To(Equal(true)) - - _, ok = tusd.EmittedAddrs[common.HexToAddress("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")] - Expect(ok).To(Equal(false)) - }) - - It("Polls given methods for each contract, using list of collected values", func() { - var testConf config.ContractConfig - testConf = mocks.ENSandTusdConfig - testConf.Methods = map[string][]string{ - ensAddr: {"owner"}, - tusdAddr: {"balanceOf"}, - } - t := transformer.NewTransformer(testConf, blockChain, db) - err = t.Init() - Expect(err).ToNot(HaveOccurred()) - err = t.Execute() - Expect(err).ToNot(HaveOccurred()) - - owner := test_helpers.Owner{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae' AND block = '6885696'", ensAddr)).StructScan(&owner) - Expect(err).ToNot(HaveOccurred()) - Expect(owner.Address).To(Equal("0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef")) - Expect(owner.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047' AND block = '6885696'", ensAddr)).StructScan(&owner) - Expect(err).ToNot(HaveOccurred()) - Expect(owner.Address).To(Equal("0x0000000000000000000000000000000000000000")) - Expect(owner.TokenName).To(Equal("")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.owner_method WHERE node_ = '0x95832c7a47ff8a7840e28b78ceMADEUPaaf4HASHc186badTHItransformers.8IS625bFAKE' AND block = '6885696'", ensAddr)).StructScan(&owner) - Expect(err).To(HaveOccurred()) - - bal := test_helpers.BalanceOf{} - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x8cA465764873E71CEa525F5EB6AE973d650c22C2' AND block = '6885701'", tusdAddr)).StructScan(&bal) - Expect(err).ToNot(HaveOccurred()) - Expect(bal.Balance).To(Equal("1954436000000000000000")) - Expect(bal.TokenName).To(Equal("TrueUSD")) - - err = db.QueryRowx(fmt.Sprintf("SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843b1234567890' AND block = '6885701'", tusdAddr)).StructScan(&bal) + It("Fails to initialize if first block cannot be fetched from vDB headers table", func() { + blockRetriever := &fakes.MockLightBlockRetriever{} + blockRetriever.FirstBlockErr = fakes.FakeError + t := getFakeTransformer(blockRetriever, &fakes.MockParser{}, &fakes.MockPoller{}) + + err := t.Init() + Expect(err).To(HaveOccurred()) + Expect(err).To(MatchError(fakes.FakeError)) }) }) }) + +func getFakeTransformer(blockRetriever retriever.BlockRetriever, parsr parser.Parser, pollr poller.Poller) transformer.Transformer { + return transformer.Transformer{ + Parser: parsr, + BlockRetriever: blockRetriever, + Poller: pollr, + HeaderRepository: &fakes.MockLightHeaderRepository{}, + Contracts: map[string]*contract.Contract{}, + Config: mocks.MockConfig, + } +} diff --git a/pkg/contract_watcher/shared/contract/contract.go b/pkg/contract_watcher/shared/contract/contract.go index d6f62d0a..b3eb3b7f 100644 --- a/pkg/contract_watcher/shared/contract/contract.go +++ b/pkg/contract_watcher/shared/contract/contract.go @@ -67,7 +67,7 @@ func (c Contract) Init() *Contract { return &c } -// Use contract info to generate event filters - full sync omni watcher only +// Use contract info to generate event filters - full sync contract watcher only func (c *Contract) GenerateFilters() error { c.Filters = map[string]filters.LogFilter{} diff --git a/pkg/contract_watcher/shared/helpers/test_helpers/mocks/entities.go b/pkg/contract_watcher/shared/helpers/test_helpers/mocks/entities.go index e65db708..f9224847 100644 --- a/pkg/contract_watcher/shared/helpers/test_helpers/mocks/entities.go +++ b/pkg/contract_watcher/shared/helpers/test_helpers/mocks/entities.go @@ -18,13 +18,13 @@ package mocks import ( "encoding/json" - "github.com/vulcanize/vulcanizedb/pkg/config" "strings" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common/hexutil" "github.com/ethereum/go-ethereum/core/types" + "github.com/vulcanize/vulcanizedb/pkg/config" "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/constants" "github.com/vulcanize/vulcanizedb/pkg/core" "github.com/vulcanize/vulcanizedb/pkg/filters" @@ -262,74 +262,46 @@ var MockNewOwnerLog2 = types.Log{ var ens = strings.ToLower(constants.EnsContractAddress) var tusd = strings.ToLower(constants.TusdContractAddress) -var TusdConfig = config.ContractConfig{ +var MockConfig = config.ContractConfig{ Network: "", Addresses: map[string]bool{ - tusd: true, + "0x1234567890abcdef": true, }, Abis: map[string]string{ - tusd: "", + "0x1234567890abcdef": "fake_abi", }, Events: map[string][]string{ - tusd: []string{"Transfer"}, + "0x1234567890abcdef": []string{"Transfer"}, }, Methods: map[string][]string{ - tusd: nil, + "0x1234567890abcdef": nil, }, MethodArgs: map[string][]string{ - tusd: nil, + "0x1234567890abcdef": nil, }, EventArgs: map[string][]string{ - tusd: nil, + "0x1234567890abcdef": nil, }, } -var ENSConfig = config.ContractConfig{ +var MockEmptyConfig = config.ContractConfig{ Network: "", Addresses: map[string]bool{ - ens: true, + "0x1234567890abcdef": true, }, Abis: map[string]string{ - ens: "", + "0x1234567890abcdef": "fake_abi", }, Events: map[string][]string{ - ens: []string{"NewOwner"}, + "0x1234567890abcdef": nil, }, Methods: map[string][]string{ - ens: nil, + "0x1234567890abcdef": nil, }, MethodArgs: map[string][]string{ - ens: nil, + "0x1234567890abcdef": nil, }, EventArgs: map[string][]string{ - ens: nil, - }, -} - -var ENSandTusdConfig = config.ContractConfig{ - Network: "", - Addresses: map[string]bool{ - ens: true, - tusd: true, - }, - Abis: map[string]string{ - ens: "", - tusd: "", - }, - Events: map[string][]string{ - ens: []string{"NewOwner"}, - tusd: []string{"Transfer"}, - }, - Methods: map[string][]string{ - ens: nil, - tusd: nil, - }, - MethodArgs: map[string][]string{ - ens: nil, - tusd: nil, - }, - EventArgs: map[string][]string{ - ens: nil, - tusd: nil, + "0x1234567890abcdef": nil, }, } diff --git a/pkg/contract_watcher/shared/helpers/test_helpers/test_data.go b/pkg/contract_watcher/shared/helpers/test_helpers/test_data.go new file mode 100644 index 00000000..c52fd20c --- /dev/null +++ b/pkg/contract_watcher/shared/helpers/test_helpers/test_data.go @@ -0,0 +1,109 @@ +// VulcanizeDB +// Copyright © 2019 Vulcanize + +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. + +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. + +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . + +package test_helpers + +import ( + "strings" + + "github.com/vulcanize/vulcanizedb/pkg/config" + "github.com/vulcanize/vulcanizedb/pkg/contract_watcher/shared/constants" +) + +var ens = strings.ToLower(constants.EnsContractAddress) +var tusd = strings.ToLower(constants.TusdContractAddress) + +var TusdConfig = config.ContractConfig{ + Network: "", + Addresses: map[string]bool{ + tusd: true, + }, + Abis: map[string]string{ + tusd: "", + }, + Events: map[string][]string{ + tusd: []string{"Transfer"}, + }, + Methods: map[string][]string{ + tusd: nil, + }, + MethodArgs: map[string][]string{ + tusd: nil, + }, + EventArgs: map[string][]string{ + tusd: nil, + }, + StartingBlocks: map[string]int64{ + tusd: 5197514, + }, +} + +var ENSConfig = config.ContractConfig{ + Network: "", + Addresses: map[string]bool{ + ens: true, + }, + Abis: map[string]string{ + ens: "", + }, + Events: map[string][]string{ + ens: []string{"NewOwner"}, + }, + Methods: map[string][]string{ + ens: nil, + }, + MethodArgs: map[string][]string{ + ens: nil, + }, + EventArgs: map[string][]string{ + ens: nil, + }, + StartingBlocks: map[string]int64{ + ens: 3327417, + }, +} + +var ENSandTusdConfig = config.ContractConfig{ + Network: "", + Addresses: map[string]bool{ + ens: true, + tusd: true, + }, + Abis: map[string]string{ + ens: "", + tusd: "", + }, + Events: map[string][]string{ + ens: []string{"NewOwner"}, + tusd: []string{"Transfer"}, + }, + Methods: map[string][]string{ + ens: nil, + tusd: nil, + }, + MethodArgs: map[string][]string{ + ens: nil, + tusd: nil, + }, + EventArgs: map[string][]string{ + ens: nil, + tusd: nil, + }, + StartingBlocks: map[string]int64{ + ens: 3327417, + tusd: 5197514, + }, +} diff --git a/pkg/contract_watcher/shared/repository/event_repository_test.go b/pkg/contract_watcher/shared/repository/event_repository_test.go index d6c3bff9..44a4cd66 100644 --- a/pkg/contract_watcher/shared/repository/event_repository_test.go +++ b/pkg/contract_watcher/shared/repository/event_repository_test.go @@ -134,7 +134,8 @@ var _ = Describe("Repository", func() { Describe("PersistLogs", func() { BeforeEach(func() { - c := fc.NewConverter(con) + c := fc.Converter{} + c.Update(con) log, err = c.Convert(mockEvent, event) Expect(err).ToNot(HaveOccurred()) }) @@ -276,7 +277,8 @@ var _ = Describe("Repository", func() { headerRepository := repositories.NewHeaderRepository(db) headerID, err = headerRepository.CreateOrUpdateHeader(mocks.MockHeader1) Expect(err).ToNot(HaveOccurred()) - c := lc.NewConverter(con) + c := lc.Converter{} + c.Update(con) logs, err = c.Convert([]geth.Log{mockLog1, mockLog2}, event, headerID) Expect(err).ToNot(HaveOccurred()) }) diff --git a/pkg/contract_watcher/shared/retriever/address_retriever_test.go b/pkg/contract_watcher/shared/retriever/address_retriever_test.go index 7ff4ced7..c8aa6fff 100644 --- a/pkg/contract_watcher/shared/retriever/address_retriever_test.go +++ b/pkg/contract_watcher/shared/retriever/address_retriever_test.go @@ -64,7 +64,8 @@ var _ = Describe("Address Retriever Test", func() { err = info.GenerateFilters() Expect(err).ToNot(HaveOccurred()) - c := converter.NewConverter(info) + c := converter.Converter{} + c.Update(info) log, err = c.Convert(mockEvent, event) Expect(err).ToNot(HaveOccurred()) diff --git a/pkg/contract_watcher/shared/types/event.go b/pkg/contract_watcher/shared/types/event.go index 7033f4ed..2818cc8d 100644 --- a/pkg/contract_watcher/shared/types/event.go +++ b/pkg/contract_watcher/shared/types/event.go @@ -38,7 +38,7 @@ type Field struct { // Struct to hold instance of an event log data type Log struct { - Id int64 // VulcanizeIdLog for full sync and header ID for light sync omni watcher + Id int64 // VulcanizeIdLog for full sync and header ID for light sync contract watcher Values map[string]string // Map of event input names to their values // Used for full sync only diff --git a/pkg/contract_watcher/shared/types/mode.go b/pkg/contract_watcher/shared/types/mode.go index 62057a16..9f37a857 100644 --- a/pkg/contract_watcher/shared/types/mode.go +++ b/pkg/contract_watcher/shared/types/mode.go @@ -47,7 +47,7 @@ func (mode Mode) MarshalText() ([]byte, error) { case FullSync: return []byte("full"), nil default: - return nil, fmt.Errorf("omni watcher: unknown mode %d, want LightSync or FullSync", mode) + return nil, fmt.Errorf("contract watcher: unknown mode %d, want LightSync or FullSync", mode) } } @@ -58,7 +58,7 @@ func (mode *Mode) UnmarshalText(text []byte) error { case "full": *mode = FullSync default: - return fmt.Errorf(`omni watcher: unknown mode %q, want "light" or "full"`, text) + return fmt.Errorf(`contract watcher: unknown mode %q, want "light" or "full"`, text) } return nil } diff --git a/pkg/fakes/mock_parser.go b/pkg/fakes/mock_parser.go index e705f413..dd03bbab 100644 --- a/pkg/fakes/mock_parser.go +++ b/pkg/fakes/mock_parser.go @@ -15,8 +15,9 @@ func (*MockParser) Parse(contractAddr string) error { return nil } -func (*MockParser) ParseAbiStr(abiStr string) error { - panic("implement me") +func (m *MockParser) ParseAbiStr(abiStr string) error { + m.AbiToReturn = abiStr + return nil } func (parser *MockParser) Abi() string { diff --git a/pkg/plugin/writer/writer.go b/pkg/plugin/writer/writer.go index 33e8a84c..e888bc5b 100644 --- a/pkg/plugin/writer/writer.go +++ b/pkg/plugin/writer/writer.go @@ -74,7 +74,7 @@ func (w *writer) WritePlugin() error { f.Func().Params(Id("e").Id("exporter")).Id("Export").Params().Parens(List( Index().Qual("github.com/vulcanize/vulcanizedb/libraries/shared/transformer", "EventTransformerInitializer"), Index().Qual("github.com/vulcanize/vulcanizedb/libraries/shared/transformer", "StorageTransformerInitializer"), - Index().Qual("github.com/vulcanize/vulcanizedb/libraries/shared/transformer", "GenericTransformerInitializer"), + Index().Qual("github.com/vulcanize/vulcanizedb/libraries/shared/transformer", "ContractTransformerInitializer"), )).Block(Return( Index().Qual( "github.com/vulcanize/vulcanizedb/libraries/shared/transformer", @@ -84,7 +84,7 @@ func (w *writer) WritePlugin() error { "StorageTransformerInitializer").Values(code[config.EthStorage]...), Index().Qual( "github.com/vulcanize/vulcanizedb/libraries/shared/transformer", - "GenericTransformerInitializer").Values(code[config.EthGeneric]...))) // Exports the collected event and storage transformer initializers + "ContractTransformerInitializer").Values(code[config.EthContract]...))) // Exports the collected event and storage transformer initializers // Write code to destination file err = f.Save(goFile) @@ -104,8 +104,8 @@ func (w *writer) collectTransformers() (map[config.TransformerType][]Code, error code[config.EthEvent] = append(code[config.EthEvent], Qual(path, "EventTransformerInitializer")) case config.EthStorage: code[config.EthStorage] = append(code[config.EthStorage], Qual(path, "StorageTransformerInitializer")) - case config.EthGeneric: - code[config.EthGeneric] = append(code[config.EthGeneric], Qual(path, "GenericTransformerInitializer")) + case config.EthContract: + code[config.EthContract] = append(code[config.EthContract], Qual(path, "ContractTransformerInitializer")) default: return nil, errors.New(fmt.Sprintf("invalid transformer type %s", transformer.Type)) }