mirror of https://github.com/crytic/slither
commit
07ea324a1a
@ -0,0 +1,141 @@ |
|||||||
|
""" |
||||||
|
""" |
||||||
|
|
||||||
|
import json |
||||||
|
from collections import defaultdict |
||||||
|
from slither.printers.abstract_printer import AbstractPrinter |
||||||
|
from slither.core.declarations.solidity_variables import SolidityVariableComposed, SolidityFunction |
||||||
|
from slither.slithir.operations.binary import Binary, BinaryType |
||||||
|
from slither.core.variables.state_variable import StateVariable |
||||||
|
from slither.slithir.variables import Constant |
||||||
|
|
||||||
|
|
||||||
|
def _extract_payable(slither): |
||||||
|
ret = {} |
||||||
|
for contract in slither.contracts: |
||||||
|
payable_functions = [f.full_name for f in contract.functions_entry_points if f.payable] |
||||||
|
if payable_functions: |
||||||
|
ret[contract.name] = payable_functions |
||||||
|
return ret |
||||||
|
|
||||||
|
def _extract_solidity_variable_usage(slither, sol_var): |
||||||
|
ret = {} |
||||||
|
for contract in slither.contracts: |
||||||
|
functions_using_sol_var = [] |
||||||
|
for f in contract.functions_entry_points: |
||||||
|
for v in f.all_solidity_variables_read(): |
||||||
|
if v == sol_var: |
||||||
|
functions_using_sol_var.append(f.full_name) |
||||||
|
break |
||||||
|
if functions_using_sol_var: |
||||||
|
ret[contract.name] = functions_using_sol_var |
||||||
|
return ret |
||||||
|
|
||||||
|
def _extract_constant_functions(slither): |
||||||
|
ret = {} |
||||||
|
for contract in slither.contracts: |
||||||
|
cst_functions = [f.full_name for f in contract.functions_entry_points if f.view or f.pure] |
||||||
|
cst_functions += [v.function_name for v in contract.state_variables if v.visibility in ['public']] |
||||||
|
if cst_functions: |
||||||
|
ret[contract.name] = cst_functions |
||||||
|
return ret |
||||||
|
|
||||||
|
def _extract_assert(slither): |
||||||
|
ret = {} |
||||||
|
for contract in slither.contracts: |
||||||
|
functions_using_assert = [] |
||||||
|
for f in contract.functions_entry_points: |
||||||
|
for v in f.all_solidity_calls(): |
||||||
|
if v == SolidityFunction('assert(bool)'): |
||||||
|
functions_using_assert.append(f.full_name) |
||||||
|
break |
||||||
|
if functions_using_assert: |
||||||
|
ret[contract.name] = functions_using_assert |
||||||
|
return ret |
||||||
|
|
||||||
|
def _extract_constants_from_irs(irs, all_cst_used, all_cst_used_in_binary, context_explored): |
||||||
|
for ir in irs: |
||||||
|
if isinstance(ir, Binary): |
||||||
|
for r in ir.read: |
||||||
|
if isinstance(r, Constant): |
||||||
|
all_cst_used_in_binary[BinaryType.str(ir.type)].append(r.value) |
||||||
|
for r in ir.read: |
||||||
|
if isinstance(r, Constant): |
||||||
|
all_cst_used.append(r.value) |
||||||
|
if isinstance(r, StateVariable): |
||||||
|
if r.node_initialization: |
||||||
|
if r.node_initialization.irs: |
||||||
|
if r.node_initialization in context_explored: |
||||||
|
continue |
||||||
|
else: |
||||||
|
context_explored.add(r.node_initialization) |
||||||
|
_extract_constants_from_irs(r.node_initialization.irs, |
||||||
|
all_cst_used, |
||||||
|
all_cst_used_in_binary, |
||||||
|
context_explored) |
||||||
|
|
||||||
|
def _extract_constants(slither): |
||||||
|
ret_cst_used = defaultdict(dict) |
||||||
|
ret_cst_used_in_binary = defaultdict(dict) |
||||||
|
for contract in slither.contracts: |
||||||
|
for function in contract.functions_entry_points: |
||||||
|
all_cst_used = [] |
||||||
|
all_cst_used_in_binary = defaultdict(list) |
||||||
|
|
||||||
|
context_explored = set() |
||||||
|
context_explored.add(function) |
||||||
|
_extract_constants_from_irs(function.all_slithir_operations(), |
||||||
|
all_cst_used, |
||||||
|
all_cst_used_in_binary, |
||||||
|
context_explored) |
||||||
|
|
||||||
|
if all_cst_used: |
||||||
|
ret_cst_used[contract.name][function.full_name] = all_cst_used |
||||||
|
if all_cst_used_in_binary: |
||||||
|
ret_cst_used_in_binary[contract.name][function.full_name] = all_cst_used_in_binary |
||||||
|
return (ret_cst_used, ret_cst_used_in_binary) |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class Echidna(AbstractPrinter): |
||||||
|
ARGUMENT = 'echidna' |
||||||
|
HELP = 'todo' |
||||||
|
|
||||||
|
WIKI = 'https://github.com/trailofbits/slither/wiki/Printer-documentation#echidna' |
||||||
|
|
||||||
|
|
||||||
|
def output(self, filename): |
||||||
|
""" |
||||||
|
Output the inheritance relation |
||||||
|
|
||||||
|
_filename is not used |
||||||
|
Args: |
||||||
|
_filename(string) |
||||||
|
""" |
||||||
|
|
||||||
|
payable = _extract_payable(self.slither) |
||||||
|
timestamp = _extract_solidity_variable_usage(self.slither, |
||||||
|
SolidityVariableComposed('block.timestamp')) |
||||||
|
block_number = _extract_solidity_variable_usage(self.slither, |
||||||
|
SolidityVariableComposed('block.number')) |
||||||
|
msg_sender = _extract_solidity_variable_usage(self.slither, |
||||||
|
SolidityVariableComposed('msg.sender')) |
||||||
|
msg_gas = _extract_solidity_variable_usage(self.slither, |
||||||
|
SolidityVariableComposed('msg.gas')) |
||||||
|
assert_usage = _extract_assert(self.slither) |
||||||
|
cst_functions = _extract_constant_functions(self.slither) |
||||||
|
(cst_used, cst_used_in_binary) = _extract_constants(self.slither) |
||||||
|
|
||||||
|
|
||||||
|
d = {'payable': payable, |
||||||
|
'timestamp': timestamp, |
||||||
|
'block_number': block_number, |
||||||
|
'msg_sender': msg_sender, |
||||||
|
'msg_gas': msg_gas, |
||||||
|
'assert': assert_usage, |
||||||
|
'constant_functions': cst_functions, |
||||||
|
'constants_used': cst_used, |
||||||
|
'constants_used_in_binary': cst_used_in_binary} |
||||||
|
|
||||||
|
print(json.dumps(d, indent=4)) |
Loading…
Reference in new issue