2018-11-07 21:50:43 +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/>.
|
2018-11-04 21:26:39 +00:00
|
|
|
|
|
|
|
package transformer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2019-01-04 18:15:22 +00:00
|
|
|
"strings"
|
2018-12-07 15:38:46 +00:00
|
|
|
|
2018-11-07 21:50:43 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/core"
|
2018-11-04 21:26:39 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/datastore"
|
2018-11-07 21:50:43 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/datastore/postgres"
|
2018-11-04 21:26:39 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/datastore/postgres/repositories"
|
2018-11-23 18:12:24 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/full/converter"
|
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/full/retriever"
|
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/contract"
|
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/parser"
|
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/poller"
|
2018-11-24 04:26:07 +00:00
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/repository"
|
|
|
|
"github.com/vulcanize/vulcanizedb/pkg/omni/shared/types"
|
2018-11-04 21:26:39 +00:00
|
|
|
)
|
|
|
|
|
2018-11-23 18:12:24 +00:00
|
|
|
// Requires a fully synced vDB and a running eth node (or infura)
|
2019-02-27 16:10:59 +00:00
|
|
|
type Transformer struct {
|
2018-11-07 21:50:43 +00:00
|
|
|
// Database interfaces
|
2018-11-20 16:38:23 +00:00
|
|
|
datastore.FilterRepository // Log filters repo; accepts filters generated by Contract.GenerateFilters()
|
|
|
|
datastore.WatchedEventRepository // Watched event log views, created by the log filters
|
2018-11-23 18:12:24 +00:00
|
|
|
repository.EventRepository // Holds transformed watched event log data
|
2018-11-04 21:26:39 +00:00
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// Pre-processing interfaces
|
|
|
|
parser.Parser // Parses events and methods out of contract abi fetched using contract address
|
|
|
|
retriever.BlockRetriever // Retrieves first block for contract and current block height
|
|
|
|
|
2018-11-07 21:50:43 +00:00
|
|
|
// Processing interfaces
|
2018-11-20 16:38:23 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
// Ethereum network name; default "" is mainnet
|
|
|
|
Network string
|
2018-11-04 21:26:39 +00:00
|
|
|
|
|
|
|
// Store contract info as mapping to contract address
|
|
|
|
Contracts map[string]*contract.Contract
|
|
|
|
|
|
|
|
// Targeted subset of events/methods
|
2018-11-07 21:50:43 +00:00
|
|
|
// Stored as map sof contract address to events/method names of interest
|
|
|
|
WatchedEvents map[string][]string // Default/empty event list means all are watched
|
|
|
|
WantedMethods map[string][]string // Default/empty method list means none are polled
|
|
|
|
|
2019-01-04 18:15:22 +00:00
|
|
|
// Starting block for contracts
|
|
|
|
ContractStart map[string]int64
|
2018-11-07 21:50:43 +00:00
|
|
|
|
|
|
|
// Lists of addresses to filter event or method data
|
|
|
|
// before persisting; if empty no filter is applied
|
2018-12-07 15:38:46 +00:00
|
|
|
EventArgs map[string][]string
|
|
|
|
MethodArgs map[string][]string
|
|
|
|
|
2018-12-14 17:52:02 +00:00
|
|
|
// Whether or not to create a list of emitted address or hashes for the contract in postgres
|
2018-12-07 15:38:46 +00:00
|
|
|
CreateAddrList map[string]bool
|
2018-12-14 17:52:02 +00:00
|
|
|
CreateHashList map[string]bool
|
|
|
|
|
|
|
|
// Method piping on/off for a contract
|
|
|
|
Piping map[string]bool
|
2018-11-04 21:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Transformer takes in config for blockchain, database, and network id
|
2019-02-27 16:10:59 +00:00
|
|
|
func NewTransformer(network string, BC core.BlockChain, DB *postgres.DB) *Transformer {
|
|
|
|
return &Transformer{
|
2018-11-24 04:26:07 +00:00
|
|
|
Poller: poller.NewPoller(BC, DB, types.FullSync),
|
2018-11-07 21:50:43 +00:00
|
|
|
Parser: parser.NewParser(network),
|
|
|
|
BlockRetriever: retriever.NewBlockRetriever(DB),
|
|
|
|
Converter: converter.NewConverter(&contract.Contract{}),
|
2018-11-04 21:26:39 +00:00
|
|
|
Contracts: map[string]*contract.Contract{},
|
2018-11-07 21:50:43 +00:00
|
|
|
WatchedEventRepository: repositories.WatchedEventRepository{DB: DB},
|
|
|
|
FilterRepository: repositories.FilterRepository{DB: DB},
|
2018-11-24 04:26:07 +00:00
|
|
|
EventRepository: repository.NewEventRepository(DB, types.FullSync),
|
2018-11-07 21:50:43 +00:00
|
|
|
WatchedEvents: map[string][]string{},
|
|
|
|
WantedMethods: map[string][]string{},
|
2019-01-04 18:15:22 +00:00
|
|
|
ContractStart: map[string]int64{},
|
2018-12-07 15:38:46 +00:00
|
|
|
EventArgs: map[string][]string{},
|
|
|
|
MethodArgs: map[string][]string{},
|
2018-12-14 17:52:02 +00:00
|
|
|
CreateAddrList: map[string]bool{},
|
|
|
|
CreateHashList: map[string]bool{},
|
|
|
|
Piping: map[string]bool{},
|
2018-11-04 21:26:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use after creating and setting transformer
|
|
|
|
// Loops over all of the addr => filter sets
|
|
|
|
// Uses parser to pull event info from abi
|
|
|
|
// Use this info to generate event filters
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) Init() error {
|
|
|
|
for contractAddr, subset := range transformer.WatchedEvents {
|
2018-11-04 21:26:39 +00:00
|
|
|
// Get Abi
|
2019-02-28 14:59:04 +00:00
|
|
|
err := transformer.Parser.Parse(contractAddr)
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-04 18:15:22 +00:00
|
|
|
// Get first block and most recent block number in the header repo
|
2019-02-28 14:59:04 +00:00
|
|
|
firstBlock, err := transformer.BlockRetriever.RetrieveFirstBlock(contractAddr)
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-02-28 14:59:04 +00:00
|
|
|
lastBlock, err := transformer.BlockRetriever.RetrieveMostRecentBlock()
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-04 18:15:22 +00:00
|
|
|
// Set to specified range if it falls within the bounds
|
2019-02-28 14:59:04 +00:00
|
|
|
if firstBlock < transformer.ContractStart[contractAddr] {
|
|
|
|
firstBlock = transformer.ContractStart[contractAddr]
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
|
|
|
|
2018-12-07 15:38:46 +00:00
|
|
|
// Get contract name if it has one
|
2018-11-07 21:50:43 +00:00
|
|
|
var name = new(string)
|
2019-02-28 14:59:04 +00:00
|
|
|
transformer.Poller.FetchContractData(transformer.Abi(), contractAddr, "name", nil, name, lastBlock)
|
2018-11-07 21:50:43 +00:00
|
|
|
|
2018-12-07 15:38:46 +00:00
|
|
|
// Remove any potential accidental duplicate inputs in arg filter values
|
|
|
|
eventArgs := map[string]bool{}
|
2019-02-28 14:59:04 +00:00
|
|
|
for _, arg := range transformer.EventArgs[contractAddr] {
|
2018-12-07 15:38:46 +00:00
|
|
|
eventArgs[arg] = true
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
2018-12-07 15:38:46 +00:00
|
|
|
methodArgs := map[string]bool{}
|
2019-02-28 14:59:04 +00:00
|
|
|
for _, arg := range transformer.MethodArgs[contractAddr] {
|
2018-12-07 15:38:46 +00:00
|
|
|
methodArgs[arg] = true
|
2018-11-04 21:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Aggregate info into contract object
|
2018-12-07 15:38:46 +00:00
|
|
|
info := contract.Contract{
|
2019-02-28 14:59:04 +00:00
|
|
|
Name: *name,
|
|
|
|
Network: transformer.Network,
|
|
|
|
Address: contractAddr,
|
|
|
|
Abi: transformer.Parser.Abi(),
|
|
|
|
ParsedAbi: transformer.Parser.ParsedAbi(),
|
|
|
|
StartingBlock: firstBlock,
|
|
|
|
LastBlock: lastBlock,
|
|
|
|
Events: transformer.Parser.GetEvents(subset),
|
|
|
|
Methods: transformer.Parser.GetSelectMethods(transformer.WantedMethods[contractAddr]),
|
2018-12-07 15:38:46 +00:00
|
|
|
FilterArgs: eventArgs,
|
|
|
|
MethodArgs: methodArgs,
|
2019-02-28 14:59:04 +00:00
|
|
|
CreateAddrList: transformer.CreateAddrList[contractAddr],
|
|
|
|
CreateHashList: transformer.CreateHashList[contractAddr],
|
|
|
|
Piping: transformer.Piping[contractAddr],
|
2018-12-07 15:38:46 +00:00
|
|
|
}.Init()
|
2018-11-04 21:26:39 +00:00
|
|
|
|
|
|
|
// Use info to create filters
|
2018-11-07 21:50:43 +00:00
|
|
|
err = info.GenerateFilters()
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// Iterate over filters and push them to the repo using filter repository interface
|
2018-11-04 21:26:39 +00:00
|
|
|
for _, filter := range info.Filters {
|
2019-02-28 14:59:04 +00:00
|
|
|
err = transformer.FilterRepository.CreateFilter(filter)
|
2018-12-14 17:52:02 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-11-04 21:26:39 +00:00
|
|
|
}
|
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// Store contract info for further processing
|
2019-02-28 14:59:04 +00:00
|
|
|
transformer.Contracts[contractAddr] = info
|
2018-11-04 21:26:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// Iterates through stored, initialized contract objects
|
|
|
|
// Iterates through contract's event filters, grabbing watched event logs
|
|
|
|
// 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
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer Transformer) Execute() error {
|
|
|
|
if len(transformer.Contracts) == 0 {
|
2018-11-20 16:38:23 +00:00
|
|
|
return errors.New("error: transformer has no initialized contracts to work with")
|
|
|
|
}
|
2018-11-04 21:26:39 +00:00
|
|
|
// Iterate through all internal contracts
|
2019-02-28 14:59:04 +00:00
|
|
|
for _, con := range transformer.Contracts {
|
2018-11-04 21:26:39 +00:00
|
|
|
// Update converter with current contract
|
2019-02-28 14:59:04 +00:00
|
|
|
transformer.Update(con)
|
2018-11-04 21:26:39 +00:00
|
|
|
|
|
|
|
// Iterate through contract filters and get watched event logs
|
2018-12-18 17:31:21 +00:00
|
|
|
for eventSig, filter := range con.Filters {
|
2019-02-28 14:59:04 +00:00
|
|
|
watchedEvents, err := transformer.GetWatchedEvents(filter.Name)
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// Iterate over watched event logs
|
2018-11-04 21:26:39 +00:00
|
|
|
for _, we := range watchedEvents {
|
2018-11-20 16:38:23 +00:00
|
|
|
// Convert them to our custom log type
|
2019-02-28 14:59:04 +00:00
|
|
|
cstm, err := transformer.Converter.Convert(*we, con.Events[eventSig])
|
2018-11-20 16:38:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-11-24 04:26:07 +00:00
|
|
|
if cstm == nil {
|
|
|
|
continue
|
2018-11-23 18:12:24 +00:00
|
|
|
}
|
2018-11-20 16:38:23 +00:00
|
|
|
|
2018-11-23 18:12:24 +00:00
|
|
|
// If log is not empty, immediately persist in repo
|
2018-11-20 16:38:23 +00:00
|
|
|
// Run this in seperate goroutine?
|
2019-02-28 14:59:04 +00:00
|
|
|
err = transformer.PersistLogs([]types.Log{*cstm}, con.Events[eventSig], con.Address, con.Name)
|
2018-11-04 21:26:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-20 16:38:23 +00:00
|
|
|
// After persisting all watched event logs
|
|
|
|
// poller polls select contract methods
|
|
|
|
// and persists the results into custom pg tables
|
|
|
|
// Run this in seperate goroutine?
|
2019-02-28 14:59:04 +00:00
|
|
|
if err := transformer.PollContract(*con); err != nil {
|
2018-11-04 21:26:39 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2018-11-07 21:50:43 +00:00
|
|
|
|
|
|
|
// Used to set which contract addresses and which of their events to watch
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetEvents(contractAddr string, filterSet []string) {
|
|
|
|
transformer.WatchedEvents[strings.ToLower(contractAddr)] = filterSet
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to set subset of account addresses to watch events for
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetEventArgs(contractAddr string, filterSet []string) {
|
|
|
|
transformer.EventArgs[strings.ToLower(contractAddr)] = filterSet
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to set which contract addresses and which of their methods to call
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetMethods(contractAddr string, filterSet []string) {
|
|
|
|
transformer.WantedMethods[strings.ToLower(contractAddr)] = filterSet
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to set subset of account addresses to poll methods on
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetMethodArgs(contractAddr string, filterSet []string) {
|
|
|
|
transformer.MethodArgs[strings.ToLower(contractAddr)] = filterSet
|
2018-12-07 15:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to set the block range to watch for a given address
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetStartingBlock(contractAddr string, start int64) {
|
|
|
|
transformer.ContractStart[strings.ToLower(contractAddr)] = start
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
|
|
|
|
2018-12-14 17:52:02 +00:00
|
|
|
// Used to set whether or not to persist an account address list
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetCreateAddrList(contractAddr string, on bool) {
|
|
|
|
transformer.CreateAddrList[strings.ToLower(contractAddr)] = on
|
2018-11-07 21:50:43 +00:00
|
|
|
}
|
2018-12-14 17:52:02 +00:00
|
|
|
|
|
|
|
// Used to set whether or not to persist an hash list
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetCreateHashList(contractAddr string, on bool) {
|
|
|
|
transformer.CreateHashList[strings.ToLower(contractAddr)] = on
|
2018-12-14 17:52:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to turn method piping on for a contract
|
2019-02-28 14:59:04 +00:00
|
|
|
func (transformer *Transformer) SetPiping(contractAddr string, on bool) {
|
|
|
|
transformer.Piping[strings.ToLower(contractAddr)] = on
|
2018-12-14 17:52:02 +00:00
|
|
|
}
|