2018-11-23 18:12:24 +00:00
// VulcanizeDB
// Copyright © 2018 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 <http://www.gnu.org/licenses/>.
package transformer_test
import (
2018-11-24 04:26:07 +00:00
"fmt"
2018-12-07 15:38:46 +00:00
"github.com/ethereum/go-ethereum/common"
2018-11-23 18:12:24 +00:00
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"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/omni/light/transformer"
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/constants"
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/helpers/test_helpers"
2018-11-24 04:26:07 +00:00
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/helpers/test_helpers/mocks"
2018-11-23 18:12:24 +00:00
)
var _ = Describe ( "Transformer" , func ( ) {
var db * postgres . DB
var err error
var blockChain core . BlockChain
var headerRepository repositories . HeaderRepository
2018-11-24 04:26:07 +00:00
var headerID int64
2018-11-23 18:12:24 +00:00
BeforeEach ( func ( ) {
db , blockChain = test_helpers . SetupDBandBC ( )
headerRepository = repositories . NewHeaderRepository ( db )
} )
AfterEach ( func ( ) {
test_helpers . TearDown ( db )
} )
Describe ( "SetEvents" , func ( ) {
It ( "Sets which events to watch from the given contract address" , func ( ) {
watchedEvents := [ ] string { "Transfer" , "Mint" }
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , watchedEvents )
Expect ( t . WatchedEvents [ constants . TusdContractAddress ] ) . To ( Equal ( watchedEvents ) )
} )
} )
Describe ( "SetEventAddrs" , func ( ) {
It ( "Sets which account addresses to watch events for" , func ( ) {
eventAddrs := [ ] string { "test1" , "test2" }
t := transformer . NewTransformer ( "" , blockChain , db )
2018-12-07 15:38:46 +00:00
t . SetEventArgs ( constants . TusdContractAddress , eventAddrs )
Expect ( t . EventArgs [ constants . TusdContractAddress ] ) . To ( Equal ( eventAddrs ) )
2018-11-23 18:12:24 +00:00
} )
} )
Describe ( "SetMethods" , func ( ) {
It ( "Sets which methods to poll at the given contract address" , func ( ) {
watchedMethods := [ ] string { "balanceOf" , "totalSupply" }
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetMethods ( constants . TusdContractAddress , watchedMethods )
Expect ( t . WantedMethods [ constants . TusdContractAddress ] ) . To ( Equal ( watchedMethods ) )
} )
} )
Describe ( "SetMethodAddrs" , func ( ) {
It ( "Sets which account addresses to poll methods against" , func ( ) {
methodAddrs := [ ] string { "test1" , "test2" }
t := transformer . NewTransformer ( "" , blockChain , db )
2018-12-07 15:38:46 +00:00
t . SetMethodArgs ( constants . TusdContractAddress , methodAddrs )
Expect ( t . MethodArgs [ constants . TusdContractAddress ] ) . To ( Equal ( methodAddrs ) )
2018-11-23 18:12:24 +00:00
} )
} )
Describe ( "SetRange" , func ( ) {
It ( "Sets the block range that the contract should be watched within" , func ( ) {
rng := [ 2 ] int64 { 1 , 100000 }
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetRange ( constants . TusdContractAddress , rng )
Expect ( t . ContractRanges [ constants . TusdContractAddress ] ) . To ( Equal ( rng ) )
} )
} )
2018-12-14 17:52:02 +00:00
Describe ( "SetCreateAddrList" , func ( ) {
It ( "Sets the block range that the contract should be watched within" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetCreateAddrList ( constants . TusdContractAddress , true )
Expect ( t . CreateAddrList [ constants . TusdContractAddress ] ) . To ( Equal ( true ) )
} )
} )
Describe ( "SetCreateHashList" , func ( ) {
It ( "Sets the block range that the contract should be watched within" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetCreateHashList ( constants . TusdContractAddress , true )
Expect ( t . CreateHashList [ constants . TusdContractAddress ] ) . To ( Equal ( true ) )
} )
} )
2018-11-23 18:12:24 +00:00
Describe ( "Init" , func ( ) {
It ( "Initializes transformer's contract objects" , func ( ) {
2018-11-24 04:26:07 +00:00
headerRepository . CreateOrUpdateHeader ( mocks . MockHeader1 )
headerRepository . CreateOrUpdateHeader ( mocks . MockHeader3 )
2018-11-23 18:12:24 +00:00
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
c , ok := t . Contracts [ constants . TusdContractAddress ]
Expect ( ok ) . To ( Equal ( true ) )
Expect ( c . StartingBlock ) . To ( Equal ( int64 ( 6194632 ) ) )
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 ( constants . TusdContractAddress ) )
} )
2018-11-24 04:26:07 +00:00
It ( "Fails to initialize if first and most recent block numbers cannot be fetched from vDB headers table" , func ( ) {
2018-11-23 18:12:24 +00:00
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
err = t . Init ( )
Expect ( err ) . To ( HaveOccurred ( ) )
} )
It ( "Does nothing if watched events are unset" , func ( ) {
2018-11-24 04:26:07 +00:00
headerRepository . CreateOrUpdateHeader ( mocks . MockHeader1 )
headerRepository . CreateOrUpdateHeader ( mocks . MockHeader3 )
2018-11-23 18:12:24 +00:00
t := transformer . NewTransformer ( "" , blockChain , db )
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
_ , ok := t . Contracts [ constants . TusdContractAddress ]
Expect ( ok ) . To ( Equal ( false ) )
} )
} )
Describe ( "Execute" , func ( ) {
2018-11-24 04:26:07 +00:00
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 ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
t . SetMethods ( constants . TusdContractAddress , nil )
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" , constants . TusdContractAddress ) ) . StructScan ( & log )
// 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" ) )
} )
2018-12-07 15:38:46 +00:00
It ( "Keeps track of contract-related addresses while transforming event data if they need to be used for later method polling" , func ( ) {
2018-11-24 04:26:07 +00:00
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
2018-12-07 15:38:46 +00:00
t . SetMethods ( constants . TusdContractAddress , [ ] string { "balanceOf" } )
2018-11-24 04:26:07 +00:00
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
c , ok := t . Contracts [ constants . TusdContractAddress ]
Expect ( ok ) . To ( Equal ( true ) )
err = t . Execute ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
2018-11-23 18:12:24 +00:00
2018-12-07 15:38:46 +00:00
b , ok := c . EmittedAddrs [ common . HexToAddress ( "0x1062a747393198f70F71ec65A582423Dba7E5Ab3" ) ]
2018-11-24 04:26:07 +00:00
Expect ( ok ) . To ( Equal ( true ) )
Expect ( b ) . To ( Equal ( true ) )
2018-12-07 15:38:46 +00:00
b , ok = c . EmittedAddrs [ common . HexToAddress ( "0x2930096dB16b4A44Ecd4084EA4bd26F7EeF1AEf0" ) ]
2018-11-24 04:26:07 +00:00
Expect ( ok ) . To ( Equal ( true ) )
Expect ( b ) . To ( Equal ( true ) )
2018-12-07 15:38:46 +00:00
_ , 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 [ "" ]
2018-11-24 04:26:07 +00:00
Expect ( ok ) . To ( Equal ( false ) )
2018-12-07 15:38:46 +00:00
_ , ok = c . EmittedAddrs [ common . HexToAddress ( "0x09THISE21a5IS5cFAKE1D82fAND43bCE06MADEUP" ) ]
2018-11-24 04:26:07 +00:00
Expect ( ok ) . To ( Equal ( false ) )
} )
It ( "Polls given methods using generated token holder address" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
t . SetMethods ( constants . TusdContractAddress , [ ] string { "balanceOf" } )
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
err = t . Execute ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
res := test_helpers . BalanceOf { }
2018-12-14 17:52:02 +00:00
err = db . QueryRowx ( fmt . Sprintf ( "SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x1062a747393198f70F71ec65A582423Dba7E5Ab3' AND block = '6791669'" , constants . TusdContractAddress ) ) . StructScan ( & res )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
Expect ( res . Balance ) . To ( Equal ( "55849938025000000000000" ) )
Expect ( res . TokenName ) . To ( Equal ( "TrueUSD" ) )
2018-11-24 04:26:07 +00:00
2018-12-14 17:52:02 +00:00
err = db . QueryRowx ( fmt . Sprintf ( "SELECT * FROM light_%s.balanceof_method WHERE who_ = '0x09BbBBE21a5975cAc061D82f7b843b1234567890' AND block = '6791669'" , constants . TusdContractAddress ) ) . StructScan ( & res )
Expect ( err ) . To ( HaveOccurred ( ) )
} )
It ( "Fails if initialization has not been done" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . TusdContractAddress , [ ] string { "Transfer" } )
t . SetMethods ( constants . TusdContractAddress , nil )
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 ( "" , blockChain , db )
t . SetEvents ( constants . EnsContractAddress , [ ] string { "NewOwner" } )
t . SetMethods ( constants . EnsContractAddress , nil )
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" , constants . EnsContractAddress ) ) . StructScan ( & log )
// 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 ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . EnsContractAddress , [ ] string { "NewOwner" } )
t . SetMethods ( constants . EnsContractAddress , [ ] string { "owner" } )
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
c , ok := t . Contracts [ constants . EnsContractAddress ]
2018-12-07 15:38:46 +00:00
Expect ( ok ) . To ( Equal ( true ) )
2018-12-14 17:52:02 +00:00
err = t . Execute ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
Expect ( len ( c . EmittedHashes ) ) . To ( Equal ( 2 ) )
b , ok := c . EmittedHashes [ common . HexToHash ( "0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae" ) ]
2018-12-07 15:38:46 +00:00
Expect ( ok ) . To ( Equal ( true ) )
Expect ( b ) . To ( Equal ( true ) )
2018-12-14 17:52:02 +00:00
b , ok = c . EmittedHashes [ common . HexToHash ( "0x95832c7a47ff8a7840e28b78ce695797aaf402b1c186bad9eca28842625b5047" ) ]
2018-12-07 15:38:46 +00:00
Expect ( ok ) . To ( Equal ( true ) )
Expect ( b ) . To ( Equal ( true ) )
2018-12-14 17:52:02 +00:00
// 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 methods using generated token holder address" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . EnsContractAddress , [ ] string { "NewOwner" } )
t . SetMethods ( constants . EnsContractAddress , [ ] string { "owner" } )
err = t . Init ( )
2018-11-24 04:26:07 +00:00
Expect ( err ) . ToNot ( HaveOccurred ( ) )
2018-12-14 17:52:02 +00:00
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'" , constants . EnsContractAddress ) ) . 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'" , constants . EnsContractAddress ) ) . 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_ = '0x95832c7a47ff8a7840e28b78ceMADEUPaaf4HASHc186badTHIS288IS625bFAKE' AND block = '6885696'" , constants . EnsContractAddress ) ) . StructScan ( & res )
2018-11-24 04:26:07 +00:00
Expect ( err ) . To ( HaveOccurred ( ) )
} )
2018-12-18 17:31:21 +00:00
It ( "It does not persist events if they do not pass the emitted arg filter" , func ( ) {
2018-11-24 04:26:07 +00:00
t := transformer . NewTransformer ( "" , blockChain , db )
2018-12-14 17:52:02 +00:00
t . SetEvents ( constants . EnsContractAddress , [ ] string { "NewOwner" } )
t . SetMethods ( constants . EnsContractAddress , nil )
t . SetEventArgs ( constants . EnsContractAddress , [ ] string { "fake_filter_value" } )
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" , constants . EnsContractAddress ) ) . StructScan ( & log )
Expect ( err ) . To ( HaveOccurred ( ) )
} )
It ( "If a method arg filter is applied, only those arguments are used in polling" , func ( ) {
t := transformer . NewTransformer ( "" , blockChain , db )
t . SetEvents ( constants . EnsContractAddress , [ ] string { "NewOwner" } )
t . SetMethods ( constants . EnsContractAddress , [ ] string { "owner" } )
t . SetMethodArgs ( constants . EnsContractAddress , [ ] string { "0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae" } )
err = t . Init ( )
Expect ( err ) . ToNot ( HaveOccurred ( ) )
2018-11-24 04:26:07 +00:00
err = t . Execute ( )
2018-12-14 17:52:02 +00:00
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'" , constants . EnsContractAddress ) ) . 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'" , constants . EnsContractAddress ) ) . StructScan ( & res )
2018-11-24 04:26:07 +00:00
Expect ( err ) . To ( HaveOccurred ( ) )
} )
2018-11-23 18:12:24 +00:00
} )
} )