Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/tintinweb/solgrep
🧠 A scriptable semantic grep utility for solidity
https://github.com/tintinweb/solgrep
grep search solidity
Last synced: 1 day ago
JSON representation
🧠 A scriptable semantic grep utility for solidity
- Host: GitHub
- URL: https://github.com/tintinweb/solgrep
- Owner: tintinweb
- Created: 2021-11-19T17:51:25.000Z (about 3 years ago)
- Default Branch: master
- Last Pushed: 2024-08-08T10:53:54.000Z (5 months ago)
- Last Synced: 2025-01-03T23:11:57.722Z (9 days ago)
- Topics: grep, search, solidity
- Language: JavaScript
- Homepage: https://www.npmjs.com/package/solgrep
- Size: 2 MB
- Stars: 144
- Watchers: 8
- Forks: 18
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
Awesome Lists containing this project
- awesome-solidity - tintinweb/solgrep - A scriptable semantic grep utility for Solidity. (Tools)
- awesome-solidity - tintinweb/solgrep - A scriptable semantic grep utility for Solidity. (Tools)
- awesome-solidity - tintinweb/solgrep - A scriptable semantic grep utility for Solidity. (Tools)
README
[](https://diligence.consensys.net)
[[ 🌐 ](https://diligence.consensys.net) [ 📩 ](https://github.com/ConsenSys/vscode-solidity-doppelganger/blob/master/mailto:[email protected]) [ 🔥 ](https://consensys.github.io/diligence/)]## 🧠 SolGrep - A scriptable semantic grep utility for solidity
So you have a set of smart contracts and want to find all contracts that have a `public` method named `withdrawEth` but lexical grep yields a lot of false-positives? Here's where [solgrep](https://github.com/tintinweb/solgrep) can help! 🙌
[💾](https://www.npmjs.com/package/solgrep) `npm install -g solgrep`
Solgrep recursively finds smart contracts in a target directory, parses the source units to understand the language semantics, and makes this information available to a powerful javascript-based filter function. This way, you can:
* extract semantic information from solidity source code based on custom filter functions
* find target contracts based on a custom filter script you define
* create & run your own or built-in rules (e.g. for CI checks)
* crunch numbers and generate statistics from a code base
* find doppelgangers! i.e. duplicate contracts sharing the same code structure (`AST_EXACT` and `AST_FUZZY` matching)Probably the most common way to use this tool is to run it with the `--find=` option, where `js-filter-statement` is a javascript one-liner that tells the engine what you are interested in. You either provide a statement that returns `boolean` ("find mode") or return information you want to extract ("extract mode").
![solgrep2](https://user-images.githubusercontent.com/2865694/142903238-af9d2d11-81bb-430c-96dc-4118da89c90b.gif)
⚠️ Make sure to only allow trusted inputs to `--find=` as this argument is being evaluated as javascript!
⚠️ Check out the proper order for arguments when using `--find`, `--rule`. See [Usage](#usage).
### Examples
Run the default rules and display some stats?
```javascript
⇒ solgrep
```Run a specific or multiple built-in rules (`solgrep -l` to list available rules)? 👉
```javascript
⇒ solgrep --rule=IsInitializable --rule=Stats
```You want to find all source-units with a contract that has a function named `withdrawEth`? 👉
```javascript
⇒ solgrep --find="function.name=='withdrawEth'"
```Do the same thing but case-insensitive? 👉
```javascript
⇒ solgrep --find="function.name.toLowerCase()=='withdraweth'"
```Find all functions that call `selfdestruct`? 👉
```javascript
⇒ solgrep --find="function.callsTo('selfdestruct')"
```Exctract all function names from all contracts? 👉
```javascript
⇒ solgrep --find="function.name"
```Get a list of all `external` functions? 👉
```javascript
⇒ solgrep --find="function.visibility.includes('external')"
```Find `ERC777` contracts? 👉
```javascript
⇒ solgrep --find="contract.name=='ERC777'"
```Extract all Contract names? 👉
```javascript
⇒ solgrep --find="contract.name"
```Extract all Interface names? 👉
```javascript
⇒ solgrep --find="contract.name && contract.kind=='interface'"
```Match against something in the `AST`? 👉
```javascript
⇒ solgrep --find="Object.keys(function.modifiers).includes('nonReentrant')"
```Lexial match a functions source code? 👉
```javascript
⇒ solgrep --find="function.getSource().includes('hi')"
```Find duplicate contracts? (AST exact and fuzzy matching) 👉
```javascript
⇒ solgrep --rule DupeFinder
```Use option `--output=` to write all results to a file.
### Built-In Keywords for `--find`
* The `sourceUnit` object matches the source-file the engine is currently processing.
* The `contract` object matches the contract the engine is currently processing.
* The `function` object matches the function the engine is currently processing.#### Available Methos
The following methods are available:
* `.getSource()` - provides access to the units source code
* `.ast` - provides direct access to the [solidity-parser](https://github.com/solidity-parser/parser) AST
* `function.callsTo('withdrawEth')` - find all functions in all contracts that call a method named `withdrawEth`
* There's even more information easily available (functions/events/pragmas/etc.). Go check out the Solidity SourceUnit/Contract/Function [Wrapper Classes](https://github.com/tintinweb/solgrep/blob/master/src/solidity.js#L18) functionality (attribs/functions).#### Special Functions
Special contract functions can be references as:
* `function.name == '__fallback__'`
* `function.name == '__receiveEther__'`
* `function.name == '__constructor__'`### Built-In Rules
* `Stats` - collects stats about how many files, contracts, unique names, etc. were processed
* `GenericGrep` - is the engine behind the `--find` feature
* ...**SHARING IS CARING** - [submit your rules!](https://github.com/tintinweb/solgrep/pulls)
## Usage
```javascript
⇒ solgrep --help
Usage: solgrep [options]Options:
-r, --rule Enable rules [array] [default: []]
-l, --list-rules List available rules [boolean] [default: false]
-f, --find Find/Extract information using custom pattern
[array] [default: []]
-o, --output Write "results" as JSON to output file path. [string]
-h, --help Show help [boolean]
-v, --version Show version number [boolean]```
⚠️ when using multi-options (`--find`, `--rule`) make sure to use this format:
```
⇒ solgrep [options]
```or
```
⇒ solgrep [options] --
```or else additional options might be interpreted as additional `--find` options!
## Library
```javascript
const solgrep = require('solgrep');let sg = new SolGrep('::memory::', rules, callbacks);
sg.analyzeDir("/path/to/smart/contracts").then(() => {
console.log(" ──────────────────────────── Results")
console.log(sg.results)
console.log(" ────────────────────────────")
sgrep.close();
})```
## Demo
Here's an example that illustrates how to extract all function names from all `sourceUnits` and `contracts`.
```javascript
⇒ solgrep ../d0 --find='function.name'
🧠 SolGrep v0.0.3 ready!Enabled Modules:
✔️ GenericGrep _function.name[████████████████████████████████████████] 100% | 🕙 ETA: 0s | 5/5 | 🌲 389 | 🔥 0 | 🗂️ ../d0
────────────────────────────
{
'../d0/D0788Af7a613b81F437a51b96594A6387c7329b1_PendleLiquidityMiningBaseV2Multi.sol': [
{
rule: 'GenericGrep',
tag: 'match-function: PendleLiquidityMiningBaseV2Multi.__constructor__',
info: '__constructor__',
loc: [Object]
},
{
rule: 'GenericGrep',
tag: 'match-function: PendleLiquidityMiningBaseV2Multi.setUpEmergencyMode',
info: 'setUpEmergencyMode',
loc: [Object]
},
...
```#### Run default Rulesets
If you provide no configuration options it will take the default built-in grep rules and crunch numbers using the `stats` module.
```javascript
⇒ solgrep ../d6
🧠 SolGrep v0.0.1 starting ...📁 ../d6
████████████████████████████████████████ 100% | ETA: 0s | 5/5──────────────────────────── Results
{
'../d6/d60a598998ed27a7C54315F2675908B628E434B1_LiquidityPool.sol': [
{
rule: 'IsInitializable',
tag: 'INITIALIZEABLE',
info: 'initialize - public initialize function; likely proxy'
},
{
rule: 'IsInitializable',
tag: 'INITIALIZEABLE',
info: 'initialize - public initialize function; likely proxy'
}
],
'../d6/d64E77C7C6A1dcC7e302F8fe31A22745e223c39c_MyStrategy.sol': [
{
rule: 'IsInitializable',
tag: 'INITIALIZEABLE',
info: 'initialize - public initialize function; likely proxy'
}
],
undefined: [ { rule: 'Stats', tag: 'STATS', info: [Object] } ]
}
────────────────────────────
{
sourceUnits: 4,
contracts: {
total: 13,
names: {
ERC20: 1,
StrategySushiEthMimLp: 1,
LiquidityPool: 1,
Getter: 1,
Governance: 1,
LibraryEvents: 1,
Perpetual: 1,
Storage: 1,
ArbiBoneman: 1,
ERC721: 1,
MyStrategy: 1,
PausableUpgradeable: 1,
SettAccessControl: 1
}
},
interfaces: {
total: 39,
names: {
IERC20: 1,
IJar: 1,
IStakingRewards: 1,
IStakingRewardsFactory: 1,
IMasterchef: 1,
UniswapRouterV2: 1,
IUniswapV2Pair: 1,
IUniswapV2Factory: 1,
IRewarder: 1,
IMiniChefV2: 1,
ILiquidityPool: 1,
IOracle: 1,
IAccessControl: 1,
IGovernor: 1,
IPoolCreatorFull: 1,
ISymbolService: 1,
IPoolCreator: 1,
ITracer: 1,
IVersionControl: 1,
IVariables: 1,
IKeeperWhitelist: 1,
IProxyAdmin: 1,
IDecimals: 1,
ILiquidityPoolGetter: 1,
ILiquidityPoolGovernance: 1,
IPerpetual: 1,
IERC721Enumerable: 1,
IERC721: 1,
IERC721Receiver: 1,
IERC721Metadata: 1,
IERC165: 1,
ICurveGauge: 1,
ICurveStableSwapREN: 1,
IUniswapRouterV2: 1,
IStrategy: 1,
IController: 2,
IERC20Upgradeable: 2
}
},
libraries: {
total: 31,
names: {
SafeMath: 1,
SafeERC20: 1,
BoringERC20: 1,
EnumerableSetUpgradeable: 1,
SafeCastUpgradeable: 1,
AMMModule: 1,
LiquidityPoolModule: 1,
PerpetualModule: 1,
SignedSafeMathUpgradeable: 1,
Constant: 1,
Math: 1,
SafeMathExt: 1,
Utils: 1,
MarginAccountModule: 1,
EnumerableSet: 1,
OrderData: 1,
CollateralModule: 1,
TradeModule: 1,
OrderModule: 1,
Signature: 1,
ECDSAUpgradeable: 1,
Strings: 1,
MathUpgradeable: 1,
Address: 2,
SafeMathUpgradeable: 2,
AddressUpgradeable: 2,
SafeERC20Upgradeable: 2
}
},
abstract: {
total: 13,
names: {
StrategyBase: 1,
StrategySushiFarmBase: 1,
ReentrancyGuardUpgradeable: 1,
ERC721Enumerable: 1,
Ownable: 1,
ERC165: 1,
BaseStrategy: 1,
Context: 2,
ContextUpgradeable: 2,
Initializable: 2
}
}
}
TOTAL FILES: 5
ERRORS: 1
────────────────────────────cheers 🙌
@tintinweb
ConsenSys Diligence @ https://consensys.net/diligence/
https://github.com/tintinweb/solgrep/```