Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Connect to eth node #1580

Open
wants to merge 113 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
113 commits
Select commit Hold shift + click to select a range
51c6091
Initial commit of connect-to-eth-node.
smoelius Jan 5, 2020
87e3b44
Add web3 to install_requires.
smoelius Jan 5, 2020
56a7bab
Ignore web3 types.
smoelius Jan 5, 2020
b395920
Handle symbolic offsets.
smoelius Jan 6, 2020
d16736a
Add typeguard to install_requires.
smoelius Jan 6, 2020
6df79e6
Blacken.
smoelius Jan 6, 2020
1b50ce0
Better mypy integration.
smoelius Jan 6, 2020
5afa862
More blackening.
smoelius Jan 6, 2020
254c6c8
Limit paths that typeguard checks.
smoelius Jan 6, 2020
d58368c
Fix broken get_storage_data types.
smoelius Jan 6, 2020
ffa39b7
Blacken.
smoelius Jan 6, 2020
7d654aa
Revert some changes related to the callstack.
smoelius Jan 7, 2020
46f7a30
Delete accounts unconditionally.
smoelius Jan 7, 2020
b1a82f7
Do not throw an exception if deleted account is not in dictionary.
smoelius Jan 7, 2020
d40d143
Add constraint argument to delete_account call.
smoelius Jan 8, 2020
043f9c0
Back out typeguard.
smoelius Jan 8, 2020
5a66521
Use "not in".
smoelius Jan 8, 2020
9eff254
Fix minor indentation issue in ci.yml.
smoelius Jan 8, 2020
081f4b4
"fromDict" -> "from_dict".
smoelius Jan 8, 2020
bc0cb70
Add (failing) test_gas_check.
smoelius Jan 9, 2020
32a19ea
Fix gas check.
smoelius Jan 9, 2020
4cbad00
Merge commit '32a19ea' into dev-connect-to-eth-node
smoelius Jan 9, 2020
8b0a5ef
Use Storage.set in set_storage_data.
smoelius Jan 9, 2020
e18fd7c
Consider BitVecConstant not symbolic in dump.
smoelius Jan 9, 2020
912366b
Add tests.
smoelius Jan 9, 2020
6222aca
Merge commit '9eff254' into dev-connect-to-eth-node
smoelius Jan 10, 2020
eec85f9
Do not thrown an exception when "evm.trace" is not present.
smoelius Jan 10, 2020
51d4494
Reduce Storage warnings.
smoelius Jan 10, 2020
566620e
Address Web3 forking bug.
smoelius Jan 10, 2020
a1d5cdd
Use count_ready_states in test_dump.
smoelius Jan 10, 2020
80c0f1f
Add command line arguments.
smoelius Jan 10, 2020
ca7d215
Run truffle tests with ganache-cli as the backend.
smoelius Jan 10, 2020
1855689
Merge remote-tracking branch 'origin/master' into dev-connect-to-eth-…
smoelius Jan 10, 2020
244554a
Don't try to convert None to an int.
smoelius Jan 10, 2020
9c871cb
Separate out truffle tests.
smoelius Jan 10, 2020
468036e
Add '-p' to mkdir, and clean-up properly otherwise.
smoelius Jan 10, 2020
6c15093
Don't try to unbox over an already unboxed truffle box.
smoelius Jan 10, 2020
920d28d
Remove unused import in evm_world_state.py.
smoelius Jan 10, 2020
3f15747
Remove unused imports from test_general.py.
smoelius Jan 10, 2020
4944d67
Simplify ci.yml
smoelius Jan 10, 2020
e1bfb4b
Merge commit '4944d67' into dev-connect-to-eth-node
smoelius Jan 11, 2020
d298c19
Remove unnecessary assignment in _web3.
smoelius Jan 13, 2020
72c95b6
Do not enable uninitialized storage detector when using RPC.
smoelius Jan 13, 2020
758e8c7
Typo.
smoelius Apr 15, 2020
053b393
Do not require argv[0] when --rpc and --txvictim are given.
smoelius Apr 15, 2020
1d0cebd
Add tests.
smoelius Apr 15, 2020
ca67c9c
Merge remote-tracking branch 'origin/master' into dev-connect-to-eth-…
smoelius Apr 15, 2020
abdca9e
Typo.
smoelius Apr 15, 2020
ef328af
Address typing issues.
smoelius Apr 16, 2020
b0a38fd
Report errors in ci.yml.
smoelius Apr 16, 2020
913f31f
Add EXPECTED_TX to ci.yml.
smoelius Apr 16, 2020
2ff301e
Better variable name in ci.yml.
smoelius Apr 16, 2020
bc5085b
Fix types.
smoelius Apr 23, 2020
e0c7272
Create DefaultWorldState only once.
smoelius Apr 23, 2020
d15cddf
"--txvictim" -> "--txtarget".
smoelius Apr 23, 2020
cd3b7e6
Create accounts when not using RPC.
smoelius Apr 23, 2020
0d3254a
Add bad IP/port tests.
smoelius Apr 24, 2020
06da949
Update option description.
smoelius Apr 24, 2020
7454fa4
Raise EthereumError on connection failure.
smoelius Apr 28, 2020
fd562db
Move OverlayWorldState comment into docstring.
smoelius Apr 28, 2020
09879e1
Store newly created Storage in world_state.
smoelius Apr 28, 2020
462982a
Add error message in OverlayWorldState.get_storage.
smoelius Apr 28, 2020
a1b06de
Add constraints parameter to WorldState.get_storage_data.
smoelius Apr 30, 2020
fbebf32
Change how Storage objects are allocated.
smoelius Apr 30, 2020
ea1b2bd
Add EVMWorld.get_storage.
smoelius Apr 30, 2020
92a1849
Add storage tests.
smoelius Apr 30, 2020
b3089b7
Blacken.
smoelius Apr 30, 2020
a7d090a
Re-add EVMWorld.get_storage docstring.
smoelius Apr 30, 2020
954e92f
Correct EVMWorld.get_storage docstring.
smoelius Apr 30, 2020
a3e70e2
Add missing newline.
smoelius May 1, 2020
18bf648
Better error handling/reporting.
smoelius May 1, 2020
ee206c4
Fix error in ci.yml.
smoelius May 1, 2020
abe52bc
Add EVMWorld.get_storage_items.
smoelius May 1, 2020
53117d2
Adjust z3 options.
smoelius May 1, 2020
6ccbd49
Experimenting with Storage simplifications.
smoelius May 4, 2020
88ddfbc
Add nonce checks to test_gas_check.
smoelius May 4, 2020
556a492
Remove reference to map.
smoelius May 8, 2020
56a38f0
Reduce log messages.
smoelius May 8, 2020
a5c2ddc
Eliminate ci.yml conflicts.
smoelius May 8, 2020
af3cc7d
Merge branch 'master' into dev-connect-to-eth-node
smoelius May 8, 2020
eecd9b5
Update ci.yml
smoelius May 8, 2020
44e6c0a
Do not store constraints in Storage.
smoelius May 15, 2020
c3ae87f
Eliminate use of dirty.
smoelius May 15, 2020
747ad69
Rename Storage.data to Storage._data.
smoelius May 15, 2020
b3558fe
Merge remote-tracking branch 'origin/master' into dev-connect-to-eth-…
smoelius May 17, 2020
235d0b5
Add docstring to Storage.new_storage.
smoelius May 24, 2020
6054033
Move get_items into ArrayProxy.
smoelius May 25, 2020
8f59f39
Merge remote-tracking branch 'origin/master' into dev-connect-to-eth-…
smoelius May 25, 2020
49707bd
Merge remote-tracking branch 'origin/master' into dev-connect-to-eth-…
smoelius Jun 11, 2020
45b87b8
Correct bitvec width in add_to/sub_from_balance.
smoelius Jun 11, 2020
60daa52
Blacken.
smoelius Jun 11, 2020
b8880cf
Interpret coinbase as hex.
smoelius Jun 11, 2020
c4a0471
Fiz __getstae__/__setstate__ names.
smoelius Jun 11, 2020
15d5366
Add comment re disabling uninitialized storage detector.
smoelius Jun 11, 2020
3a01360
Set blockheader information.
smoelius Jun 11, 2020
1de8eba
Use self.get_storage.
smoelius Jun 11, 2020
91d48ed
Use self._set_storage.
smoelius Jun 11, 2020
02a540e
Nominal changes.
smoelius Jun 11, 2020
a4c599d
Callstack type has changed.
smoelius Jun 11, 2020
2215ba8
Eliminate unused imports.
smoelius Jun 12, 2020
6b3dbf3
Simplify Storage.get.
smoelius Jun 12, 2020
c29a24a
Fix bugs introduced by merge.
smoelius Jun 12, 2020
f257ec8
More merge bug fixes.
smoelius Jun 12, 2020
399dd08
Handle case where EXP's exponent is a symbolic constant.
smoelius Jun 12, 2020
7a38281
Turn off detectors in truffle maze test.
smoelius Jun 12, 2020
3446e21
Implement Storage.__copy__.
smoelius Jun 12, 2020
cc97a38
Convert gas to constant in make_VMTests.py.
smoelius Jun 12, 2020
94055f9
Coverage fixes in ci.yml.
smoelius Jun 12, 2020
8af6869
More ci.yml fixes.
smoelius Jun 12, 2020
4a23d8a
Up SMT timeout for metacoin tests.
smoelius Jun 12, 2020
3fc4264
Fix bug concerning balance width.
smoelius Jun 12, 2020
ebd8cc0
Balance width fix was incomplete.
smoelius Jun 12, 2020
cca9dbf
Eliminate new_storage method.
smoelius Jun 12, 2020
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
59 changes: 40 additions & 19 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ jobs:
runs-on: ubuntu-latest
strategy:
matrix:
type: ["ethereum_truffle", "ethereum_bench", "examples", "ethereum", "ethereum_vm", "native", "wasm", "wasm_sym", "other"]
type: ["ethereum_bench", "examples", "ethereum", "ethereum_metacoin", "ethereum_truffle", "ethereum_vm", "native", "wasm", "wasm_sym", "other"]
steps:
- uses: actions/checkout@v1
- name: Set up Python 3.6
Expand Down Expand Up @@ -155,26 +155,29 @@ jobs:
cd $DIR
}

