Skip to content

Commit

Permalink
Adding trace options
Browse files Browse the repository at this point in the history
  • Loading branch information
nagarev committed Feb 9, 2022
1 parent f783c05 commit f931ab4
Show file tree
Hide file tree
Showing 5 changed files with 240 additions and 14 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -68,9 +68,13 @@ public String wireProtocolQueueSize() {
public JsonNode traceTransaction(String transactionHash, Map<String, String> traceOptions) {
logger.trace("debug_traceTransaction({}, {})", transactionHash, traceOptions);

if (traceOptions != null && !traceOptions.isEmpty()) {
// TODO: implement the logic that takes into account trace options.
logger.warn("Received {} trace options. For now trace options are being ignored", traceOptions);
TraceOptions options = new TraceOptions(traceOptions);

if (options.getUnsupportedOptions().size() > 0) {
// TODO: implement the logic that takes into account the remaining trace options.
logger.warn(
"Received {} unsupported trace options.",
options.getUnsupportedOptions());
}

byte[] hash = stringHexToByteArray(transactionHash);
Expand All @@ -86,7 +90,7 @@ public JsonNode traceTransaction(String transactionHash, Map<String, String> tra
Transaction tx = block.getTransactionsList().get(txInfo.getIndex());
txInfo.setTransaction(tx);

ProgramTraceProcessor programTraceProcessor = new ProgramTraceProcessor();
ProgramTraceProcessor programTraceProcessor = new ProgramTraceProcessor(options);
blockExecutor.traceBlock(programTraceProcessor, 0, block, parent.getHeader(), false, false);

return programTraceProcessor.getProgramTraceAsJsonNode(tx.getHash());
Expand All @@ -96,9 +100,13 @@ public JsonNode traceTransaction(String transactionHash, Map<String, String> tra
public JsonNode traceBlock(String blockHash, Map<String, String> traceOptions) {
logger.trace("debug_traceBlockByHash({}, {})", blockHash, traceOptions);

if (traceOptions != null && !traceOptions.isEmpty()) {
// TODO: implement the logic that takes into account trace options.
logger.warn("Received {} trace options. For now trace options are being ignored", traceOptions);
TraceOptions options = new TraceOptions(traceOptions);

if (options.getUnsupportedOptions().size() > 0) {
// TODO: implement the logic that takes into account the remaining trace options.
logger.warn(
"Received {} unsupported trace options.",
options.getUnsupportedOptions());
}

byte[] bHash = stringHexToByteArray(blockHash);
Expand All @@ -110,7 +118,7 @@ public JsonNode traceBlock(String blockHash, Map<String, String> traceOptions) {

Block parent = blockStore.getBlockByHash(block.getParentHash().getBytes());

ProgramTraceProcessor programTraceProcessor = new ProgramTraceProcessor();
ProgramTraceProcessor programTraceProcessor = new ProgramTraceProcessor(options);
blockExecutor.traceBlock(programTraceProcessor, 0, block, parent.getHeader(), false, false);

List<Keccak256> txHashes = block.getTransactionsList().stream()
Expand Down
65 changes: 65 additions & 0 deletions rskj-core/src/main/java/co/rsk/rpc/modules/debug/TraceOptions.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/*
* This file is part of RskJ
* Copyright (C) 2017 RSK Labs Ltd.
* (derived from ethereumJ library, Copyright (c) 2016 <ether.camp>)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/

package co.rsk.rpc.modules.debug;

import java.util.*;

public class TraceOptions {

private Set<String> disabledFields;
private Set<String> unsupportedOptions;

private TraceOptions() {}

public TraceOptions(Map<String, String> traceOptions) {
disabledFields = new HashSet<>();
unsupportedOptions = new HashSet<>();
if (traceOptions != null) {
if (traceOptions.containsKey("disableStorage")) {
if (Boolean.parseBoolean(traceOptions.get("disableStorage"))) {
disabledFields.add("storage");
}
traceOptions.remove("disableStorage");
}
if (traceOptions.containsKey("disableMemory")) {
if (Boolean.parseBoolean(traceOptions.get("disableMemory"))) {
disabledFields.add("memory");
}
traceOptions.remove("disableMemory");
}
if (traceOptions.containsKey("disableStack")) {
if (Boolean.parseBoolean(traceOptions.get("disableStack"))) {
disabledFields.add("stack");
}
traceOptions.remove("disableStack");
}
unsupportedOptions = traceOptions.keySet();
}
}

public Set<String> getDisabledFields() {
return disabledFields;
}

public Set<String> getUnsupportedOptions() {
return unsupportedOptions;
}

}
2 changes: 2 additions & 0 deletions rskj-core/src/main/java/org/ethereum/vm/trace/Op.java
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@

package org.ethereum.vm.trace;

import com.fasterxml.jackson.annotation.JsonFilter;
import org.ethereum.util.ByteUtil;
import org.ethereum.vm.OpCode;
import org.ethereum.vm.program.Memory;
Expand All @@ -29,6 +30,7 @@
import java.util.List;
import java.util.Map;

@JsonFilter("opFilter")
public class Op {

private OpCode op;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,14 +20,14 @@
package org.ethereum.vm.trace;

import co.rsk.crypto.Keccak256;
import co.rsk.rpc.modules.debug.TraceOptions;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ser.impl.SimpleBeanPropertyFilter;
import com.fasterxml.jackson.databind.ser.impl.SimpleFilterProvider;

import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.*;
import java.util.stream.Collectors;

/**
Expand All @@ -39,6 +39,17 @@ public class ProgramTraceProcessor {

private final Map<Keccak256, ProgramTrace> traces = new HashMap<>();

private TraceOptions traceOptions;

public ProgramTraceProcessor() {
traceOptions = new TraceOptions(Collections.emptyMap());
}

public ProgramTraceProcessor(TraceOptions options) {
this();
traceOptions = options;
}

public void processProgramTrace(ProgramTrace programTrace, Keccak256 txHash) {
this.traces.put(txHash, programTrace);
}
Expand All @@ -53,7 +64,11 @@ public JsonNode getProgramTracesAsJsonNode(List<Keccak256> txHashes) {
.filter(Objects::nonNull)
.collect(Collectors.toList());

return OBJECT_MAPPER.valueToTree(txTraces);
SimpleFilterProvider filterProvider = new SimpleFilterProvider();
filterProvider.addFilter("opFilter",
SimpleBeanPropertyFilter.serializeAllExcept(traceOptions.getDisabledFields()));

return OBJECT_MAPPER.setFilterProvider(filterProvider).valueToTree(txTraces);
}

public JsonNode getProgramTraceAsJsonNode(Keccak256 txHash) {
Expand All @@ -63,7 +78,11 @@ public JsonNode getProgramTraceAsJsonNode(Keccak256 txHash) {
return null;
}

return OBJECT_MAPPER.valueToTree(trace);
SimpleFilterProvider filterProvider = new SimpleFilterProvider();
filterProvider.addFilter("opFilter",
SimpleBeanPropertyFilter.serializeAllExcept(traceOptions.getDisabledFields()));

return makeObjectMapper().setFilterProvider(filterProvider).valueToTree(trace);
}

private static ObjectMapper makeObjectMapper() {
Expand Down
132 changes: 132 additions & 0 deletions rskj-core/src/test/java/co/rsk/rpc/modules/debug/TraceOptionsTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,132 @@
/*
* This file is part of RskJ
* Copyright (C) 2017 RSK Labs Ltd.
* (derived from ethereumJ library, Copyright (c) 2016 <ether.camp>)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/

package co.rsk.rpc.modules.debug;

import org.junit.Assert;
import org.junit.Test;

import java.util.HashMap;
import java.util.Map;

public class TraceOptionsTest {

@Test
public void testTraceOptions_allFieldsSetAsDisabled_disabledFieldsShouldReturnAllFields() {
// Given
Map<String, String> traceOptions = new HashMap<>();
traceOptions.put("disableStorage", "true");
traceOptions.put("disableMemory", "true");
traceOptions.put("disableStack", "true");

// When
TraceOptions options = new TraceOptions(traceOptions);

// Then
Assert.assertEquals(3, options.getDisabledFields().size());
Assert.assertTrue(options.getDisabledFields().contains("storage"));
Assert.assertTrue(options.getDisabledFields().contains("memory"));
Assert.assertTrue(options.getDisabledFields().contains("stack"));
}

@Test
public void testTraceOptions_anyFieldsSetAsDisabled_disabledFieldsShouldReturnEmptySet() {
// Given
Map<String, String> traceOptions = new HashMap<>();
traceOptions.put("disableStorages", "false");
traceOptions.put("disablesMemory", "false");
traceOptions.put("disableStack", "false");

// When
TraceOptions options = new TraceOptions(traceOptions);

// Then
Assert.assertEquals(0, options.getDisabledFields().size());
Assert.assertFalse(options.getDisabledFields().contains("storage"));
Assert.assertFalse(options.getDisabledFields().contains("memory"));
Assert.assertFalse(options.getDisabledFields().contains("stack"));
}

@Test
public void testTraceOptions_someFieldsSetAsDisabled_disabledFieldsShouldReturnDisabledOnes() {
// Given
Map<String, String> traceOptions = new HashMap<>();
traceOptions.put("disableStorage", "true");
traceOptions.put("disableMemory", "false");
traceOptions.put("disableStack", "true");

// When
TraceOptions options = new TraceOptions(traceOptions);

// Then
Assert.assertEquals(2, options.getDisabledFields().size());
Assert.assertTrue(options.getDisabledFields().contains("storage"));
Assert.assertFalse(options.getDisabledFields().contains("memory"));
Assert.assertTrue(options.getDisabledFields().contains("stack"));
}

@Test
public void testTraceOptions_nullTraceOptionsGiven_disabledFieldsAndUnsupportedOptionsShouldReturnEmptySet() {
// When
TraceOptions options = new TraceOptions(null);

// Then
Assert.assertEquals(0, options.getDisabledFields().size());
Assert.assertEquals(0, options.getUnsupportedOptions().size());
}

@Test
public void testTraceOptions_unsupportedOptionsGiven_unsupportedOptionsShouldReturnAllOfThem() {
// Given
Map<String, String> traceOptions = new HashMap<>();
traceOptions.put("unsupportedOption.1", "1");
traceOptions.put("unsupportedOption.2", null);

// When
TraceOptions options = new TraceOptions(traceOptions);

// Then
Assert.assertEquals(2, options.getUnsupportedOptions().size());
Assert.assertTrue(options.getUnsupportedOptions().contains("unsupportedOption.1"));
Assert.assertTrue(options.getUnsupportedOptions().contains("unsupportedOption.2"));
}

@Test
public void testTraceOptions_mixOfSupportedAndUnsupportedOptionsGiven_disabledFieldsAndUnsupportedOptionsShouldReturnOK() {
// Given
Map<String, String> traceOptions = new HashMap<>();
traceOptions.put("disableMemory", "true");
traceOptions.put("disableStorage", "True"); // True != true but should also work
traceOptions.put("unsupportedOption.1", "1");
traceOptions.put("unsupportedOption.2", null);

// When
TraceOptions options = new TraceOptions(traceOptions);

// Then
Assert.assertEquals(2, options.getDisabledFields().size());
Assert.assertTrue(options.getDisabledFields().contains("storage"));
Assert.assertTrue(options.getDisabledFields().contains("memory"));

Assert.assertEquals(2, options.getUnsupportedOptions().size());
Assert.assertTrue(options.getUnsupportedOptions().contains("unsupportedOption.1"));
Assert.assertTrue(options.getUnsupportedOptions().contains("unsupportedOption.2"));
}

}

0 comments on commit f931ab4

Please sign in to comment.