ஸ்மார்ட் ஒப்பந்தப் பிழைகளைக் கண்டறிய Slither-ஐ எவ்வாறு பயன்படுத்துவது
Slither-ஐ எவ்வாறு பயன்படுத்துவது
ஸ்மார்ட் ஒப்பந்தங்களில் உள்ள பிழைகளைத் தானாகவே கண்டறிய Slither-ஐ எவ்வாறு பயன்படுத்துவது என்பதைக் காட்டுவதே இந்தப் பயிற்சியின் நோக்கமாகும்.
- நிறுவல்
- கட்டளை வரிப் பயன்பாடு
- நிலையான பகுப்பாய்வுக்கான அறிமுகம்: நிலையான பகுப்பாய்வு பற்றிய சுருக்கமான அறிமுகம்
- API: Python API விளக்கம்
நிறுவல்
Slither-க்கு Python >= 3.6 தேவை. இதை pip மூலமாகவோ அல்லது docker-ஐப் பயன்படுத்தியோ நிறுவலாம்.
pip மூலம் Slither:
pip3 install --user slither-analyzerdocker மூலம் Slither:
docker pull trailofbits/eth-security-toolboxdocker run -it -v "$PWD":/home/trufflecon trailofbits/eth-security-toolboxகடைசி கட்டளையானது உங்கள் தற்போதைய கோப்பகத்திற்கான அணுகலைக் கொண்ட ஒரு docker-இல் eth-security-toolbox-ஐ இயக்குகிறது. உங்கள் ஹோஸ்டிலிருந்து கோப்புகளை மாற்றலாம், மேலும் docker-இலிருந்து கோப்புகளில் கருவிகளை இயக்கலாம்
docker-இன் உள்ளே, இயக்கவும்:
solc-select 0.5.11cd /home/trufflecon/ஸ்கிரிப்டை இயக்குதல்
python 3 உடன் python ஸ்கிரிப்டை இயக்க:
python3 script.pyகட்டளை வரி
கட்டளை வரி மற்றும் பயனர் வரையறுத்த ஸ்கிரிப்டுகள். பல பொதுவான பிழைகளைக் கண்டறியும் முன்வரையறுக்கப்பட்ட டிடெக்டர்களின் (detectors) தொகுப்புடன் Slither வருகிறது. கட்டளை வரியிலிருந்து Slither-ஐ அழைப்பது அனைத்து டிடெக்டர்களையும் இயக்கும், நிலையான பகுப்பாய்வு பற்றிய விரிவான அறிவு தேவையில்லை:
slither project_pathsடிடெக்டர்களுக்கு கூடுதலாக, Slither அதன் printers (opens in a new tab) மற்றும் tools (opens in a new tab) மூலம் குறியீடு மதிப்பாய்வு திறன்களைக் கொண்டுள்ளது.
தனிப்பட்ட டிடெக்டர்கள் மற்றும் GitHub ஒருங்கிணைப்பிற்கான அணுகலைப் பெற crytic.io (opens in a new tab)-ஐப் பயன்படுத்தவும்.
நிலையான பகுப்பாய்வு
Slither நிலையான பகுப்பாய்வு கட்டமைப்பின் திறன்கள் மற்றும் வடிவமைப்பு வலைப்பதிவு இடுகைகள் (1 (opens in a new tab), 2 (opens in a new tab)) மற்றும் ஒரு கல்வி ஆய்வுக் கட்டுரையில் (opens in a new tab) விவரிக்கப்பட்டுள்ளன.
நிலையான பகுப்பாய்வு வெவ்வேறு வகைகளில் உள்ளது. clang (opens in a new tab) மற்றும் gcc (opens in a new tab) போன்ற கம்பைலர்கள் இந்த ஆராய்ச்சி நுட்பங்களைச் சார்ந்து இருப்பதை நீங்கள் பெரும்பாலும் உணர்ந்திருப்பீர்கள், ஆனால் இது (Infer (opens in a new tab), CodeClimate (opens in a new tab), FindBugs (opens in a new tab) மற்றும் Frama-C (opens in a new tab) மற்றும் Polyspace (opens in a new tab) போன்ற முறையான வழிமுறைகளை அடிப்படையாகக் கொண்ட கருவிகளுக்கும் அடிப்படையாக அமைகிறது.
நிலையான பகுப்பாய்வு நுட்பங்கள் மற்றும் ஆராய்ச்சியாளர்களை நாங்கள் இங்கு முழுமையாக மதிப்பாய்வு செய்யப் போவதில்லை. அதற்குப் பதிலாக, Slither எவ்வாறு செயல்படுகிறது என்பதைப் புரிந்துகொள்ள என்ன தேவை என்பதில் கவனம் செலுத்துவோம், இதன் மூலம் பிழைகளைக் கண்டறியவும் குறியீட்டைப் புரிந்துகொள்ளவும் நீங்கள் அதை மிகவும் திறம்படப் பயன்படுத்தலாம்.
குறியீடு பிரதிநிதித்துவம்
ஒற்றை செயலாக்கப் பாதையைப் பற்றி ஆராயும் மாறும் பகுப்பாய்விற்கு (dynamic analysis) மாறாக, நிலையான பகுப்பாய்வு அனைத்துப் பாதைகளையும் ஒரே நேரத்தில் ஆராய்கிறது. அவ்வாறு செய்ய, இது வேறுபட்ட குறியீடு பிரதிநிதித்துவத்தை நம்பியுள்ளது. சுருக்கமான தொடரியல் மரம் (abstract syntax tree - AST) மற்றும் கட்டுப்பாட்டு ஓட்ட வரைபடம் (control flow graph - CFG) ஆகியவை மிகவும் பொதுவான இரண்டு ஆகும்.
சுருக்கமான தொடரியல் மரங்கள் (AST)
கம்பைலர் குறியீட்டைப் பாகுபடுத்தும் ஒவ்வொரு முறையும் AST பயன்படுத்தப்படுகிறது. நிலையான பகுப்பாய்வைச் செய்யக்கூடிய மிக அடிப்படையான கட்டமைப்பு இதுவாகத்தான் இருக்கும்.
சுருக்கமாகச் சொன்னால், AST என்பது ஒரு கட்டமைக்கப்பட்ட மரமாகும், இதில் பொதுவாக, ஒவ்வொரு இலையும் ஒரு மாறி (variable) அல்லது மாறிலியைக் (constant) கொண்டிருக்கும் மற்றும் உள் முனைகள் (internal nodes) செயலிழப்புகள் (operands) அல்லது கட்டுப்பாட்டு ஓட்ட செயல்பாடுகளாகும். பின்வரும் குறியீட்டைக் கவனியுங்கள்:
1function safeAdd(uint a, uint b) pure internal returns(uint){2 if(a + b <= a){3 revert();4 }5 return a + b;6}அதனுடன் தொடர்புடைய AST இதில் காட்டப்பட்டுள்ளது:
solc மூலம் ஏற்றுமதி செய்யப்பட்ட AST-ஐ Slither பயன்படுத்துகிறது.
உருவாக்க எளிமையானதாக இருந்தாலும், AST என்பது ஒரு உள்ளமைக்கப்பட்ட (nested) கட்டமைப்பாகும். சில நேரங்களில், இது பகுப்பாய்வு செய்ய மிகவும் நேரடியானதாக இருக்காது. எடுத்துக்காட்டாக, a + b <= a என்ற வெளிப்பாட்டால் பயன்படுத்தப்படும் செயல்பாடுகளை அடையாளம் காண, நீங்கள் முதலில் <=-ஐயும் பின்னர் +-ஐயும் பகுப்பாய்வு செய்ய வேண்டும். மரத்தின் வழியாக சுழல்நிலையாக (recursively) செல்லும் விசிட்டர் பேட்டர்ன் (visitor pattern) என்று அழைக்கப்படுவதைப் பயன்படுத்துவது ஒரு பொதுவான அணுகுமுறையாகும். Slither-இல் ExpressionVisitor (opens in a new tab)-இல் ஒரு பொதுவான விசிட்டர் உள்ளது.
வெளிப்பாட்டில் கூட்டல் உள்ளதா என்பதைக் கண்டறிய பின்வரும் குறியீடு ExpressionVisitor-ஐப் பயன்படுத்துகிறது:
1from slither.visitors.expression.expression import ExpressionVisitor2from slither.core.expressions.binary_operation import BinaryOperationType3
4class HasAddition(ExpressionVisitor):5
6 def result(self):7 return self._result8
9 def _post_binary_operation(self, expression):10 if expression.type == BinaryOperationType.ADDITION:11 self._result = True12
13visitor = HasAddition(expression) # expression என்பது சோதிக்கப்பட வேண்டிய கோவை ஆகும்14print(f'The expression {expression} has a addition: {visitor.result()}')கட்டுப்பாட்டு ஓட்ட வரைபடம் (CFG)
இரண்டாவது மிகவும் பொதுவான குறியீடு பிரதிநிதித்துவம் கட்டுப்பாட்டு ஓட்ட வரைபடம் (CFG) ஆகும். அதன் பெயர் குறிப்பிடுவது போல, இது அனைத்து செயலாக்கப் பாதைகளையும் வெளிப்படுத்தும் வரைபட அடிப்படையிலான பிரதிநிதித்துவமாகும். ஒவ்வொரு முனையும் ஒன்று அல்லது பல வழிமுறைகளைக் கொண்டுள்ளது. வரைபடத்தில் உள்ள விளிம்புகள் கட்டுப்பாட்டு ஓட்ட செயல்பாடுகளைக் குறிக்கின்றன (if/then/else, loop போன்றவை). நமது முந்தைய எடுத்துக்காட்டின் CFG:
CFG என்பது பெரும்பாலான பகுப்பாய்வுகள் கட்டமைக்கப்பட்டுள்ள பிரதிநிதித்துவமாகும்.
வேறு பல குறியீடு பிரதிநிதித்துவங்கள் உள்ளன. நீங்கள் செய்ய விரும்பும் பகுப்பாய்வைப் பொறுத்து ஒவ்வொரு பிரதிநிதித்துவத்திற்கும் நன்மைகள் மற்றும் குறைபாடுகள் உள்ளன.
பகுப்பாய்வு
Slither மூலம் நீங்கள் செய்யக்கூடிய எளிமையான பகுப்பாய்வு தொடரியல் பகுப்பாய்வு (syntactic analysis) ஆகும்.
தொடரியல் பகுப்பாய்வு
பேட்டர்ன் மேட்சிங் (pattern matching) போன்ற அணுகுமுறையைப் பயன்படுத்தி முரண்பாடுகள் மற்றும் குறைபாடுகளைக் கண்டறிய Slither குறியீட்டின் வெவ்வேறு கூறுகள் மற்றும் அவற்றின் பிரதிநிதித்துவத்தின் வழியாகச் செல்ல முடியும்.
எடுத்துக்காட்டாக, பின்வரும் டிடெக்டர்கள் தொடரியல் தொடர்பான சிக்கல்களைத் தேடுகின்றன:
-
State variable shadowing (opens in a new tab): அனைத்து நிலை மாறிகள் (state variables) மீதும் மீண்டும் செயல்பட்டு, மரபுரிமையாகப் பெற்ற ஒப்பந்தத்திலிருந்து ஏதேனும் ஒரு மாறியை மறைக்கிறதா (shadow) எனச் சரிபார்க்கிறது (state.py#L51-L62 (opens in a new tab))
-
Incorrect ERC20 interface (opens in a new tab): தவறான ERC20 செயல்பாட்டு கையொப்பங்களைத் தேடுகிறது (incorrect_erc20_interface.py#L34-L55 (opens in a new tab))
சொற்பொருள் பகுப்பாய்வு
தொடரியல் பகுப்பாய்விற்கு மாறாக, ஒரு சொற்பொருள் பகுப்பாய்வு (semantic analysis) ஆழமாகச் சென்று குறியீட்டின் "பொருளை" பகுப்பாய்வு செய்யும். இந்தக் குடும்பத்தில் சில பரந்த வகையான பகுப்பாய்வுகள் அடங்கும். அவை மிகவும் சக்திவாய்ந்த மற்றும் பயனுள்ள முடிவுகளுக்கு வழிவகுக்கின்றன, ஆனால் எழுதுவதற்கும் மிகவும் சிக்கலானவை.
மிகவும் மேம்பட்ட பாதிப்புகளைக் கண்டறிவதற்கு சொற்பொருள் பகுப்பாய்வுகள் பயன்படுத்தப்படுகின்றன.
தரவு சார்பு பகுப்பாய்வு
variable_a இன் மதிப்பு variable_b ஆல் பாதிக்கப்படும் ஒரு பாதை இருந்தால், variable_a என்ற மாறி variable_b இன் தரவு சார்புடையது என்று கூறப்படுகிறது.
பின்வரும் குறியீட்டில், variable_a ஆனது variable_b-ஐச் சார்ந்துள்ளது:
1// ...2variable_a = variable_b + 1;Slither அதன் இடைநிலை பிரதிநிதித்துவத்தின் (பிற்பட்ட பிரிவில் விவாதிக்கப்பட்டுள்ளது) காரணமாக, உள்ளமைக்கப்பட்ட தரவு சார்பு (opens in a new tab) திறன்களுடன் வருகிறது.
தரவு சார்பு பயன்பாட்டிற்கான ஒரு எடுத்துக்காட்டை ஆபத்தான கண்டிப்பான சமத்துவ டிடெக்டரில் (dangerous strict equality detector) (opens in a new tab) காணலாம். இங்கே Slither ஒரு ஆபத்தான மதிப்பிற்கான கண்டிப்பான சமத்துவ ஒப்பீட்டைத் தேடும் (incorrect_strict_equality.py#L86-L87 (opens in a new tab)), மேலும் தாக்குபவர் ஒப்பந்தத்தைப் பொறிக்குள் சிக்க வைப்பதைத் தடுக்க, == என்பதற்குப் பதிலாக >= அல்லது <=-ஐப் பயன்படுத்த வேண்டும் என்று பயனருக்குத் தெரிவிக்கும். மற்றவற்றுடன், balanceOf(address)-க்கான அழைப்பின் திரும்பப் பெறும் மதிப்பை டிடெக்டர் ஆபத்தானதாகக் கருதும் (incorrect_strict_equality.py#L63-L64 (opens in a new tab)), மேலும் அதன் பயன்பாட்டைக் கண்காணிக்க தரவு சார்பு இயந்திரத்தைப் பயன்படுத்தும்.
நிலையான-புள்ளி கணக்கீடு
உங்கள் பகுப்பாய்வு CFG வழியாகச் சென்று விளிம்புகளைப் பின்பற்றினால், ஏற்கனவே பார்வையிட்ட முனைகளை நீங்கள் காண வாய்ப்புள்ளது. எடுத்துக்காட்டாக, கீழே காட்டப்பட்டுள்ளபடி ஒரு லூப் (loop) வழங்கப்பட்டால்:
1for(uint i; i < range; ++){2 variable_a += 13}எப்போது நிறுத்த வேண்டும் என்பதை உங்கள் பகுப்பாய்வு தெரிந்து கொள்ள வேண்டும். இங்கே இரண்டு முக்கிய உத்திகள் உள்ளன: (1) ஒவ்வொரு முனையிலும் ஒரு குறிப்பிட்ட எண்ணிக்கையிலான முறை மீண்டும் செயல்படுவது, (2) ஃபிக்ஸ்பாயிண்ட் (fixpoint) என்று அழைக்கப்படுவதைக் கணக்கிடுவது. ஒரு ஃபிக்ஸ்பாயிண்ட் என்பது அடிப்படையில் இந்த முனையை பகுப்பாய்வு செய்வது எந்த அர்த்தமுள்ள தகவலையும் வழங்காது என்பதாகும்.
பயன்படுத்தப்படும் ஃபிக்ஸ்பாயிண்டின் உதாரணத்தை ரீஎன்ட்ரன்சி டிடெக்டர்களில் (reentrancy detectors) காணலாம்: Slither முனைகளை ஆராய்கிறது, மேலும் வெளிப்புற அழைப்புகள், சேமிப்பகத்தில் எழுதுதல் மற்றும் படித்தல் ஆகியவற்றைத் தேடுகிறது. அது ஒரு ஃபிக்ஸ்பாயிண்டை அடைந்தவுடன் (reentrancy.py#L125-L131 (opens in a new tab)), அது ஆய்வை நிறுத்துகிறது, மேலும் வெவ்வேறு ரீஎன்ட்ரன்சி பேட்டர்ன்கள் மூலம் ரீஎன்ட்ரன்சி உள்ளதா என்பதைப் பார்க்க முடிவுகளை பகுப்பாய்வு செய்கிறது (reentrancy_benign.py (opens in a new tab), reentrancy_read_before_write.py (opens in a new tab), reentrancy_eth.py (opens in a new tab)).
திறமையான நிலையான புள்ளி கணக்கீட்டைப் பயன்படுத்தி பகுப்பாய்வுகளை எழுதுவதற்கு, பகுப்பாய்வு அதன் தகவலை எவ்வாறு பரப்புகிறது என்பதைப் பற்றிய நல்ல புரிதல் தேவை.
இடைநிலை பிரதிநிதித்துவம்
ஒரு இடைநிலை பிரதிநிதித்துவம் (IR) என்பது அசலை விட நிலையான பகுப்பாய்விற்கு மிகவும் ஏற்றதாக இருக்கும் ஒரு மொழியாகும். Slither Solidity-ஐ அதன் சொந்த IR-க்கு மொழிபெயர்க்கிறது: SlithIR (opens in a new tab).
நீங்கள் அடிப்படைச் சரிபார்ப்புகளை மட்டுமே எழுத விரும்பினால் SlithIR-ஐப் புரிந்து கொள்ள வேண்டிய அவசியமில்லை. இருப்பினும், மேம்பட்ட சொற்பொருள் பகுப்பாய்வுகளை எழுத நீங்கள் திட்டமிட்டால் இது கைக்கு வரும். SlithIR (opens in a new tab) மற்றும் SSA (opens in a new tab) பிரிண்டர்கள் குறியீடு எவ்வாறு மொழிபெயர்க்கப்படுகிறது என்பதைப் புரிந்துகொள்ள உங்களுக்கு உதவும்.
API அடிப்படைகள்
ஒப்பந்தம் மற்றும் அதன் செயல்பாடுகளின் அடிப்படை பண்புகளை ஆராய உங்களை அனுமதிக்கும் API-ஐ Slither கொண்டுள்ளது.
குறியீட்டுத் தளத்தை (codebase) ஏற்ற:
1from slither import Slither2slither = Slither('/path/to/project')3
ஒப்பந்தங்கள் மற்றும் செயல்பாடுகளை ஆராய்தல்
ஒரு Slither ஆப்ஜெக்ட் (object) இவற்றைக் கொண்டுள்ளது:
contracts (list(Contract): ஒப்பந்தங்களின் பட்டியல்contracts_derived (list(Contract): மற்றொரு ஒப்பந்தத்தால் மரபுரிமையாகப் பெறப்படாத ஒப்பந்தங்களின் பட்டியல் (ஒப்பந்தங்களின் துணைக்குழு)get_contract_from_name (str): அதன் பெயரிலிருந்து ஒரு ஒப்பந்தத்தைத் திருப்பித் தரும்
ஒரு Contract ஆப்ஜெக்ட் இவற்றைக் கொண்டுள்ளது:
name (str): ஒப்பந்தத்தின் பெயர்functions (list(Function)): செயல்பாடுகளின் பட்டியல்modifiers (list(Modifier)): செயல்பாடுகளின் பட்டியல்all_functions_called (list(Function/Modifier)): ஒப்பந்தத்தால் அடையக்கூடிய அனைத்து உள் செயல்பாடுகளின் பட்டியல்inheritance (list(Contract)): மரபுரிமையாகப் பெற்ற ஒப்பந்தங்களின் பட்டியல்get_function_from_signature (str): அதன் கையொப்பத்திலிருந்து ஒரு செயல்பாட்டைத் திருப்பித் தரும்get_modifier_from_signature (str): அதன் கையொப்பத்திலிருந்து ஒரு மாற்றுவியைத் (Modifier) திருப்பித் தரும்get_state_variable_from_name (str): அதன் பெயரிலிருந்து ஒரு StateVariable-ஐத் திருப்பித் தரும்
ஒரு Function அல்லது Modifier ஆப்ஜெக்ட் இவற்றைக் கொண்டுள்ளது:
name (str): செயல்பாட்டின் பெயர்contract (contract): செயல்பாடு அறிவிக்கப்பட்ட ஒப்பந்தம்nodes (list(Node)): செயல்பாடு/மாற்றுவியின் CFG-ஐ உருவாக்கும் முனைகளின் பட்டியல்entry_point (Node): CFG-இன் நுழைவுப் புள்ளிvariables_read (list(Variable)): படிக்கப்பட்ட மாறிகளின் பட்டியல்variables_written (list(Variable)): எழுதப்பட்ட மாறிகளின் பட்டியல்state_variables_read (list(StateVariable)): படிக்கப்பட்ட நிலை மாறிகளின் பட்டியல் (variables`read இன் துணைக்குழு)state_variables_written (list(StateVariable)): எழுதப்பட்ட நிலை மாறிகளின் பட்டியல் (variables`written இன் துணைக்குழு)
பக்கம் கடைசியாகப் புதுப்பிக்கப்பட்டது: 3 மார்ச், 2026