install_truffle(){
npm install -g truffle
install_node_packages(){
npm install -g ganache-cli truffle
}

run_truffle_tests(){
run_metacoin_tests(){
COVERAGE_RCFILE=$GITHUB_WORKSPACE/.coveragerc
mkdir truffle_tests
cd truffle_tests
truffle unbox metacoin
coverage run -m manticore . --contract MetaCoin --workspace output --exclude-all --evm.oog ignore --evm.txfail optimistic
ADDITIONAL_ARGS=.
EXPECTED_TX=27
if [ "$1" = "deploy" ]; then
ADDRESS="$(truffle deploy | grep '> contract address:.*' | tail -n 1 | grep -o '0x.*$')"
ADDITIONAL_ARGS="--rpc 127.0.0.1:7545 --txtarget $ADDRESS"
EXPECTED_TX=26
fi
coverage run -p -m manticore --contract MetaCoin --workspace output --exclude-all --evm.oog ignore --evm.txfail optimistic --smt.timeout 600 $ADDITIONAL_ARGS
# Truffle smoke test. We test if manticore is able to generate states
# from a truffle project.
if [ "$(ls output/*tx -l | wc -l)" != "26" ]; then
echo "Truffle test failed" `ls output/*tx -l | wc -l` "!= 26"
ACTUAL_TX="$(ls output/*tx -l | wc -l)"
rm -rf output
if [ "$ACTUAL_TX" != "$EXPECTED_TX" ]; then
echo "MetaCoin test failed: $ACTUAL_TX != $EXPECTED_TX"
return 1
fi
echo "Truffle test succeded"
coverage xml
cd ..
cp truffle_tests/coverage.xml .
echo "MetaCoin test succeded"
return 0
}

Expand All @@ -183,6 +186,9 @@ jobs:
COVERAGE_RCFILE=$GITHUB_WORKSPACE/.coveragerc
echo "Running only the tests from 'tests/$DIR' directory"
pytest --durations=100 --cov=manticore --cov-config=$GITHUB_WORKSPACE/.coveragerc -n auto "tests/$DIR"
if [ $? -ne 0 ]; then
return 1
fi
coverage xml
}

Expand All @@ -207,15 +213,30 @@ jobs:

# Test type
case $TEST_TYPE in
ethereum_metacoin)
echo "Running metacoin tests"
install_node_packages
ganache-cli -p 7545 -i 5777 &
mkdir metacoin_tests \
&& cd metacoin_tests \
&& truffle unbox metacoin \
&& run_metacoin_tests \
&& run_metacoin_tests deploy \
&& coverage combine \
&& coverage xml \
&& cd .. \
&& cp metacoin_tests/coverage.xml .
;;
ethereum_truffle)
echo "Running truffle tests"
install_node_packages
ganache-cli -p 7545 -i 5777 &
run_tests_from_dir $TEST_TYPE
;;
ethereum_vm)
make_vmtests
run_tests_from_dir $TEST_TYPE
;;
ethereum_truffle)
echo "Running truffle test"
install_truffle
run_truffle_tests
;;
wasm)
make_wasm_tests
run_tests_from_dir $TEST_TYPE
Expand Down
19 changes: 18 additions & 1 deletion manticore/__main__.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,12 @@ def main() -> None:
resources.check_disk_usage()
resources.check_memory_usage()

