mirror of https://github.com/ConsenSys/mythril
blockchainethereumsmart-contractssoliditysecurityprogram-analysissecurity-analysissymbolic-execution
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
124 lines
3.9 KiB
124 lines
3.9 KiB
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
from mythril.disassembler.disassembly import Disassembly
|
|
from mythril.laser.ethereum.call import SymbolicCalldata
|
|
from mythril.laser.ethereum.evm_exceptions import WriteProtection
|
|
from mythril.laser.ethereum.instructions import Instruction
|
|
from mythril.laser.ethereum.state.account import Account
|
|
from mythril.laser.ethereum.state.environment import Environment
|
|
from mythril.laser.ethereum.state.global_state import GlobalState
|
|
from mythril.laser.ethereum.state.machine_state import MachineState
|
|
from mythril.laser.ethereum.state.world_state import WorldState
|
|
from mythril.laser.ethereum.transaction import TransactionStartSignal
|
|
from mythril.laser.ethereum.transaction.transaction_models import MessageCallTransaction
|
|
from mythril.laser.smt import symbol_factory
|
|
|
|
|
|
def get_global_state():
|
|
active_account = Account("0x0", code=Disassembly("60606040"))
|
|
environment = Environment(
|
|
active_account, None, SymbolicCalldata("2"), None, None, None, None
|
|
)
|
|
world_state = WorldState()
|
|
world_state.put_account(active_account)
|
|
state = GlobalState(world_state, environment, None, MachineState(gas_limit=8000000))
|
|
state.transaction_stack.append(
|
|
(MessageCallTransaction(world_state=world_state, gas_limit=8000000), None)
|
|
)
|
|
return state
|
|
|
|
|
|
@patch(
|
|
"mythril.laser.ethereum.instructions.get_call_parameters",
|
|
return_value=(
|
|
"0",
|
|
Account(code=Disassembly(code="0x00"), address="0x19"),
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
),
|
|
)
|
|
def test_staticcall(f1):
|
|
# Arrange
|
|
state = get_global_state()
|
|
state.mstate.stack = [10, 10, 10, 10, 10, 10, 10, 10, 0]
|
|
instruction = Instruction("staticcall", dynamic_loader=None)
|
|
|
|
# Act and Assert
|
|
with pytest.raises(TransactionStartSignal) as ts:
|
|
instruction.evaluate(state)
|
|
assert ts.value.transaction.static
|
|
assert ts.value.transaction.initial_global_state().environment.static
|
|
|
|
|
|
test_data = (
|
|
"selfdestruct",
|
|
"create",
|
|
"create2",
|
|
"log0",
|
|
"log1",
|
|
"log2",
|
|
"log3",
|
|
"log4",
|
|
"sstore",
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize("input", test_data)
|
|
def test_staticness(input):
|
|
# Arrange
|
|
state = get_global_state()
|
|
state.environment.static = True
|
|
state.mstate.stack = []
|
|
instruction = Instruction(input, dynamic_loader=None)
|
|
|
|
# Act and Assert
|
|
with pytest.raises(WriteProtection):
|
|
instruction.evaluate(state)
|
|
|
|
|
|
test_data_call = ((0, True), (100, False))
|
|
|
|
|
|
@pytest.mark.parametrize("input, success", test_data_call)
|
|
@patch("mythril.laser.ethereum.instructions.get_call_parameters")
|
|
def test_staticness_call_concrete(f1, input, success):
|
|
# Arrange
|
|
state = get_global_state()
|
|
state.environment.static = True
|
|
state.mstate.stack = [10] * 100
|
|
code = Disassembly(code="616263")
|
|
f1.return_value = ("0", Account(code=code, address="0x19"), 0, input, 0, 0, 0)
|
|
instruction = Instruction("call", dynamic_loader=None)
|
|
|
|
# Act and Assert
|
|
if success:
|
|
with pytest.raises(TransactionStartSignal) as ts:
|
|
instruction.evaluate(state)
|
|
assert ts.value.transaction.static
|
|
else:
|
|
with pytest.raises(WriteProtection):
|
|
instruction.evaluate(state)
|
|
|
|
|
|
@patch("mythril.laser.ethereum.instructions.get_call_parameters")
|
|
def test_staticness_call_symbolic(f1):
|
|
# Arrange
|
|
state = get_global_state()
|
|
state.environment.static = True
|
|
state.mstate.stack = [10] * 100
|
|
call_value = symbol_factory.BitVecSym("x", 256)
|
|
code = Disassembly(code="616263")
|
|
f1.return_value = ("0", Account(code=code, address="0x19"), 0, call_value, 0, 0, 0)
|
|
instruction = Instruction("call", dynamic_loader=None)
|
|
|
|
# Act and Assert
|
|
with pytest.raises(TransactionStartSignal) as ts:
|
|
instruction.evaluate(state)
|
|
|
|
assert ts.value.transaction.static
|
|
assert ts.value.global_state.world_state.constraints[-1] == (call_value == 0)
|
|
|