Skip to content

Commit 5b06b8e

Browse files
feat: moved assets to their respective subdirectories and add --vyper flag
1 parent b7bf60a commit 5b06b8e

15 files changed

+384
-29
lines changed
File renamed without changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// SPDX-License-Identifier: UNLICENSED
2+
pragma solidity ^0.8.13;
3+
4+
import {Script} from "forge-std/Script.sol";
5+
import {VyperDeployer} from "../src/utils/VyperDeployer.sol";
6+
7+
contract CounterScript is Script {
8+
uint256 public constant INITIAL_COUNTER = 42;
9+
10+
function run() external {
11+
uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY");
12+
13+
VyperDeployer vyperDeployer = new VyperDeployer();
14+
15+
vm.startBroadcast(deployerPrivateKey);
16+
17+
address deployedAddress = vyperDeployer.deployContract("Counter", abi.encode(INITIAL_COUNTER));
18+
19+
require(deployedAddress != address(0), "Could not deploy contract");
20+
21+
vm.stopBroadcast();
22+
}
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
// SPDX-License-Identifier: UNLICENSED
2+
pragma solidity ^0.8.13;
3+
4+
import {Test} from "forge-std/Test.sol";
5+
import {ICounter} from "../src/interface/ICounter.sol";
6+
import {VyperDeployer} from "../src/utils/VyperDeployer.sol";
7+
8+
contract CounterTest is Test {
9+
VyperDeployer public vyperDeployer;
10+
ICounter public counterContract;
11+
uint256 public constant INITIAL_COUNTER = 42;
12+
13+
function setUp() public {
14+
vyperDeployer = new VyperDeployer();
15+
counterContract = ICounter(vyperDeployer.deployContract("Counter", abi.encode(INITIAL_COUNTER)));
16+
}
17+
18+
function test_getCounter() public view {
19+
uint256 counter = counterContract.counter();
20+
assertEq(counter, INITIAL_COUNTER);
21+
}
22+
23+
function test_setCounter() public {
24+
uint256 newCounter = 100;
25+
counterContract.set_counter(newCounter);
26+
uint256 counter = counterContract.counter();
27+
assertEq(counter, newCounter);
28+
}
29+
30+
function test_increment() public {
31+
uint256 counterBefore = counterContract.counter();
32+
counterContract.increment();
33+
uint256 counterAfter = counterContract.counter();
34+
assertEq(counterAfter, counterBefore + 1);
35+
}
36+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
counter: public(uint256)
2+
3+
@deploy
4+
@payable
5+
def __init__(initial_counter: uint256):
6+
self.counter = initial_counter
7+
8+
@external
9+
def set_counter(new_counter: uint256):
10+
self.counter = new_counter
11+
12+
@external
13+
def increment():
14+
self.counter += 1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// SPDX-License-Identifier: UNLICENSED
2+
pragma solidity ^0.8.13;
3+
4+
interface ICounter {
5+
function counter() external view returns (uint256);
6+
function set_counter(uint256 new_counter) external;
7+
function increment() external;
8+
}

crates/forge/assets/vyper/README.md

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
## Foundry
2+
3+
**Foundry is a blazing fast, portable and modular toolkit for Ethereum application development written in Rust.**
4+
5+
Foundry consists of:
6+
7+
- **Forge**: Ethereum testing framework (like Truffle, Hardhat and DappTools).
8+
- **Cast**: Swiss army knife for interacting with EVM smart contracts, sending transactions and getting chain data.
9+
- **Anvil**: Local Ethereum node, akin to Ganache, Hardhat Network.
10+
- **Chisel**: Fast, utilitarian, and verbose solidity REPL.
11+
12+
## Documentation
13+
14+
https://book.getfoundry.sh/
15+
16+
## Getting Started
17+
18+
### Prerequisites
19+
20+
- [Vyper](https://vyper.readthedocs.io/)
21+
22+
23+
## Usage
24+
25+
### Build
26+
27+
```shell
28+
$ forge build
29+
```
30+
31+
### Test
32+
33+
```shell
34+
$ forge test
35+
```
36+
37+
### Format
38+
39+
```shell
40+
$ forge fmt
41+
```
42+
43+
### Anvil
44+
45+
```shell
46+
$ anvil
47+
```
48+
49+
### Deploy
50+
51+
```shell
52+
$ forge script script/Counter.s.sol:CounterScript --rpc-url <your_rpc_url> --private-key <your_private_key>
53+
```
54+
55+
### Cast
56+
57+
```shell
58+
$ cast <subcommand>
59+
```
60+
61+
### Help
62+
63+
```shell
64+
$ forge --help
65+
$ anvil --help
66+
$ cast --help
67+
```
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
// SPDX-License-Identifier: UNLICENSED
2+
pragma solidity ^0.8.13;
3+
4+
import {Vm} from "forge-std/Vm.sol";
5+
6+
contract VyperDeployer {
7+
address private constant HEVM_ADDRESS = address(uint160(uint256(keccak256("hevm cheat code"))));
8+
Vm private constant vm = Vm(HEVM_ADDRESS);
9+
// Base directory for Vyper contracts
10+
string private constant BASE_PATH = "src/";
11+
12+
/**
13+
* Compiles a Vyper contract and returns the `CREATE` address.
14+
* @param fileName The file name of the Vyper contract.
15+
* @return deployedAddress The address calculated through create operation.
16+
*/
17+
function deployContract(string memory fileName) public returns (address) {
18+
return deployContract(BASE_PATH, fileName, "");
19+
}
20+
21+
/**
22+
* Compiles a Vyper contract and returns the `CREATE` address.
23+
* @param fileName The file name of the Vyper contract.
24+
* @param args The constructor arguments for the contract
25+
* @return deployedAddress The address calculated through create operation.
26+
*/
27+
function deployContract(string memory fileName, bytes memory args) public returns (address) {
28+
return deployContract(BASE_PATH, fileName, args);
29+
}
30+
31+
/**
32+
* Compiles a Vyper contract with constructor arguments and returns the `CREATE` address.
33+
* @param basePath The base directory path where the Vyper contract is located
34+
* @param fileName The file name of the Vyper contract.
35+
* @param args The constructor arguments for the contract
36+
* @return deployedAddress The address calculated through create operation.
37+
*/
38+
function deployContract(string memory basePath, string memory fileName, bytes memory args)
39+
public
40+
returns (address)
41+
{
42+
// Compile the Vyper contract
43+
bytes memory bytecode = compileVyperContract(basePath, fileName);
44+
45+
// Add constructor arguments if provided
46+
if (args.length > 0) {
47+
bytecode = abi.encodePacked(bytecode, args);
48+
}
49+
50+
// Deploy the contract
51+
address deployedAddress = deployBytecode(bytecode);
52+
53+
// Return the deployed address
54+
return deployedAddress;
55+
}
56+
57+
/**
58+
* Compiles a Vyper contract and returns the bytecode
59+
* @param basePath The base directory path where the Vyper contract is located
60+
* @param fileName The file name of the Vyper contract
61+
* @return The compiled bytecode of the contract
62+
*/
63+
function compileVyperContract(string memory basePath, string memory fileName) internal returns (bytes memory) {
64+
// create a list of strings with the commands necessary to compile Vyper contracts
65+
string[] memory cmds = new string[](2);
66+
cmds[0] = "vyper";
67+
cmds[1] = string.concat(basePath, fileName, ".vy");
68+
69+
// compile the Vyper contract and return the bytecode
70+
return vm.ffi(cmds);
71+
}
72+
73+
/**
74+
* Deploys bytecode using the create instruction
75+
* @param bytecode - The bytecode to deploy
76+
* @return deployedAddress The address calculated through create operation.
77+
*/
78+
function deployBytecode(bytes memory bytecode) internal returns (address deployedAddress) {
79+
// deploy the bytecode with the create instruction
80+
assembly {
81+
deployedAddress := create(0, add(bytecode, 0x20), mload(bytecode))
82+
}
83+
84+
// check that the deployment was successful
85+
require(deployedAddress != address(0), "VyperDeployer could not deploy contract");
86+
}
87+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
name: test
2+
3+
on: [push, pull_request, workflow_dispatch]
4+
5+
env:
6+
FOUNDRY_PROFILE: ci
7+
8+
jobs:
9+
check:
10+
runs-on: ${{ matrix.os }}
11+
strategy:
12+
matrix:
13+
os:
14+
- ubuntu-latest
15+
python_version:
16+
- 3.12
17+
architecture:
18+
- x64
19+
node_version:
20+
- 20
21+
22+
steps:
23+
- name: Checkout
24+
uses: actions/checkout@v4
25+
with:
26+
submodules: recursive
27+
28+
- name: Setup Python
29+
uses: actions/setup-python@v5
30+
with:
31+
python-version: ${{ matrix.python_version }}
32+
architecture: ${{ matrix.architecture }}
33+
34+
- name: Install latest Vyper
35+
run: pip install --force-reinstall vyper
36+
37+
- name: Show the Vyper version
38+
run: vyper --version
39+
40+
- name: Install pnpm
41+
uses: pnpm/action-setup@v3
42+
with:
43+
version: latest
44+
run_install: false
45+
46+
- name: Get pnpm cache directory path
47+
id: pnpm-cache-dir-path
48+
run: echo "dir=$(pnpm store path --silent)" >> $GITHUB_OUTPUT
49+
50+
- name: Restore pnpm cache
51+
uses: actions/cache@v4
52+
id: pnpm-cache
53+
with:
54+
path: ${{ steps.pnpm-cache-dir-path.outputs.dir }}
55+
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/pnpm-lock.yaml') }}
56+
restore-keys: ${{ runner.os }}-pnpm-store-
57+
58+
- name: Use Node.js ${{ matrix.node_version }}
59+
uses: actions/setup-node@v4
60+
with:
61+
node-version: ${{ matrix.node_version }}
62+
63+
- name: Install pnpm project with a clean slate
64+
run: pnpm install --prefer-offline --frozen-lockfile
65+
66+
- name: Install Foundry
67+
uses: foundry-rs/foundry-toolchain@v1
68+
with:
69+
version: nightly
70+
71+
- name: Show the Foundry CI config
72+
run: forge config
73+
env:
74+
FOUNDRY_PROFILE: ci
75+
76+
- name: Foundry tests
77+
run: forge test
78+
env:
79+
FOUNDRY_PROFILE: ci

0 commit comments

Comments
 (0)