if args.argv[0].endswith(".sol") or is_supported(args.argv[0]):
if (
args.url is not None
or args.txtarget is not None
or args.argv[0].endswith(".sol")
or is_supported(args.argv[0])
):
ethereum_main(args, logger)
elif args.argv[0].endswith(".wasm") or args.argv[0].endswith(".wat"):
wasm_main(args, logger)
Expand Down Expand Up @@ -165,6 +170,8 @@ def positive(value):
"--txnoether", action="store_true", help="Do not attempt to send ether to contract"
)

eth_flags.add_argument("--txtarget", type=str, help="Address of a deployed contract to target")

eth_flags.add_argument(
"--txaccount",
type=str,
Expand All @@ -183,6 +190,13 @@ def positive(value):
"--contract", type=str, help="Contract name to analyze in case of multiple contracts"
)

eth_flags.add_argument(
"--rpc",
type=str,
dest="url",
help="URL of an Ethereum node to connect to. Must be of the form 'IP:PORT'",
)

eth_detectors = parser.add_argument_group("Ethereum detectors")

eth_detectors.add_argument(
Expand Down Expand Up @@ -242,6 +256,9 @@ def positive(value):
parsed = parser.parse_args(sys.argv[1:])
config.process_config_values(parser, parsed)

if parsed.txtarget is not None and not parsed.argv:
parsed.argv = [None]

if not parsed.argv:
print(parser.format_usage() + "error: the following arguments are required: argv")
sys.exit(1)
Expand Down
12 changes: 10 additions & 2 deletions manticore/core/smtlib/expression.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

import re
import copy
from typing import Union, Optional, Dict, List
from typing import Union, Optional, Dict, List, Tuple


class ExpressionException(SmtlibError):
Expand Down Expand Up @@ -670,7 +670,7 @@ def __init__(
self, index_bits: int, index_max: Optional[int], value_bits: int, *operands, **kwargs
):
assert index_bits in (32, 64, 256)
assert value_bits in (8, 16, 32, 64, 256)
assert value_bits in (1, 8, 16, 32, 64, 256)
smoelius marked this conversation as resolved.
Show resolved Hide resolved
assert index_max is None or index_max >= 0 and index_max < 2 ** index_bits
self._index_bits = index_bits
self._index_max = index_max
Expand Down Expand Up @@ -1205,6 +1205,14 @@ def get(self, index, default=None):
value = self._array.select(index)
return BitVecITE(self._array.value_bits, is_known, value, default)

def get_items(self) -> List[Tuple[Union[int, BitVec], Union[int, BitVec]]]:
items = []
array = self.array
while not isinstance(array, ArrayVariable):
items.append((array.index, array.value))
array = array.array
return items


class ArraySelect(BitVec):
__slots__ = ["_operands"]
Expand Down
44 changes: 33 additions & 11 deletions manticore/ethereum/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
DetectManipulableBalance,
)
from ..core.plugin import Profiler
from ..exceptions import EthereumError
from .manticore import ManticoreEVM
from .plugins import (
FilterFunctions,
Expand All @@ -23,6 +24,7 @@
KeepOnlyIfStorageChanges,
SkipRevertBasicBlocks,
)
from ..platforms.evm_world_state import RemoteWorldState
from ..utils.nointerrupt import WithKeyboardInterruptAs
from ..utils import config

