All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.keyko.monitoring.services.EventLogService Maven / Gradle / Ivy

package io.keyko.monitoring.services;

import io.keyko.monitoring.exceptions.AbiNotFoundException;
import io.keyko.monitoring.exceptions.EtherscanException;
import io.keyko.monitoring.exceptions.EventFromLogException;
import io.keyko.monitoring.model.ContractData;
import io.keyko.monitoring.schemas.*;
import org.apache.log4j.Logger;
import org.web3j.abi.EventEncoder;
import org.web3j.abi.FunctionReturnDecoder;
import org.web3j.abi.datatypes.Event;
import org.web3j.abi.datatypes.Type;
import org.web3j.protocol.core.methods.response.AbiDefinition;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.IntStream;

public class EventLogService {

  private static Logger log = Logger.getLogger(EventLogService.class);

  public static EventRecord getEventFromLog(LogRecord log, String getContractAbiUrl, String apiKey) throws EventFromLogException {

    String contractAddress = log.getAddress();
    ContractData contractData = null;

    try {
      contractData = EtherscanService.getContractData(getContractAbiUrl, contractAddress, apiKey);
    } catch (EtherscanException e) {
        throw new EventFromLogException("Error getting the data from Contract", e);
    }

    List eventsAbiDefinition = getAbiDefinitionByType("event", contractData.getAbiDefinitionList());

    AbiDefinition eventAbi = null;
    try {
      eventAbi = getEventAbi(log, eventsAbiDefinition);
    } catch (AbiNotFoundException e) {
        throw new EventFromLogException("Error getting the ABI Definition for the event", e);
    }

    Event event = getEventFromAbi(eventAbi);

    List indexed = getIndexedParametersFromLog(event, log);
    List nonIndexed = getNonIndexedParametersFromLog(event, log);

    return generateEventRecord(contractData.getName(),
                                                  event.getName(),
                                                  getEventHashFromAbi(eventAbi),
                                                  web3ToMonitoringType(indexed, eventAbi, true),
                                                  web3ToMonitoringType(nonIndexed, eventAbi, false),
                                                  log);
  }



  private static ContractEventStatus getEventStatus(LogStatus logStatus){
    switch(logStatus){
      case CONFIRMED: return ContractEventStatus.CONFIRMED;
      case UNCONFIRMED: return ContractEventStatus.UNCONFIRMED;
      case INVALIDATED: return ContractEventStatus.INVALIDATED;
    }

    return ContractEventStatus.UNCONFIRMED;
  }

  private static EventRecord generateEventRecord(String contractName, String eventName, String eventHash, List indexedParameters, List nonIndexedParameters, LogRecord log){

    EventRecord eventRecord = new EventRecord();
    eventRecord.setRetries(log.getRetries());
    eventRecord.setId(log.getId());

    eventRecord.setContractName(contractName);
    eventRecord.setBlockNumber(log.getBlockNumber());
    eventRecord.setId(log.getId());
    eventRecord.setName(eventName);
    eventRecord.setStatus(getEventStatus(log.getStatus()));
    eventRecord.setBlockHash(log.getBlockHash());
    eventRecord.setAddress(log.getAddress());
    eventRecord.setEventSpecificationSignature(eventHash);
    eventRecord.setFilterId(eventName);
    eventRecord.setLogIndex(log.getLogIndex());
    eventRecord.setNetworkName(log.getNetworkName());
    eventRecord.setNodeName(log.getNodeName());
    eventRecord.setTransactionHash(log.getTransactionHash());

    eventRecord.setIndexedParameters(indexedParameters);
    eventRecord.setNonIndexedParameters(nonIndexedParameters);


    return eventRecord;

  }

  private static List web3ToMonitoringType(List parameters,  AbiDefinition eventAbi, Boolean indexed) {

    List monitoringParameters = new ArrayList<>();

    List namedTypes = eventAbi.getInputs()
      .stream()
      .filter( input -> input.isIndexed() == indexed)
      .collect(Collectors.toList());

    for(int i=0; i getNonIndexedParametersFromLog(Event event, LogRecord log) {

    if (event.getNonIndexedParameters().isEmpty() ) {
      return Collections.EMPTY_LIST;
    }

    return FunctionReturnDecoder.decode(
      log.getData(), event.getNonIndexedParameters());
  }

  private static List getIndexedParametersFromLog(Event event, LogRecord log) {

    if (event.getIndexedParameters().isEmpty() ) {
      return Collections.EMPTY_LIST;
    }

    List topics =  log.getTopics().stream().map( o -> o.toString()).collect(Collectors.toList());
    final List encodedParameters = topics.subList(1, topics.size());

    return IntStream.range(0, encodedParameters.size())
      .mapToObj(i -> FunctionReturnDecoder.decodeIndexedValue(topics.get(i), event.getIndexedParameters().get(i)))
      .collect(Collectors.toList());
  }


  private static String getEventHashFromAbi(AbiDefinition eventAbi) throws EventFromLogException {
    return EventEncoder.encode(getEventFromAbi(eventAbi));
  }

  public static Event getEventFromAbi(AbiDefinition eventAbi) throws EventFromLogException {

    List params = eventAbi.getInputs();
    List> typeReferences = new ArrayList<>();

    for(AbiDefinition.NamedType param: params){

      org.web3j.abi.TypeReference type = null;

      try {

        type = org.web3j.abi.TypeReference.makeTypeReference(param.getType(), param.isIndexed(), false);

      } catch (ClassNotFoundException e) {
        String message = "Error making type reference";
        log.error(message + " "+ e.getMessage());
        throw new EventFromLogException(message, e);
      }

      typeReferences.add(type);

    }

    return new Event(eventAbi.getName(), typeReferences);

  }

  public static AbiDefinition getEventAbi(LogRecord log, List eventsAbiDefinition ) throws AbiNotFoundException{

    String eventLogHash = log.getTopics().get(0);

    AbiDefinition abiDefinition = eventsAbiDefinition
      .stream()
      .filter( eventAbi ->
      {
        // TODO HANDLE EXCEPTION
        Boolean found = false;
        try {
          found = getEventHashFromAbi(eventAbi).equalsIgnoreCase(eventLogHash);
        } catch (Exception e) {
          e.printStackTrace();
        }
        return found;
      })
      .findFirst()
      .orElse(null);

    if (abiDefinition==null)
      throw new AbiNotFoundException("The ABI for event " + eventLogHash + "was not found in the current ABI Definition ");

    return abiDefinition;

  }


  public static List getAbiDefinitionByType(String type, List abiDefinitionList){

    return abiDefinitionList
      .stream()
      .filter(abi -> abi.getType().equalsIgnoreCase(type))
      .collect(Collectors.toList());
  }


}