Skip to content

Commit cf03bb6

Browse files
raxhvlRahul RavindranEvalir
authored
feat(forge) - Test scaffolding (#5495)
* feat: #5466 - Test scaffolding * reafactor: removed return * chore: fmt * refactor: named imports * refactor: std::fs -> foundry_common::fs --------- Co-authored-by: Rahul Ravindran <[email protected]> Co-authored-by: Enrique Ortiz <[email protected]>
1 parent 8d342d3 commit cf03bb6

File tree

5 files changed

+93
-2
lines changed

5 files changed

+93
-2
lines changed
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
// SPDX-License-Identifier: UNLICENSED
2+
pragma solidity ^0.8.13;
3+
4+
import {Test, console2} from "forge-std/Test.sol";
5+
import {{contract_name}} from "../src/{contract_name}.sol";
6+
7+
contract {contract_name}Test is Test {
8+
{contract_name} public {instance_name};
9+
10+
function setUp() public {
11+
{instance_name} = new {contract_name}();
12+
}
13+
}

cli/src/cmd/forge/generate/mod.rs

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
//! generate command
2+
3+
use clap::{Parser, Subcommand};
4+
use foundry_common::fs;
5+
use std::path::Path;
6+
use yansi::Paint;
7+
8+
/// CLI arguments for `forge generate`.
9+
#[derive(Debug, Parser)]
10+
pub struct GenerateArgs {
11+
#[clap(subcommand)]
12+
pub sub: GenerateSubcommands,
13+
}
14+
15+
#[derive(Debug, Subcommand)]
16+
pub enum GenerateSubcommands {
17+
/// Scaffolds test file for given contract.
18+
Test(GenerateTestArgs),
19+
}
20+
21+
#[derive(Debug, Parser)]
22+
pub struct GenerateTestArgs {
23+
/// Contract name for test generation.
24+
#[clap(long, short, value_name = "CONTRACT_NAME")]
25+
pub contract_name: String,
26+
}
27+
28+
impl GenerateTestArgs {
29+
pub fn run(self) -> eyre::Result<()> {
30+
let contract_name = format_identifier(&self.contract_name, true);
31+
let instance_name = format_identifier(&self.contract_name, false);
32+
33+
// Create the test file content.
34+
let test_content = include_str!("../../../../assets/generated/TestTemplate.t.sol");
35+
let test_content = test_content
36+
.replace("{contract_name}", &contract_name)
37+
.replace("{instance_name}", &instance_name);
38+
39+
// Create the test directory if it doesn't exist.
40+
fs::create_dir_all("test")?;
41+
42+
// Define the test file path
43+
let test_file_path = Path::new("test").join(format!("{}.t.sol", contract_name));
44+
45+
// Write the test content to the test file.
46+
fs::write(&test_file_path, test_content)?;
47+
48+
println!("{} test file: {}", Paint::green("Generated"), test_file_path.to_str().unwrap());
49+
Ok(())
50+
}
51+
}
52+
53+
/// Utility function to convert an identifier to pascal or camel case.
54+
fn format_identifier(input: &str, is_pascal_case: bool) -> String {
55+
let mut result = String::new();
56+
let mut capitalize_next = is_pascal_case;
57+
58+
for word in input.split_whitespace() {
59+
if !word.is_empty() {
60+
let (first, rest) = word.split_at(1);
61+
let formatted_word = if capitalize_next {
62+
format!("{}{}", first.to_uppercase(), rest)
63+
} else {
64+
format!("{}{}", first.to_lowercase(), rest)
65+
};
66+
capitalize_next = true;
67+
result.push_str(&formatted_word);
68+
}
69+
}
70+
result
71+
}

cli/src/cmd/forge/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -51,6 +51,7 @@ pub mod flatten;
5151
pub mod fmt;
5252
pub mod fourbyte;
5353
pub mod geiger;
54+
pub mod generate;
5455
pub mod init;
5556
pub mod inspect;
5657
pub mod install;

cli/src/forge.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ use clap::{CommandFactory, Parser};
22
use clap_complete::generate;
33
use foundry_cli::{
44
cmd::{
5-
forge::{cache::CacheSubcommands, watch},
5+
forge::{cache::CacheSubcommands, generate::GenerateSubcommands, watch},
66
Cmd,
77
},
88
handler,
@@ -97,5 +97,8 @@ fn main() -> eyre::Result<()> {
9797
}
9898
Subcommands::Doc(cmd) => cmd.run(),
9999
Subcommands::Selectors { command } => utils::block_on(command.run()),
100+
Subcommands::Generate(cmd) => match cmd.sub {
101+
GenerateSubcommands::Test(cmd) => cmd.run(),
102+
},
100103
}
101104
}

cli/src/opts/forge.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use crate::cmd::forge::{
99
flatten,
1010
fmt::FmtArgs,
1111
fourbyte::UploadSelectorsArgs,
12-
geiger,
12+
geiger, generate,
1313
init::InitArgs,
1414
inspect,
1515
install::InstallArgs,
@@ -163,6 +163,9 @@ pub enum Subcommands {
163163
#[clap(subcommand)]
164164
command: SelectorsSubcommands,
165165
},
166+
167+
/// Generate scaffold files.
168+
Generate(generate::GenerateArgs),
166169
}
167170

168171
// A set of solc compiler settings that can be set via command line arguments, which are intended

0 commit comments

Comments
 (0)