Expand Down Expand Up @@ -79,6 +81,10 @@ def choose_detectors(args):
f"{e} is not a detector name, must be one of {arguments}. See also `--list-detectors`."
)

# sam.moelius: Do not enable uninitialized storage detector when using RPC. It generates too much noise.
if args.url is not None:
exclude.append(DetectUninitializedStorage.ARGUMENT)

for arg, detector_cls in detectors.items():
if arg not in exclude:
detectors_to_run.append(detector_cls)
Expand All @@ -87,7 +93,11 @@ def choose_detectors(args):


def ethereum_main(args, logger):
m = ManticoreEVM(workspace_url=args.workspace)
world_state = None
if args.url is not None:
world_state = RemoteWorldState(args.url)

m = ManticoreEVM(workspace_url=args.workspace, world_state=world_state)

if args.quick_mode:
args.avoid_constant = True
Expand Down Expand Up @@ -130,16 +140,28 @@ def ethereum_main(args, logger):
logger.info("Beginning analysis")

with m.kill_timeout():
m.multi_tx_analysis(
args.argv[0],
contract_name=args.contract,
tx_limit=args.txlimit,
tx_use_coverage=not args.txnocoverage,
tx_send_ether=not args.txnoether,
tx_account=args.txaccount,
tx_preconstrain=args.txpreconstrain,
compile_args=vars(args), # FIXME
)
try:
contract_account = None
if args.txtarget is not None:
contract_account = int(args.txtarget, base=0)
if world_state is not None and not world_state.get_code(contract_account):
raise EthereumError(
"Could not get code for target account: " + args.txtarget
)

