స్మార్ట్ కాంట్రాక్ట్లలో బగ్లను కనుగొనడానికి మాంటికోర్ను ఎలా ఉపయోగించాలి
ఈ ట్యుటోరియల్ యొక్క లక్ష్యం స్మార్ట్ కాంట్రాక్ట్లలో స్వయంచాలకంగా బగ్లను కనుగొనడానికి మాంటికోర్ను ఎలా ఉపయోగించాలో చూపించడం.
సంస్థాపన
మాంటికోర్కు >= పైథాన్ 3.6 అవసరం. దీనిని పిప్ ద్వారా లేదా డాకర్ని ఉపయోగించి ఇన్స్టాల్ చేయవచ్చు.
డాకర్ ద్వారా మాంటికోర్
docker pull trailofbits/eth-security-toolboxdocker run -it -v "$PWD":/home/training trailofbits/eth-security-toolboxచివరి కమాండ్ మీ ప్రస్తుత డైరెక్టరీకి యాక్సెస్ ఉన్న డాకర్లో eth-security-toolboxను రన్ చేస్తుంది. మీరు మీ హోస్ట్ నుండి ఫైల్లను మార్చవచ్చు, మరియు డాకర్ నుండి ఫైల్లపై సాధనాలను అమలు చేయండి
డాకర్ లోపల, రన్ చేయండి:
solc-select 0.5.11cd /home/trufflecon/పిప్ ద్వారా మాంటికోర్
pip3 install --user manticoresolc 0.5.11 సిఫార్సు చేయబడింది.
ఒక స్క్రిప్ట్ను అమలు చేయడం
పైథాన్ 3 తో పైథాన్ స్క్రిప్ట్ను అమలు చేయడానికి:
python3 script.pyడైనమిక్ సింబాలిక్ ఎగ్జిక్యూషన్కు పరిచయం
సంక్షిప్తంగా డైనమిక్ సింబాలిక్ ఎగ్జిక్యూషన్
డైనమిక్ సింబాలిక్ ఎగ్జిక్యూషన్ (DSE) అనేది అధిక స్థాయి సెమాంటిక్ అవగాహనతో ఒక స్టేట్ స్పేస్ను అన్వేషించే ఒక ప్రోగ్రామ్ విశ్లేషణ పద్ధతి. ఈ పద్ధతి "ప్రోగ్రామ్ పాత్ల" ఆవిష్కరణపై ఆధారపడి ఉంటుంది, వీటిని పాత్ ప్రిడికేట్స్ అని పిలిచే గణిత సూత్రాలుగా సూచిస్తారు. సంభావితంగా, ఈ పద్ధతి పాత్ ప్రిడికేట్లపై రెండు దశలలో పనిచేస్తుంది:
- ప్రోగ్రామ్ ఇన్పుట్పై పరిమితులను ఉపయోగించి అవి నిర్మించబడ్డాయి.
- అనుబంధిత పాత్లను అమలు చేయడానికి కారణమయ్యే ప్రోగ్రామ్ ఇన్పుట్లను రూపొందించడానికి అవి ఉపయోగించబడతాయి.
గుర్తించబడిన అన్ని ప్రోగ్రామ్ స్టేట్లు కాంక్రీట్ ఎగ్జిక్యూషన్ సమయంలో ప్రేరేపించబడతాయి అనే అర్థంలో ఈ విధానం ఎటువంటి తప్పుడు పాజిటివ్లను ఉత్పత్తి చేయదు. ఉదాహరణకు, విశ్లేషణలో ఒక పూర్ణాంక ఓవర్ఫ్లో కనుగొనబడితే, అది పునరుత్పత్తి చేయబడుతుందని హామీ ఇవ్వబడింది.
పాత్ ప్రిడికేట్ ఉదాహరణ
DSE ఎలా పనిచేస్తుందో ఒక అవగాహన పొందడానికి, కింది ఉదాహరణను పరిగణించండి:
1function f(uint a){23 if (a == 65) {4 // ఒక బగ్ ఉంది5 }67}f() రెండు పాత్లను కలిగి ఉన్నందున, ఒక DSE రెండు వేర్వేరు పాత్ ప్రిడికేట్లను నిర్మిస్తుంది:
- పాత్ 1:
a == 65 - పాత్ 2:
కాదు (a == 65)
ప్రతి పాత్ ప్రిడికేట్ ఒక గణిత సూత్రం, దీనిని SMT సాల్వర్ (opens in a new tab) అని పిలవబడే దానికి ఇవ్వవచ్చు, అది సమీకరణాన్ని పరిష్కరించడానికి ప్రయత్నిస్తుంది. పాత్ 1 కోసం, సాల్వర్ a = 65 తో పాత్ను అన్వేషించవచ్చని చెబుతుంది. పాత్ 2 కోసం, సాల్వర్ a కు 65 కాకుండా ఏదైనా విలువను ఇవ్వగలదు, ఉదాహరణకు a = 0.
లక్షణాలను ధృవీకరించడం
మాంటికోర్ ప్రతి పాత్ యొక్క అన్ని ఎగ్జిక్యూషన్పై పూర్తి నియంత్రణను అనుమతిస్తుంది. ఫలితంగా, ఇది దాదాపు దేనికైనా యాదృచ్ఛిక పరిమితులను జోడించడానికి మిమ్మల్ని అనుమతిస్తుంది. ఈ నియంత్రణ కాంట్రాక్ట్పై లక్షణాల సృష్టికి అనుమతిస్తుంది.
కింది ఉదాహరణను పరిగణించండి:
1function unsafe_add(uint a, uint b) returns(uint c){2 c = a + b; // ఓవర్ఫ్లో రక్షణ లేదు3 return c;4}ఇక్కడ ఫంక్షన్లో అన్వేషించడానికి ఒకే ఒక్క పాత్ ఉంది:
- పాత్ 1:
c = a + b
మాంటికోర్ను ఉపయోగించి, మీరు ఓవర్ఫ్లో కోసం తనిఖీ చేయవచ్చు, మరియు పాత్ ప్రిడికేట్కు పరిమితులను జోడించవచ్చు:
c = a + b AND (c < a OR c < b)
పైన ఉన్న పాత్ ప్రిడికేట్ సాధ్యమయ్యే a మరియు b యొక్క మూల్యాంకనాన్ని కనుగొనడం సాధ్యమైతే, మీరు ఒక ఓవర్ఫ్లోను కనుగొన్నారని అర్థం. ఉదాహరణకు సాల్వర్ ఇన్పుట్ a = 10 , b = MAXUINT256ను రూపొందించగలదు.
మీరు ఒక స్థిరమైన సంస్కరణను పరిగణించినట్లయితే:
1function safe_add(uint a, uint b) returns(uint c){2 c = a + b;3 require(c>=a);4 require(c>=b);5 return c;6}ఓవర్ఫ్లో తనిఖీతో అనుబంధిత సూత్రం ఇలా ఉంటుంది:
c = a + b AND (c >= a) AND (c=>b) AND (c < a OR c < b)
ఈ సూత్రాన్ని పరిష్కరించలేము; మరో మాటలో చెప్పాలంటే, ఇది safe_addలో c ఎల్లప్పుడూ పెరుగుతుందనడానికి ఒక రుజువు.
అందువల్ల DSE ఒక శక్తివంతమైన సాధనం, ఇది మీ కోడ్పై యాదృచ్ఛిక పరిమితులను ధృవీకరించగలదు.
మాంటికోర్ కింద నడుస్తోంది
మాంటికోర్ API తో స్మార్ట్ కాంట్రాక్ట్ను ఎలా అన్వేషించాలో మనం చూస్తాము. లక్ష్యం కింది స్మార్ట్ కాంట్రాక్ట్ example.sol (opens in a new tab):
1pragma solidity >=0.4.24 <0.6.0;23contract Simple {4 function f(uint a) payable public{5 if (a == 65) {6 revert();7 }8 }9}అన్నీ చూపించుఒక స్వతంత్ర అన్వేషణను అమలు చేయండి
మీరు కింది కమాండ్ ద్వారా నేరుగా స్మార్ట్ కాంట్రాక్ట్పై మాంటికోర్ను అమలు చేయవచ్చు (project ఒక సాలిడిటీ ఫైల్ లేదా ప్రాజెక్ట్ డైరెక్టరీ కావచ్చు):
$ manticore projectమీరు ఇలాంటి టెస్ట్కేసుల అవుట్పుట్ను పొందుతారు (ఆర్డర్ మారవచ్చు):
1...2... m.c.manticore:INFO: Generated testcase No. 0 - STOP3... m.c.manticore:INFO: Generated testcase No. 1 - REVERT4... m.c.manticore:INFO: Generated testcase No. 2 - RETURN5... m.c.manticore:INFO: Generated testcase No. 3 - REVERT6... m.c.manticore:INFO: Generated testcase No. 4 - STOP7... m.c.manticore:INFO: Generated testcase No. 5 - REVERT8... m.c.manticore:INFO: Generated testcase No. 6 - REVERT9... m.c.manticore:INFO: Results in /home/ethsec/workshops/Automated Smart Contracts Audit - TruffleCon 2018/manticore/examples/mcore_t6vi6ij310...అన్నీ చూపించుఅదనపు సమాచారం లేకుండా, మాంటికోర్ కొత్త సింబాలిక్ లావాదేవీలతో కాంట్రాక్ట్ను అన్వేషిస్తుంది, అది కాంట్రాక్ట్లో కొత్త పాత్లను అన్వేషించే వరకు. విఫలమైన దాని తర్వాత మాంటికోర్ కొత్త లావాదేవీలను అమలు చేయదు (ఉదా: రివర్ట్ తర్వాత).
మాంటికోర్ సమాచారాన్ని mcore_* డైరెక్టరీలో అవుట్పుట్ చేస్తుంది. ఇతర వాటితో పాటు, మీరు ఈ డైరెక్టరీలో కనుగొంటారు:
global.summary: కవరేజ్ మరియు కంపైలర్ హెచ్చరికలుtest_XXXXX.summary: కవరేజ్, చివరి సూచన, ప్రతి టెస్ట్ కేసుకు ఖాతా బ్యాలెన్స్లుtest_XXXXX.tx: ప్రతి టెస్ట్ కేసుకు లావాదేవీల వివరణాత్మక జాబితా
ఇక్కడ మాంటికోర్ 7 టెస్ట్ కేసులను కనుగొంది, అవి వీటికి అనుగుణంగా ఉంటాయి (ఫైల్నేమ్ ఆర్డర్ మారవచ్చు):
| లావాదేవీ 0 | లావాదేవీ 1 | లావాదేవీ 2 | ఫలితం | |
|---|---|---|---|---|
| test_00000000.tx | కాంట్రాక్ట్ సృష్టి | f(!=65) | f(!=65) | STOP |
| test_00000001.tx | కాంట్రాక్ట్ సృష్టి | ఫాల్బ్యాక్ ఫంక్షన్ | REVERT | |
| test_00000002.tx | కాంట్రాక్ట్ సృష్టి | RETURN | ||
| test_00000003.tx | కాంట్రాక్ట్ సృష్టి | f(65) | REVERT | |
| test_00000004.tx | కాంట్రాక్ట్ సృష్టి | f(!=65) | STOP | |
| test_00000005.tx | కాంట్రాక్ట్ సృష్టి | f(!=65) | f(65) | REVERT |
| test_00000006.tx | కాంట్రాక్ట్ సృష్టి | f(!=65) | ఫాల్బ్యాక్ ఫంక్షన్ | REVERT |
అన్వేషణ సారాంశం f(!=65) అంటే 65 కంటే భిన్నమైన ఏదైనా విలువతో f పిలువబడిందని సూచిస్తుంది.
మీరు గమనించినట్లుగా, మాంటికోర్ ప్రతి విజయవంతమైన లేదా రివర్ట్ చేయబడిన లావాదేవీకి ఒక ప్రత్యేకమైన టెస్ట్ కేసును ఉత్పత్తి చేస్తుంది.
మీరు వేగవంతమైన కోడ్ అన్వేషణను కోరుకుంటే --quick-mode ఫ్లాగ్ను ఉపయోగించండి (ఇది బగ్ డిటెక్టర్లు, గ్యాస్ గణన మొదలైనవాటిని నిలిపివేస్తుంది)
API ద్వారా స్మార్ట్ కాంట్రాక్ట్ను మార్చడం
ఈ విభాగం మాంటికోర్ పైథాన్ API ద్వారా స్మార్ట్ కాంట్రాక్ట్ను ఎలా మార్చాలో వివరిస్తుంది. మీరు పైథాన్ ఎక్స్టెన్షన్ *.py తో కొత్త ఫైల్ను సృష్టించవచ్చు మరియు API ఆదేశాలను (వీటి ప్రాథమికాలు క్రింద వివరించబడతాయి) ఈ ఫైల్లోకి జోడించడం ద్వారా అవసరమైన కోడ్ను వ్రాయవచ్చు, ఆపై $ python3 *.py ఆదేశంతో దాన్ని అమలు చేయండి. అలాగే మీరు కింది ఆదేశాలను నేరుగా పైథాన్ కన్సోల్లోకి అమలు చేయవచ్చు, కన్సోల్ను అమలు చేయడానికి $ python3 ఆదేశాన్ని ఉపయోగించండి.
ఖాతాలను సృష్టించడం
మీరు చేయవలసిన మొదటి విషయం కింది ఆదేశాలతో కొత్త బ్లాక్ చైనును ప్రారంభించడం:
1from manticore.ethereum import ManticoreEVM23m = ManticoreEVM()ఒక నాన్-కాంట్రాక్ట్ ఖాతా m.create_account (opens in a new tab) ఉపయోగించి సృష్టించబడుతుంది:
1user_account = m.create_account(balance=1000)ఒక సాలిడిటీ కాంట్రాక్ట్ను m.solidity_create_contract (opens in a new tab) ఉపయోగించి అమలు చేయవచ్చు:
1source_code = '''2pragma solidity >=0.4.24 <0.6.0;3contract Simple {4 function f(uint a) payable public{5 if (a == 65) {6 revert();7 }8 }9}10'''11# Initiate the contract12contract_account = m.solidity_create_contract(source_code, owner=user_account)అన్నీ చూపించుసారాంశం
- మీరు m.create_account (opens in a new tab) మరియు m.solidity_create_contract (opens in a new tab) తో వినియోగదారు మరియు కాంట్రాక్ట్ ఖాతాలను సృష్టించవచ్చు.
లావాదేవీలను అమలు చేయడం
మాంటికోర్ రెండు రకాల లావాదేవీలకు మద్దతు ఇస్తుంది:
- రా లావాదేవీ: అన్ని ఫంక్షన్లు అన్వేషించబడతాయి
- పేరున్న లావాదేవీ: ఒకే ఒక ఫంక్షన్ అన్వేషించబడుతుంది
రా లావాదేవీ
ఒక రా లావాదేవీ m.transaction (opens in a new tab) ఉపయోగించి అమలు చేయబడుతుంది:
1m.transaction(caller=user_account,2 address=contract_account,3 data=data,4 value=value)కాలర్, చిరునామా, డేటా, లేదా లావాదేవీ యొక్క విలువ కాంక్రీట్ లేదా సింబాలిక్ కావచ్చు:
- m.make_symbolic_value (opens in a new tab) ఒక సింబాలిక్ విలువను సృష్టిస్తుంది.
- m.make_symbolic_buffer(size) (opens in a new tab) ఒక సింబాలిక్ బైట్ శ్రేణిని సృష్టిస్తుంది.
ఉదాహరణకి:
1symbolic_value = m.make_symbolic_value()2symbolic_data = m.make_symbolic_buffer(320)3m.transaction(caller=user_account,4 address=contract_address,5 data=symbolic_data,6 value=symbolic_value)డేటా సింబాలిక్ అయితే, మాంటికోర్ లావాదేవీ అమలు సమయంలో కాంట్రాక్ట్ యొక్క అన్ని ఫంక్షన్లను అన్వేషిస్తుంది. ఫంక్షన్ ఎంపిక ఎలా పనిచేస్తుందో అర్థం చేసుకోవడానికి హ్యాండ్స్ ఆన్ ది ఈథర్నాట్ CTF (opens in a new tab) కథనంలో ఫాల్బ్యాక్ ఫంక్షన్ వివరణను చూడటం సహాయకరంగా ఉంటుంది.
పేరున్న లావాదేవీ
ఫంక్షన్లను వాటి పేరు ద్వారా అమలు చేయవచ్చు.
సింబాలిక్ విలువతో f(uint var) ను user_account నుండి, మరియు 0 ఈథర్తో అమలు చేయడానికి, ఉపయోగించండి:
1symbolic_var = m.make_symbolic_value()2contract_account.f(symbolic_var, caller=user_account, value=0)లావాదేవీ యొక్క value పేర్కొనకపోతే, అది డిఫాల్ట్గా 0 అవుతుంది.
సారాంశం
- ఒక లావాదేవీ యొక్క వాదనలు కాంక్రీట్ లేదా సింబాలిక్ కావచ్చు
- ఒక రా లావాదేవీ అన్ని ఫంక్షన్లను అన్వేషిస్తుంది
- ఫంక్షన్ను వాటి పేరు ద్వారా పిలవవచ్చు
వర్క్స్పేస్
m.workspace అనేది రూపొందించబడిన అన్ని ఫైల్ల కోసం అవుట్పుట్ డైరెక్టరీగా ఉపయోగించబడే డైరెక్టరీ:
1print("Results are in {}".format(m.workspace))అన్వేషణను ముగించండి
అన్వేషణను ఆపడానికి m.finalize() (opens in a new tab)ని ఉపయోగించండి. ఈ పద్ధతిని పిలిచిన తర్వాత తదుపరి లావాదేవీలు పంపకూడదు మరియు మాంటికోర్ అన్వేషించబడిన ప్రతి పాత్కు టెస్ట్ కేసులను ఉత్పత్తి చేస్తుంది.
సారాంశం: మాంటికోర్ కింద నడుస్తోంది
మునుపటి దశలన్నింటినీ కలిపి, మనం పొందుతాము:
1from manticore.ethereum import ManticoreEVM23m = ManticoreEVM()45with open('example.sol') as f:6 source_code = f.read()78user_account = m.create_account(balance=1000)9contract_account = m.solidity_create_contract(source_code, owner=user_account)1011symbolic_var = m.make_symbolic_value()12contract_account.f(symbolic_var)1314print("Results are in {}".format(m.workspace))15m.finalize() # stop the explorationఅన్నీ చూపించుపై కోడ్ అంతా మీరు example_run.py (opens in a new tab)లో కనుగొనవచ్చు
త్రోయింగ్ పాత్లను పొందడం
ఇప్పుడు మనం f()లో మినహాయింపును పెంచే పాత్ల కోసం నిర్దిష్ట ఇన్పుట్లను రూపొందిస్తాము. లక్ష్యం ఇప్పటికీ కింది స్మార్ట్ కాంట్రాక్ట్ example.sol (opens in a new tab):
1pragma solidity >=0.4.24 <0.6.0;2contract Simple {3 function f(uint a) payable public{4 if (a == 65) {5 revert();6 }7 }8}స్టేట్ సమాచారాన్ని ఉపయోగించడం
అమలు చేయబడిన ప్రతి పాత్ బ్లాక్ చైను యొక్క దాని స్థితిని కలిగి ఉంటుంది. ఒక స్థితి సిద్ధంగా ఉంటుంది లేదా అది చంపబడుతుంది, అంటే అది THROW లేదా REVERT సూచనను చేరుకుంటుంది:
- m.ready_states (opens in a new tab): సిద్ధంగా ఉన్న స్థితుల జాబితా (అవి REVERT/INVALID ను అమలు చేయలేదు)
- m.killed_states (opens in a new tab): చంపబడిన స్థితుల జాబితా
- m.all_states (opens in a new tab): అన్ని స్థితులు
1for state in m.all_states:2 # స్టేట్తో ఏదైనా చేయండిమీరు స్థితి సమాచారాన్ని యాక్సెస్ చేయవచ్చు. ఉదాహరణకి:
state.platform.get_balance(account.address): ఖాతా యొక్క బ్యాలెన్స్state.platform.transactions: లావాదేవీల జాబితాstate.platform.transactions[-1].return_data: చివరి లావాదేవీ ద్వారా తిరిగి ఇవ్వబడిన డేటా
చివరి లావాదేవీ ద్వారా తిరిగి ఇవ్వబడిన డేటా ఒక శ్రేణి, దీనిని ABI.deserialize తో ఒక విలువగా మార్చవచ్చు, ఉదాహరణకు:
1data = state.platform.transactions[0].return_data2data = ABI.deserialize("uint", data)టెస్ట్కేస్ను ఎలా రూపొందించాలి
టెస్ట్కేస్ను రూపొందించడానికి m.generate_testcase(state, name) (opens in a new tab)ని ఉపయోగించండి:
1m.generate_testcase(state, 'BugFound')సారాంశం
- మీరు m.all_states తో స్థితిపై పునరావృతం చేయవచ్చు
state.platform.get_balance(account.address)ఖాతా బ్యాలెన్స్ను తిరిగి ఇస్తుందిstate.platform.transactionsలావాదేవీల జాబితాను తిరిగి ఇస్తుందిtransaction.return_dataఅనేది తిరిగి ఇవ్వబడిన డేటాm.generate_testcase(state, name)స్థితి కోసం ఇన్పుట్లను ఉత్పత్తి చేస్తుంది
సారాంశం: త్రోయింగ్ పాత్ను పొందడం
1from manticore.ethereum import ManticoreEVM23m = ManticoreEVM()45with open('example.sol') as f:6 source_code = f.read()78user_account = m.create_account(balance=1000)9contract_account = m.solidity_create_contract(source_code, owner=user_account)1011symbolic_var = m.make_symbolic_value()12contract_account.f(symbolic_var)1314## Check if an execution ends with a REVERT or INVALID1516for state in m.terminated_states:17 last_tx = state.platform.transactions[-1]18 if last_tx.result in ['REVERT', 'INVALID']:19 print('Throw found {}'.format(m.workspace))20 m.generate_testcase(state, 'ThrowFound')అన్నీ చూపించుపై కోడ్ అంతా మీరు example_run.py (opens in a new tab)లో కనుగొనవచ్చు
గమనిక మేము చాలా సరళమైన స్క్రిప్ట్ను రూపొందించగలిగేవాళ్లం, ఎందుకంటే terminated_state ద్వారా తిరిగి ఇవ్వబడిన అన్ని స్థితుల ఫలితంలో REVERT లేదా INVALID ఉంది: ఈ ఉదాహరణ కేవలం APIని ఎలా మార్చాలో ప్రదర్శించడానికి మాత్రమే ఉద్దేశించబడింది.
పరిమితులను జోడించడం
అన్వేషణను ఎలా నియంత్రించాలో చూద్దాం. ఫంక్షన్ ఎప్పుడూ a == 65 తో పిలువబడదని f() యొక్క డాక్యుమెంటేషన్ పేర్కొంటుందని మేము ఊహిస్తాము, కాబట్టి a == 65 తో ఉన్న ఏ బగ్ అయినా నిజమైన బగ్ కాదు. లక్ష్యం ఇప్పటికీ కింది స్మార్ట్ కాంట్రాక్ట్ example.sol (opens in a new tab):
1pragma solidity >=0.4.24 <0.6.0;2contract Simple {3 function f(uint a) payable public{4 if (a == 65) {5 revert();6 }7 }8}ఆపరేటర్లు
ఆపరేటర్లు (opens in a new tab) మాడ్యూల్ పరిమితుల మార్పును సులభతరం చేస్తుంది, ఇతర వాటితో పాటు ఇది అందిస్తుంది:
- Operators.AND,
- Operators.OR,
- Operators.UGT (unsigned greater than),
- Operators.UGE (unsigned greater than or equal to),
- Operators.ULT (unsigned lower than),
- Operators.ULE (unsigned lower than or equal to).
మాడ్యూల్ను దిగుమతి చేయడానికి కింది వాటిని ఉపయోగించండి:
1from manticore.core.smtlib import OperatorsOperators.CONCAT అనేది ఒక శ్రేణిని ఒక విలువకు కలపడానికి ఉపయోగించబడుతుంది. ఉదాహరణకు, ఒక లావాదేవీ యొక్క return_data ను మరొక విలువతో సరిపోల్చడానికి ఒక విలువగా మార్చాలి:
1last_return = Operators.CONCAT(256, *last_return)పరిమితులు
మీరు పరిమితులను ప్రపంచవ్యాప్తంగా లేదా ఒక నిర్దిష్ట స్థితి కోసం ఉపయోగించవచ్చు.
గ్లోబల్ పరిమితి
గ్లోబల్ పరిమితిని జోడించడానికి m.constrain(constraint) ను ఉపయోగించండి.
ఉదాహరణకు, మీరు ఒక సింబాలిక్ చిరునామా నుండి ఒక కాంట్రాక్ట్ను పిలవవచ్చు మరియు ఈ చిరునామాను నిర్దిష్ట విలువలకు పరిమితం చేయవచ్చు:
1symbolic_address = m.make_symbolic_value()2m.constraint(Operators.OR(symbolic == 0x41, symbolic_address == 0x42))3m.transaction(caller=user_account,4 address=contract_account,5 data=m.make_symbolic_buffer(320),6 value=0)స్థితి పరిమితి
ఒక నిర్దిష్ట స్థితికి పరిమితిని జోడించడానికి state.constrain(constraint) (opens in a new tab)ని ఉపయోగించండి. దానిపై కొన్ని లక్షణాలను తనిఖీ చేయడానికి దాని అన్వేషణ తర్వాత స్థితిని పరిమితం చేయడానికి దీనిని ఉపయోగించవచ్చు.
పరిమితిని తనిఖీ చేయడం
ఒక పరిమితి ఇప్పటికీ సాధ్యమేనా అని తెలుసుకోవడానికి solver.check(state.constraints) ను ఉపయోగించండి.
ఉదాహరణకు, కిందిది symbolic_value ను 65 కంటే భిన్నంగా ఉండేలా పరిమితం చేస్తుంది మరియు స్థితి ఇప్పటికీ సాధ్యమేనా అని తనిఖీ చేస్తుంది:
1state.constrain(symbolic_var != 65)2if solver.check(state.constraints):3 # state is feasibleసారాంశం: పరిమితులను జోడించడం
మునుపటి కోడ్కు పరిమితిని జోడిస్తే, మనం పొందుతాము:
1from manticore.ethereum import ManticoreEVM2from manticore.core.smtlib.solver import Z3Solver34solver = Z3Solver.instance()56m = ManticoreEVM()78with open("example.sol") as f:9 source_code = f.read()1011user_account = m.create_account(balance=1000)12contract_account = m.solidity_create_contract(source_code, owner=user_account)1314symbolic_var = m.make_symbolic_value()15contract_account.f(symbolic_var)1617no_bug_found = True1819## ఒక ఎగ్జిక్యూషన్ REVERT లేదా INVALID తో ముగుస్తుందో లేదో తనిఖీ చేయండి2021for state in m.terminated_states:22 last_tx = state.platform.transactions[-1]23 if last_tx.result in ['REVERT', 'INVALID']:24 # a == 65 అయిన పాత్ను మేము పరిగణించము25 condition = symbolic_var != 6526 if m.generate_testcase(state, name="BugFound", only_if=condition):27 print(f'బగ్ కనుగొనబడింది, ఫలితాలు {m.workspace} లో ఉన్నాయి')28 no_bug_found = False2930if no_bug_found:31 print(f'బగ్ ఏదీ కనుగొనబడలేదు')అన్నీ చూపించుపై కోడ్ అంతా మీరు example_run.py (opens in a new tab)లో కనుగొనవచ్చు
పేజీ చివరి అప్డేట్: 26 ఏప్రిల్, 2024