hardhat-deploy 2.0.0-next.7 → 2.0.0-next.70

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (124) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +120 -0
  3. package/dist/cli.d.ts +3 -0
  4. package/dist/cli.d.ts.map +1 -0
  5. package/dist/cli.js +197 -0
  6. package/dist/cli.js.map +1 -0
  7. package/dist/config/default.d.ts.map +1 -0
  8. package/dist/config/default.js +5 -0
  9. package/dist/config/default.js.map +1 -0
  10. package/dist/config/get-config.d.ts.map +1 -0
  11. package/dist/config/get-config.js +9 -0
  12. package/dist/config/get-config.js.map +1 -0
  13. package/dist/config/validation.d.ts.map +1 -0
  14. package/dist/config/validation.js +17 -0
  15. package/dist/config/validation.js.map +1 -0
  16. package/dist/generate-types.d.ts +5 -0
  17. package/dist/generate-types.d.ts.map +1 -0
  18. package/dist/generate-types.js +244 -0
  19. package/dist/generate-types.js.map +1 -0
  20. package/dist/helpers.d.ts +34 -0
  21. package/dist/helpers.d.ts.map +1 -0
  22. package/dist/{esm/helpers.js → helpers.js} +106 -73
  23. package/dist/helpers.js.map +1 -0
  24. package/dist/hook-handlers/config.d.ts.map +1 -0
  25. package/dist/hook-handlers/config.js +68 -0
  26. package/dist/hook-handlers/config.js.map +1 -0
  27. package/dist/hook-handlers/solidity.d.ts.map +1 -0
  28. package/dist/hook-handlers/solidity.js +21 -0
  29. package/dist/hook-handlers/solidity.js.map +1 -0
  30. package/dist/index.d.ts +8 -0
  31. package/dist/index.d.ts.map +1 -0
  32. package/dist/{esm/index.js → index.js} +20 -3
  33. package/dist/index.js.map +1 -0
  34. package/dist/postinstall.d.ts +3 -0
  35. package/dist/postinstall.d.ts.map +1 -0
  36. package/dist/postinstall.js +148 -0
  37. package/dist/postinstall.js.map +1 -0
  38. package/dist/{esm/tasks → tasks}/deploy.d.ts +1 -0
  39. package/dist/tasks/deploy.d.ts.map +1 -0
  40. package/dist/tasks/deploy.js +32 -0
  41. package/dist/tasks/deploy.js.map +1 -0
  42. package/dist/{esm/type-extensions.d.ts → type-extensions.d.ts} +2 -2
  43. package/dist/type-extensions.d.ts.map +1 -0
  44. package/dist/type-extensions.js.map +1 -0
  45. package/dist/types.d.ts +13 -0
  46. package/dist/types.d.ts.map +1 -0
  47. package/dist/types.js.map +1 -0
  48. package/dist/utils/files.d.ts +9 -0
  49. package/dist/utils/files.d.ts.map +1 -0
  50. package/dist/utils/files.js +25 -0
  51. package/dist/utils/files.js.map +1 -0
  52. package/dist/v1-detection.d.ts +12 -0
  53. package/dist/v1-detection.d.ts.map +1 -0
  54. package/dist/v1-detection.js +118 -0
  55. package/dist/v1-detection.js.map +1 -0
  56. package/dist/v1-entry.cjs +92 -0
  57. package/package.json +58 -26
  58. package/src/cli.ts +246 -0
  59. package/src/config/default.ts +6 -0
  60. package/src/config/get-config.ts +12 -0
  61. package/src/config/validation.ts +25 -0
  62. package/src/generate-types.ts +281 -0
  63. package/src/helpers.ts +311 -0
  64. package/src/hook-handlers/config.ts +80 -0
  65. package/src/hook-handlers/solidity.ts +33 -0
  66. package/src/index.ts +53 -0
  67. package/src/postinstall.ts +166 -0
  68. package/src/tasks/deploy.ts +43 -0
  69. package/src/type-extensions.ts +12 -0
  70. package/src/types.ts +9 -0
  71. package/src/utils/files.ts +37 -0
  72. package/src/v1-entry.cjs +92 -0
  73. package/templates/basic/README.md +44 -0
  74. package/templates/basic/contracts/Counter.sol +19 -0
  75. package/templates/basic/contracts/Counter.t.sol +29 -0
  76. package/templates/basic/deploy/01_deploy_counter.ts +13 -0
  77. package/templates/basic/hardhat.config.ts +50 -0
  78. package/templates/basic/package.json +31 -0
  79. package/templates/basic/pnpm-lock.yaml +1643 -0
  80. package/templates/basic/rocketh/config.ts +65 -0
  81. package/templates/basic/rocketh/deploy.ts +20 -0
  82. package/templates/basic/rocketh/environment.ts +22 -0
  83. package/templates/basic/test/Counter.ts +61 -0
  84. package/templates/basic/tsconfig.json +13 -0
  85. package/dist/esm/config/default.d.ts.map +0 -1
  86. package/dist/esm/config/default.js +0 -8
  87. package/dist/esm/config/default.js.map +0 -1
  88. package/dist/esm/config/get-config.d.ts.map +0 -1
  89. package/dist/esm/config/get-config.js +0 -8
  90. package/dist/esm/config/get-config.js.map +0 -1
  91. package/dist/esm/config/validation.d.ts.map +0 -1
  92. package/dist/esm/config/validation.js +0 -16
  93. package/dist/esm/config/validation.js.map +0 -1
  94. package/dist/esm/generate-types.d.ts +0 -6
  95. package/dist/esm/generate-types.d.ts.map +0 -1
  96. package/dist/esm/generate-types.js +0 -198
  97. package/dist/esm/generate-types.js.map +0 -1
  98. package/dist/esm/helpers.d.ts +0 -18
  99. package/dist/esm/helpers.d.ts.map +0 -1
  100. package/dist/esm/helpers.js.map +0 -1
  101. package/dist/esm/hook-handlers/config.d.ts.map +0 -1
  102. package/dist/esm/hook-handlers/config.js +0 -16
  103. package/dist/esm/hook-handlers/config.js.map +0 -1
  104. package/dist/esm/hook-handlers/solidity.d.ts.map +0 -1
  105. package/dist/esm/hook-handlers/solidity.js +0 -15
  106. package/dist/esm/hook-handlers/solidity.js.map +0 -1
  107. package/dist/esm/index.d.ts +0 -5
  108. package/dist/esm/index.d.ts.map +0 -1
  109. package/dist/esm/index.js.map +0 -1
  110. package/dist/esm/tasks/deploy.d.ts.map +0 -1
  111. package/dist/esm/tasks/deploy.js +0 -21
  112. package/dist/esm/tasks/deploy.js.map +0 -1
  113. package/dist/esm/type-extensions.d.ts.map +0 -1
  114. package/dist/esm/type-extensions.js.map +0 -1
  115. package/dist/esm/types.d.ts +0 -15
  116. package/dist/esm/types.d.ts.map +0 -1
  117. package/dist/esm/types.js.map +0 -1
  118. /package/dist/{esm/config → config}/default.d.ts +0 -0
  119. /package/dist/{esm/config → config}/get-config.d.ts +0 -0
  120. /package/dist/{esm/config → config}/validation.d.ts +0 -0
  121. /package/dist/{esm/hook-handlers → hook-handlers}/config.d.ts +0 -0
  122. /package/dist/{esm/hook-handlers → hook-handlers}/solidity.d.ts +0 -0
  123. /package/dist/{esm/type-extensions.js → type-extensions.js} +0 -0
  124. /package/dist/{esm/types.js → types.js} +0 -0
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2020 Ronan Sandford
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,120 @@
1
+ <div align="center">
2
+ <img alt="Rocketh Logo" src="./public/logo.svg" width="100"/><br/>
3
+ <a href="https://rocketh.dev/hardhat-deploy/">hardhat-deploy</a>
4
+ <hr/>
5
+
6
+ <!-- <a href="https://npmjs.com/package/hardhat-deploy"><img alt="Version" src="https://img.shields.io/npm/v/hardhat-deploy" /></a> -->
7
+
8
+ <a href="https://github.com/wighawag/hardhat-deploy/blob/main/LICENSE"><img alt="License" src="https://img.shields.io/github/license/wighawag/hardhat-deploy" /></a>
9
+ <a href="https://npmjs.com/package/hardhat-deploy"><img src="https://img.shields.io/npm/dw/hardhat-deploy" alt="weekly downloads" /></a>
10
+ <a href="https://npmjs.com/package/hardhat-deploy"><img alt="Dependents" src="https://img.shields.io/librariesio/dependents/npm/hardhat-deploy" /></a>
11
+
12
+ <!-- <a href="https://github.com/wighawag/hardhat-deploy/stargazers"><img alt="Github Stars" src="https://img.shields.io/github/stars/wighawag/hardhat-deploy" /></a> -->
13
+ <!-- <a href="https://npmjs.com/package/hardhat-deploy"><img alt="Node Version" src="https://img.shields.io/node/v/hardhat-deploy"></a> -->
14
+
15
+ <a href="https://github.com/wighawag/hardhat-deploy/issues"><img alt="Issues and PRs" src="https://img.shields.io/github/issues-pr/wighawag/hardhat-deploy" /></a>
16
+ <a href="https://github.com/wighawag/hardhat-deploy/actions"><img alt="Tests Status" src="https://img.shields.io/github/actions/workflow/status/wighawag/hardhat-deploy/test.yml?label=test" /></a>
17
+ <a href="https://github.com/wighawag/hardhat-deploy/commits/main/"><img alt="Commit activity" src="https://img.shields.io/github/commit-activity/w/wighawag/hardhat-deploy" /></a>
18
+ <a href="https://github.com/wighawag/hardhat-deploy/commits/main/"><img alt="Last commit" src="https://img.shields.io/github/last-commit/wighawag/hardhat-deploy" /></a>
19
+
20
+ <!-- <a href="https://npmjs.com/package/hardhat-deploy"><img alt="dependencies status" src="https://img.shields.io/librariesio/release/npm/hardhat-deploy" /></a>-->
21
+
22
+ </div>
23
+
24
+ <h1>hardhat-deploy</h1>
25
+
26
+ _A [Hardhat](https://hardhat.org) Plugin For Replicable Deployments And Easy Testing_
27
+
28
+ > **Note: This is the documentation for hardhat-deploy v2 that works with [hardhat v3](https://hardhat.org/hardhat3-alpha)**
29
+ >
30
+ > Documentation for hardhat-deploy v1 can be found on the [v1 branch](https://github.com/wighawag/hardhat-deploy/tree/v1#readme)
31
+
32
+ ## What is hardhat-deploy?
33
+
34
+ **hardhat-deploy** makes it easy for you to deploy your EVM smart contracts across many chains while keeping track of them and replicating the same environment for testing.
35
+
36
+ It adds a mechanism to associate names to addresses, so test and deployment scripts can be reconfigured by simply changing the address a name points to. This results in much clearer tests and deployment scripts — no more `accounts[0]` in your code.
37
+
38
+ ## Why hardhat-deploy + rocketh?
39
+
40
+ While Hardhat's official [Ignition](https://hardhat.org/ignition) plugin offers a robust deployment system, it comes with a rigid DSL that limits flexibility. **hardhat-deploy + rocketh** provides:
41
+
42
+ - **Intuitive Deployment Scripts**: Write deployment logic in plain TypeScript without learning a new DSL.
43
+ - **Browser-Compatible Deployments**: Thanks to [rocketh](https://github.com/wighawag/rocketh)'s framework-agnostic design, your deployment scripts can be executed directly in the browser — enabling in-app deployments, testing in web environments, and seamless integration with web frontends.
44
+ - **Hot Contract Replacement (HCR)**: The equivalent of HMR (Hot Module Replacement) for smart contracts. Edit your contracts and see changes live while developing your app or game using proxy patterns with conventions that make it work seamlessly.
45
+ - **Flexible Proxy Patterns**: Declarative proxy deployment with `deployViaProxy` for upgradeable contracts, including support for [OpenZeppelin](https://openzeppelin.com) transparent proxies.
46
+ - **Diamond Support**: Deploy [EIP-2535 Diamonds](https://eips.ethereum.org/EIPS/eip-2535) declaratively — specify the new state and let hardhat-deploy generate the `diamondCut` for you.
47
+ - **Full Control**: Access to all deployment parameters and lifecycle hooks.
48
+
49
+ ## Key Features
50
+
51
+ - **Chain configuration export** (via `@rocketh/export`) — listing deployed contracts' addresses and their ABIs (useful for web apps)
52
+ - **Library linking** at the time of deployment
53
+ - **Deterministic deployment** across networks
54
+ - **Support for specific deploy scripts per environment** (L1 vs L2 for example)
55
+ - **Deployment dependency system** — only deploy what is needed
56
+ - **Deployment retrying** (by saving pending tx) — recover confidently from interruptions
57
+ - **Deployments as test fixtures** via Hardhat helpers with caching optimization
58
+ - **Helpers to read and execute transactions** on deployed contracts by name
59
+ - **Save metadata** of deployed contracts for full verification via Sourcify or Etherscan
60
+ - **Contract verification** submission at any time since all necessary info is saved
61
+ - **Support for Hardhat's fork feature** — access deployments even when running through fork
62
+
63
+ ## Architecture
64
+
65
+ Version 2 is a full rewrite that has been used in production for several years. It is fully modular, making it much easier to contribute new deployment mechanisms.
66
+
67
+ Under the hood, hardhat-deploy uses [rocketh](https://github.com/wighawag/rocketh), a framework-agnostic system that provides a minimal API to save and load deployments. Everything else is handled by external modules:
68
+
69
+ - **`@rocketh/deploy`** — provides a `deploy` function to deploy contracts
70
+ - **`@rocketh/proxy`** — deploy proxies declaratively like in hardhat-deploy v1
71
+ - **`@rocketh/diamond`** — deploy diamonds declaratively
72
+ - **`@rocketh/read-execute`** — helpers for reading and executing transactions
73
+ - **`@rocketh/viem`** — viem client integration
74
+
75
+ You can also provide your own modules for advanced use cases.
76
+
77
+ ## Quick Example
78
+
79
+ ```typescript
80
+ import { deployScript, artifacts } from "../rocketh/deploy.js";
81
+
82
+ export default deployScript(
83
+ async ({ deployViaProxy, namedAccounts }) => {
84
+ const { deployer, admin } = namedAccounts;
85
+
86
+ await deployViaProxy(
87
+ "GreetingsRegistry",
88
+ {
89
+ account: deployer,
90
+ artifact: artifacts.GreetingsRegistry,
91
+ args: ["prefix:"],
92
+ },
93
+ {
94
+ owner: admin,
95
+ },
96
+ );
97
+ },
98
+ { tags: ["GreetingsRegistry"] },
99
+ );
100
+ ```
101
+
102
+ ## Documentation
103
+
104
+ Please find the [full documentation here](https://rocketh.dev/hardhat-deploy/)
105
+
106
+ ## Template
107
+
108
+ Get started quickly with the [template-ethereum-contracts](https://github.com/wighawag/template-ethereum-contracts) template that provides a production-ready setup with hardhat-deploy and rocketh.
109
+
110
+ ## License
111
+
112
+ MIT
113
+
114
+ ## Sponsor
115
+
116
+ If you find this project useful, please consider sponsoring it! Your support helps me continue developing and maintaining this tool.
117
+
118
+ <a href="https://github.com/sponsors/wighawag">
119
+ <img src="https://img.shields.io/badge/Sponsor-GitHub-181717?style=for-the-badge&logo=github&logoColor=white" alt="Sponsor on GitHub" />
120
+ </a>
package/dist/cli.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+ export {};
3
+ //# sourceMappingURL=cli.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":""}
package/dist/cli.js ADDED
@@ -0,0 +1,197 @@
1
+ #!/usr/bin/env node
2
+ import { Command } from 'commander';
3
+ import { readFileSync, readdirSync, mkdirSync, copyFileSync, existsSync, writeFileSync, statSync } from 'fs';
4
+ import { join, dirname, basename } from 'path';
5
+ import { fileURLToPath } from 'url';
6
+ import * as readline from 'readline';
7
+ import pkg from '../package.json' with { type: 'json' };
8
+ const __filename = fileURLToPath(import.meta.url);
9
+ const __dirname = dirname(__filename);
10
+ const program = new Command();
11
+ // Get the current version of hardhat-deploy
12
+ const getHardhatDeployVersion = () => {
13
+ return pkg.version;
14
+ };
15
+ const askFolder = async () => {
16
+ const rl = readline.createInterface({
17
+ input: process.stdin,
18
+ output: process.stdout,
19
+ });
20
+ return new Promise((resolve) => {
21
+ rl.question('Enter folder path (default: ./): ', (answer) => {
22
+ rl.close();
23
+ resolve(answer.trim() || './');
24
+ });
25
+ });
26
+ };
27
+ const askAutoInstall = async () => {
28
+ const rl = readline.createInterface({
29
+ input: process.stdin,
30
+ output: process.stdout,
31
+ });
32
+ return new Promise((resolve) => {
33
+ rl.question('Auto-install dependencies with pnpm? (Y/n): ', (answer) => {
34
+ rl.close();
35
+ const trimmed = answer.trim().toLowerCase();
36
+ resolve(trimmed === '' || trimmed === 'y' || trimmed === 'yes');
37
+ });
38
+ });
39
+ };
40
+ const isFolderEmpty = (folderPath) => {
41
+ if (!existsSync(folderPath)) {
42
+ return true;
43
+ }
44
+ try {
45
+ const files = readdirSync(folderPath);
46
+ return files.length === 0;
47
+ }
48
+ catch (error) {
49
+ // If we can't read the directory, treat it as not empty
50
+ return false;
51
+ }
52
+ };
53
+ const copyFile = (source, target, replacements = {}, gitignorePatterns = []) => {
54
+ const fileName = basename(source);
55
+ // Check if file should be skipped based on gitignore patterns
56
+ for (const pattern of gitignorePatterns) {
57
+ if (fileName === pattern || fileName.endsWith(pattern.replace('*', ''))) {
58
+ return; // Skip this file
59
+ }
60
+ }
61
+ let content = readFileSync(source, 'utf-8');
62
+ // Apply replacements
63
+ for (const [search, replace] of Object.entries(replacements)) {
64
+ content = content.replaceAll(search, replace);
65
+ }
66
+ mkdirSync(dirname(target), { recursive: true });
67
+ // For binary files, just copy as-is
68
+ if (source.endsWith('.lock') || source.endsWith('.so') || source.endsWith('.wasm')) {
69
+ copyFileSync(source, target);
70
+ }
71
+ else {
72
+ writeFileSync(target, content, 'utf-8');
73
+ }
74
+ };
75
+ const parseGitignore = (gitignorePath) => {
76
+ if (!existsSync(gitignorePath)) {
77
+ return [];
78
+ }
79
+ const content = readFileSync(gitignorePath, 'utf-8');
80
+ return content
81
+ .split('\n')
82
+ .map((line) => line.trim())
83
+ .filter((line) => line && !line.startsWith('#'));
84
+ };
85
+ const copyFolder = (source, target, replacements = {}, gitignorePatterns = []) => {
86
+ if (!existsSync(target)) {
87
+ mkdirSync(target, { recursive: true });
88
+ }
89
+ const files = readdirSync(source);
90
+ files.forEach((file) => {
91
+ const sourcePath = join(source, file);
92
+ const targetPath = join(target, file);
93
+ const stat = statSync(sourcePath);
94
+ if (stat.isDirectory()) {
95
+ // Check if directory should be skipped based on gitignore patterns
96
+ const shouldSkip = gitignorePatterns.some(pattern => file === pattern.replace('/', '') || pattern.startsWith('/') && file === pattern.slice(1));
97
+ if (!shouldSkip) {
98
+ copyFolder(sourcePath, targetPath, replacements, gitignorePatterns);
99
+ }
100
+ }
101
+ else {
102
+ copyFile(sourcePath, targetPath, replacements, gitignorePatterns);
103
+ }
104
+ });
105
+ };
106
+ const generateProject = (targetFolder, projectName) => {
107
+ // find template in published package
108
+ const templatePath = join(__dirname, '../templates/basic');
109
+ const gitignorePath = join(templatePath, '.gitignore');
110
+ // Parse gitignore patterns
111
+ const gitignorePatterns = parseGitignore(gitignorePath);
112
+ // Determine project name from folder or use placeholder
113
+ const folderName = projectName || basename(targetFolder === './' ? process.cwd() : targetFolder);
114
+ // Get the current version of hardhat-deploy
115
+ const hardhatDeployVersion = getHardhatDeployVersion();
116
+ const replacements = {
117
+ 'template-hardhat-node-test-runner': `${folderName}`,
118
+ 'workspace:*': hardhatDeployVersion,
119
+ };
120
+ console.log(`Generating project in: ${targetFolder}`);
121
+ copyFolder(templatePath, targetFolder, replacements, gitignorePatterns);
122
+ console.log('✓ Project initialized successfully!');
123
+ };
124
+ const runPnpmInstall = async (folderPath) => {
125
+ console.log(`Installing dependencies...`);
126
+ const { spawn } = await import('child_process');
127
+ return new Promise((resolve, reject) => {
128
+ // Use --ignore-workspace to ensure dependencies are installed locally
129
+ // This prevents pnpm from treating the target folder as part of a parent workspace
130
+ const pnpm = spawn('pnpm', ['install', '--ignore-workspace', `--no-frozen-lockfile`], {
131
+ cwd: folderPath,
132
+ stdio: 'inherit',
133
+ });
134
+ pnpm.on('close', (code) => {
135
+ if (code === 0) {
136
+ console.log('✓ Dependencies installed successfully!');
137
+ resolve();
138
+ }
139
+ else {
140
+ reject(new Error(`pnpm install failed with exit code ${code}`));
141
+ }
142
+ });
143
+ pnpm.on('error', (error) => {
144
+ reject(error);
145
+ });
146
+ });
147
+ };
148
+ program
149
+ .name('hardhat-deploy')
150
+ .description('CLI for hardhat-deploy')
151
+ .version(pkg.version);
152
+ program
153
+ .command('init')
154
+ .argument('[folder]', 'folder to initialize the project in')
155
+ .option('--install', 'auto-install dependencies with pnpm')
156
+ .description('Initialize a new hardhat-deploy project')
157
+ .action(async (folder, options) => {
158
+ let targetFolder = folder;
159
+ let autoInstall = options?.install ?? false;
160
+ // If no folder specified, ask user
161
+ if (!targetFolder) {
162
+ targetFolder = await askFolder();
163
+ // If we prompted for folder, also prompt for auto-install
164
+ autoInstall = await askAutoInstall();
165
+ }
166
+ // Normalize path
167
+ targetFolder = targetFolder.trim();
168
+ // Check if folder is empty
169
+ if (!isFolderEmpty(targetFolder)) {
170
+ console.error(`Error: Folder "${targetFolder}" is not empty. Please specify an empty folder or a new folder path.`);
171
+ process.exit(1);
172
+ }
173
+ // Generate project
174
+ generateProject(targetFolder);
175
+ // Auto-install if requested
176
+ if (autoInstall) {
177
+ try {
178
+ await runPnpmInstall(targetFolder);
179
+ }
180
+ catch (error) {
181
+ console.error('Failed to install dependencies:', error);
182
+ console.log('\nYou can install dependencies manually:');
183
+ console.log(` cd ${targetFolder === './' ? '.' : targetFolder}`);
184
+ console.log(' pnpm install');
185
+ process.exit(1);
186
+ }
187
+ }
188
+ // Show next steps
189
+ console.log(`\nNext steps:`);
190
+ console.log(` cd ${targetFolder === './' ? '.' : targetFolder}`);
191
+ if (!autoInstall) {
192
+ console.log(` pnpm install`);
193
+ }
194
+ console.log(` pnpm hardhat test`);
195
+ });
196
+ program.parse();
197
+ //# sourceMappingURL=cli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"default.d.ts","sourceRoot":"","sources":["../../src/config/default.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAC,wBAAwB,EAAC,MAAM,aAAa,CAAC;AAE1D,eAAO,MAAM,cAAc,EAAE,wBAG5B,CAAC"}
@@ -0,0 +1,5 @@
1
+ export const DEFAULT_CONFIG = {
2
+ // externalArtifacts: [],
3
+ destinations: [{ mode: 'javascript', folder: './generated' }],
4
+ };
5
+ //# sourceMappingURL=default.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"default.js","sourceRoot":"","sources":["../../src/config/default.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,MAAM,cAAc,GAA6B;IACvD,yBAAyB;IACzB,YAAY,EAAE,CAAC,EAAC,IAAI,EAAE,YAAY,EAAE,MAAM,EAAE,aAAa,EAAC,CAAC;CAC3D,CAAC"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-config.d.ts","sourceRoot":"","sources":["../../src/config/get-config.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAC,wBAAwB,EAAE,4BAA4B,EAAC,MAAM,aAAa,CAAC;AAIxF,wBAAgB,SAAS,CAAC,UAAU,EAAE,4BAA4B,GAAG,SAAS,GAAG,wBAAwB,CAOxG"}
@@ -0,0 +1,9 @@
1
+ import { DEFAULT_CONFIG } from './default.js';
2
+ export function getConfig(userConfig) {
3
+ return {
4
+ destinations: userConfig?.destinations?.map((v) => ({ mode: v.mode || 'javascript', folder: v.folder || './generated' })) ||
5
+ DEFAULT_CONFIG.destinations,
6
+ // externalArtifacts: userConfig?.externalArtifacts || DEFAULT_CONFIG.externalArtifacts,
7
+ };
8
+ }
9
+ //# sourceMappingURL=get-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-config.js","sourceRoot":"","sources":["../../src/config/get-config.ts"],"names":[],"mappings":"AAEA,OAAO,EAAC,cAAc,EAAC,MAAM,cAAc,CAAC;AAE5C,MAAM,UAAU,SAAS,CAAC,UAAoD;IAC7E,OAAO;QACN,YAAY,EACX,UAAU,EAAE,YAAY,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAC,IAAI,EAAE,CAAC,CAAC,IAAI,IAAI,YAAY,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,IAAI,aAAa,EAAC,CAAC,CAAC;YACzG,cAAc,CAAC,YAAY;QAC5B,wFAAwF;KACxF,CAAC;AACH,CAAC"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.d.ts","sourceRoot":"","sources":["../../src/config/validation.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAC,gCAAgC,EAAC,MAAM,oCAAoC,CAAC;AACzF,OAAO,KAAK,EAAC,iBAAiB,EAAC,MAAM,gBAAgB,CAAC;AAmBtD,wBAAsB,2BAA2B,CAChD,UAAU,EAAE,iBAAiB,GAC3B,OAAO,CAAC,gCAAgC,EAAE,CAAC,CAE7C"}
@@ -0,0 +1,17 @@
1
+ import { validateUserConfigZodType } from '@nomicfoundation/hardhat-zod-utils';
2
+ import { z } from 'zod';
3
+ const artifactGenerationUserConfigSchema = z
4
+ .object({
5
+ // externalArtifacts: z.array(z.string()).optional(),
6
+ destinations: z
7
+ .array(z.object({
8
+ mode: z.union([z.literal('javascript'), z.literal('typescript')]).optional(),
9
+ folder: z.string().optional(),
10
+ }))
11
+ .optional(),
12
+ })
13
+ .optional();
14
+ export async function validateTypechainUserConfig(userConfig) {
15
+ return validateUserConfigZodType(userConfig.generateTypedArtifacts, artifactGenerationUserConfigSchema);
16
+ }
17
+ //# sourceMappingURL=validation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.js","sourceRoot":"","sources":["../../src/config/validation.ts"],"names":[],"mappings":"AAGA,OAAO,EAAC,yBAAyB,EAAC,MAAM,oCAAoC,CAAC;AAC7E,OAAO,EAAC,CAAC,EAAC,MAAM,KAAK,CAAC;AAEtB,MAAM,kCAAkC,GAAG,CAAC;KAC1C,MAAM,CAAC;IACP,qDAAqD;IACrD,YAAY,EAAE,CAAC;SACb,KAAK,CACL,CAAC,CAAC,MAAM,CAAC;QACR,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;QAC5E,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE;KAC7B,CAAC,CACF;SACA,QAAQ,EAAE;CACZ,CAAC;KACD,QAAQ,EAAE,CAAC;AAEb,MAAM,CAAC,KAAK,UAAU,2BAA2B,CAChD,UAA6B;IAE7B,OAAO,yBAAyB,CAAC,UAAU,CAAC,sBAAsB,EAAE,kCAAkC,CAAC,CAAC;AACzG,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { ArtifactGenerationConfig } from './types.js';
2
+ export declare function generateTypes(paths: {
3
+ artifacts: string[];
4
+ }, config: ArtifactGenerationConfig): Promise<void>;
5
+ //# sourceMappingURL=generate-types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generate-types.d.ts","sourceRoot":"","sources":["../src/generate-types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAC,wBAAwB,EAAC,MAAM,YAAY,CAAC;AAqKzD,wBAAsB,aAAa,CAAC,KAAK,EAAE;IAAC,SAAS,EAAE,MAAM,EAAE,CAAA;CAAC,EAAE,MAAM,EAAE,wBAAwB,GAAG,OAAO,CAAC,IAAI,CAAC,CAmHjH"}