m.multi_tx_analysis(
args.argv[0],
contract_name=args.contract,
tx_limit=args.txlimit,
tx_use_coverage=not args.txnocoverage,
tx_send_ether=not args.txnoether,
contract_account=contract_account,
tx_account=args.txaccount,
tx_preconstrain=args.txpreconstrain,
compile_args=vars(args), # FIXME
)
except EthereumError as e:
logger.error("%r", e.args[0])

if not args.no_testcases:
m.finalize(only_alive_states=args.only_alive_testcases)
Expand Down
42 changes: 24 additions & 18 deletions manticore/ethereum/manticore.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
from .state import State
from ..exceptions import EthereumError, DependencyError, NoAliveStates
from ..platforms import evm
from ..platforms.evm_world_state import WorldState
from ..utils import config, log
from ..utils.deprecated import deprecated
from ..utils.helpers import PickleSerializer, printable_bytes
Expand Down Expand Up @@ -390,15 +391,15 @@ def contract_accounts(self):
def get_account(self, name):
return self._accounts[name]

def __init__(self, plugins=None, **kwargs):
def __init__(self, world_state: Optional[WorldState] = None, plugins=None, **kwargs):
"""
A Manticore EVM manager
:param plugins: the plugins to register in this manticore manager
"""
# Make the constraint store
constraints = ConstraintSet()
# make the ethereum world state
world = evm.EVMWorld(constraints)
world = evm.EVMWorld(constraints, world_state=world_state)
initial_state = State(constraints, world)
super().__init__(initial_state, **kwargs)
if plugins is not None:
Expand Down Expand Up @@ -1035,30 +1036,35 @@ def multi_tx_analysis(
tx_limit=None,
tx_use_coverage=True,
tx_send_ether=True,
contract_account: Optional[int] = None,
tx_account="attacker",
tx_preconstrain=False,
args=None,
compile_args=None,
):
owner_account = self.create_account(balance=10 ** 10, name="owner")
attacker_account = self.create_account(balance=10 ** 10, name="attacker")
# Pretty print
logger.info("Starting symbolic create contract")

if tx_send_ether:
create_value = self.make_symbolic_value()
else:
create_value = 0

contract_account = self.solidity_create_contract(
solidity_filename,
contract_name=contract_name,
owner=owner_account,
args=args,
compile_args=compile_args,
balance=create_value,
gas=230000,
)
if contract_account is None:
feliam marked this conversation as resolved.
Show resolved Hide resolved
# Pretty print
logger.info("Starting symbolic create contract")

if tx_send_ether:
create_value = self.make_symbolic_value()
else:
create_value = 0

contract_account = self.solidity_create_contract(
solidity_filename,
contract_name=contract_name,
owner=owner_account,
args=args,
compile_args=compile_args,
balance=create_value,
gas=230000,
)
elif solidity_filename is not None:
raise EthereumError("A target contract and a Solidty filename cannot both be specified")

if tx_account == "attacker":
tx_account = [attacker_account]
Expand Down
Loading