Skip to content

Commit aa208e3

Browse files
committed
#6: try
1 parent f646979 commit aa208e3

File tree

9 files changed

+255
-10
lines changed

9 files changed

+255
-10
lines changed

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,6 @@ coverage/
66
rdoc/
77
doc/
88
.yardoc/
9+
node_modules/
10+
package.json
11+
package-lock.json

Gemfile

+2
Original file line numberDiff line numberDiff line change
@@ -23,11 +23,13 @@
2323
source 'https://rubygems.org'
2424
gemspec
2525

26+
gem 'donce', '>0', require: false
2627
gem 'loog', '>0', require: false
2728
gem 'minitest', '5.25.4', require: false
2829
gem 'minitest-reporters', '1.7.1', require: false
2930
gem 'minitest-retry', '0.2.5', require: false
3031
gem 'rake', '13.2.1', require: false
32+
gem 'random-port', '>0', require: false
3133
gem 'rspec-rails', '7.1.0', require: false
3234
gem 'rubocop', '1.71.2', require: false
3335
gem 'rubocop-minitest', '0.36.0', require: false

Gemfile.lock

+19-2
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,14 @@ GEM
5151
date (3.4.1)
5252
diff-lcs (1.5.1)
5353
docile (1.4.1)
54+
donce (0.0.5)
55+
backtrace (> 0)
56+
os (> 0)
57+
qbash (> 0)
5458
drb (2.2.1)
59+
elapsed (0.0.1)
60+
loog (> 0)
61+
tago (> 0)
5562
erubi (1.13.1)
5663
faraday (2.12.2)
5764
faraday-net_http (>= 2.0, < 3.5)
@@ -95,6 +102,7 @@ GEM
95102
racc (~> 1.4)
96103
nokogiri (1.18.2-x86_64-linux-gnu)
97104
racc (~> 1.4)
105+
os (1.1.4)
98106
parallel (1.26.3)
99107
parser (3.3.7.1)
100108
ast (~> 2.4.1)
@@ -105,6 +113,11 @@ GEM
105113
psych (5.2.3)
106114
date
107115
stringio
116+
qbash (0.3.2)
117+
backtrace (> 0)
118+
elapsed (> 0)
119+
loog (> 0)
120+
tago (> 0)
108121
racc (1.8.1)
109122
rack (3.1.9)
110123
rack-session (2.1.0)
@@ -131,13 +144,15 @@ GEM
131144
zeitwerk (~> 2.6)
132145
rainbow (3.1.1)
133146
rake (13.2.1)
134-
rdoc (6.11.0)
147+
random-port (0.7.5)
148+
tago (> 0)
149+
rdoc (6.12.0)
135150
psych (>= 4.0.0)
136151
regexp_parser (2.10.0)
137152
reline (0.6.0)
138153
io-console (~> 0.5)
139154
rexml (3.4.0)
140-
rspec-core (3.13.2)
155+
rspec-core (3.13.3)
141156
rspec-support (~> 3.13.0)
142157
rspec-expectations (3.13.3)
143158
diff-lcs (>= 1.2.0, < 2.0)
@@ -213,12 +228,14 @@ PLATFORMS
213228
x86_64-linux
214229

215230
DEPENDENCIES
231+
donce (> 0)
216232
erc20!
217233
loog (> 0)
218234
minitest (= 5.25.4)
219235
minitest-reporters (= 1.7.1)
220236
minitest-retry (= 0.2.5)
221237
rake (= 13.2.1)
238+
random-port (> 0)
222239
rspec-rails (= 7.1.0)
223240
rubocop (= 1.71.2)
224241
rubocop-minitest (= 0.36.0)

hardhat/.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
artifacts/
2+
cache/

hardhat/Dockerfile

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
# Copyright (c) 2025 Yegor Bugayenko
2+
#
3+
# Permission is hereby granted, free of charge, to any person obtaining a copy
4+
# of this software and associated documentation files (the 'Software'), to deal
5+
# in the Software without restriction, including without limitation the rights
6+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7+
# copies of the Software, and to permit persons to whom the Software is
8+
# furnished to do so, subject to the following conditions:
9+
#
10+
# The above copyright notice and this permission notice shall be included in all
11+
# copies or substantial portions of the Software.
12+
#
13+
# THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFINGEMENT. IN NO EVENT SHALL THE
16+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
19+
# SOFTWARE.
20+
21+
FROM node:22
22+
23+
WORKDIR /app
24+
EXPOSE 8545
25+
COPY . hardhat.config.ts
26+
27+
ENV HOME=/app
28+
29+
RUN npm install -g hardhat
30+
#RUN npx hardhat ignition deploy ./ignition/modules/Lock.js

hardhat/contracts/usdt.sol

+146
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
// SPDX-License-Identifier: MIT
2+
pragma solidity ^0.8.28;
3+
4+
/**
5+
* @title Minimal ERC20 Token
6+
* @dev Implements the basic ERC20 interface.
7+
*/
8+
contract MyToken {
9+
// Token details
10+
string public name = "MyToken";
11+
string public symbol = "MTK";
12+
uint8 public decimals = 18;
13+
14+
// Tracks the total token supply
15+
uint256 public totalSupply;
16+
17+
// Owner of the token contract (for minting, etc., if desired)
18+
address public owner;
19+
20+
// Balances for each account
21+
mapping(address => uint256) private _balances;
22+
23+
// Allowances for each account, where allowances[from][spender] = amount
24+
mapping(address => mapping(address => uint256)) private _allowances;
25+
26+
// Events
27+
event Transfer(address indexed from, address indexed to, uint256 value);
28+
event Approval(address indexed owner, address indexed spender, uint256 value);
29+
30+
// Constructor: Mint an initial supply to the deployer
31+
constructor(uint256 initialSupply) {
32+
owner = msg.sender;
33+
_mint(owner, initialSupply);
34+
}
35+
36+
/**
37+
* @dev Returns the balance of a given account.
38+
*/
39+
function balanceOf(address account) external view returns (uint256) {
40+
return _balances[account];
41+
}
42+
43+
/**
44+
* @dev Moves `amount` tokens from caller's account to `recipient`.
45+
*
46+
* Emits a {Transfer} event.
47+
*/
48+
function transfer(address recipient, uint256 amount) external returns (bool) {
49+
_transfer(msg.sender, recipient, amount);
50+
return true;
51+
}
52+
53+
/**
54+
* @dev Returns the remaining number of tokens that `spender` is allowed
55+
* to spend on behalf of `owner` through `transferFrom`.
56+
*/
57+
function allowance(address _owner, address _spender) external view returns (uint256) {
58+
return _allowances[_owner][_spender];
59+
}
60+
61+
/**
62+
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
63+
*
64+
* Emits an {Approval} event.
65+
*/
66+
function approve(address spender, uint256 amount) external returns (bool) {
67+
_approve(msg.sender, spender, amount);
68+
return true;
69+
}
70+
71+
/**
72+
* @dev Moves `amount` tokens from `sender` to `recipient` using the
73+
* allowance mechanism. `amount` is then deducted from the caller's allowance.
74+
*
75+
* Emits a {Transfer} event.
76+
*/
77+
function transferFrom(
78+
address sender,
79+
address recipient,
80+
uint256 amount
81+
) external returns (bool) {
82+
uint256 currentAllowance = _allowances[sender][msg.sender];
83+
require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance");
84+
85+
_approve(sender, msg.sender, currentAllowance - amount);
86+
_transfer(sender, recipient, amount);
87+
88+
return true;
89+
}
90+
91+
/**
92+
* @dev Creates new tokens and assigns them to `account`, increasing the total supply.
93+
*
94+
* Emits a {Transfer} event.
95+
*
96+
* WARNING: This function is public. In production, consider restricting access (e.g. onlyOwner).
97+
*/
98+
function mint(address account, uint256 amount) public {
99+
require(msg.sender == owner, "ERC20: only owner can mint");
100+
_mint(account, amount);
101+
}
102+
103+
// ---------------------
104+
// Internal / Private
105+
// ---------------------
106+
107+
/**
108+
* @dev Internal function that moves `amount` tokens from `sender` to `recipient`.
109+
*/
110+
function _transfer(address sender, address recipient, uint256 amount) internal {
111+
require(sender != address(0), "ERC20: transfer from the zero address");
112+
require(recipient != address(0), "ERC20: transfer to the zero address");
113+
require(_balances[sender] >= amount, "ERC20: transfer amount exceeds balance");
114+
115+
_balances[sender] -= amount;
116+
_balances[recipient] += amount;
117+
118+
emit Transfer(sender, recipient, amount);
119+
}
120+
121+
/**
122+
* @dev Internal function that sets `amount` as the allowance of `spender` over the `owner` s tokens.
123+
*/
124+
function _approve(
125+
address tokenOwner,
126+
address spender,
127+
uint256 amount
128+
) internal {
129+
require(tokenOwner != address(0), "ERC20: approve from the zero address");
130+
require(spender != address(0), "ERC20: approve to the zero address");
131+
132+
_allowances[tokenOwner][spender] = amount;
133+
emit Approval(tokenOwner, spender, amount);
134+
}
135+
136+
/**
137+
* @dev Internal function to create `amount` tokens and assign them to `account`.
138+
*/
139+
function _mint(address account, uint256 amount) internal {
140+
require(account != address(0), "ERC20: mint to the zero address");
141+
142+
totalSupply += amount;
143+
_balances[account] += amount;
144+
emit Transfer(address(0), account, amount);
145+
}
146+
}

hardhat/hardhat.config.js

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
solidity: "0.8.28",
3+
};

hardhat/ignition/modules/deploy.ts

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
// This setup uses Hardhat Ignition to manage smart contract deployments.
2+
// Learn more about it at https://hardhat.org/ignition
3+
4+
import { buildModule } from "@nomicfoundation/hardhat-ignition/modules";
5+
6+
const JAN_1ST_2030 = 1893456000;
7+
const ONE_GWEI: bigint = 1_000_000_000n;
8+
9+
const LockModule = buildModule("LockModule", (m) => {
10+
const unlockTime = m.getParameter("unlockTime", JAN_1ST_2030);
11+
const lockedAmount = m.getParameter("lockedAmount", ONE_GWEI);
12+
13+
const lock = m.contract("Lock", [unlockTime], {
14+
value: lockedAmount,
15+
});
16+
17+
return { lock };
18+
});
19+
20+
export default LockModule;

test/erc20/test_wallet.rb

+30-8
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,10 @@
2020
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
2121
# SOFTWARE.
2222

23-
require 'minitest/autorun'
23+
require 'donce'
2424
require 'loog'
25+
require 'random-port'
26+
require 'minitest/autorun'
2527
require_relative '../../lib/erc20'
2628
require_relative '../../lib/erc20/wallet'
2729

@@ -30,9 +32,12 @@
3032
# Copyright:: Copyright (c) 2025 Yegor Bugayenko
3133
# License:: MIT
3234
class TestWallet < Minitest::Test
35+
# At this address, in the mainnet, there are a few USDT tokens. I won't
36+
# move them anyway, that's why tests can use this address forever.
37+
STABLE_ADDRESS = '0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6'
38+
3339
def test_checks_balance_on_mainnet
34-
a = '0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6'
35-
b = mainnet.balance(a)
40+
b = mainnet.balance(STABLE_ADDRESS)
3641
refute_nil(b)
3742
assert_equal(27_258_889, b)
3843
end
@@ -49,17 +54,34 @@ def test_fails_with_invalid_infura_key
4954
rpc: 'https://mainnet.infura.io/v3/invalid-key-here',
5055
log: Loog::NULL
5156
)
52-
assert_raises(StandardError) { w.balance('0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6') }
57+
assert_raises(StandardError) { w.balance(STABLE_ADDRESS) }
5358
end
5459

55-
def test_checks_balance_on_sepolia
60+
def test_checks_balance_on_testnet
5661
skip('does not work')
57-
a = '0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6'
58-
b = sepolia.balance(a)
62+
b = testnet.balance(STABLE_ADDRESS)
5963
refute_nil(b)
6064
assert_predicate(b, :positive?)
6165
end
6266

67+
def test_checks_balance_on_hardhat
68+
RandomPort::Pool::SINGLETON.acquire do |port|
69+
qbash('npx --config=/app/hardhat.config.ts hardhat node') do |pid|
70+
# donce(
71+
# home: File.join(__dir__, '../../hardhat'),
72+
# ports: { port => 8545 },
73+
# command: 'npx --config=/app/hardhat.config.ts hardhat node',
74+
# log: Loog::VERBOSE,
75+
# root: true
76+
# ) do |_|
77+
sleep 3
78+
w = ERC20::Wallet.new(rpc: "https://#{donce_host}:#{port}", log: Loog::VERBOSE)
79+
b = w.balance('0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266')
80+
assert_equal(11000, b)
81+
end
82+
end
83+
end
84+
6385
def test_sends_payment
6486
skip('does not work yet')
6587
w = ERC20::Wallet.new(log: Loog::VERBOSE)
@@ -103,7 +125,7 @@ def mainnet
103125
end.sample
104126
end
105127

106-
def sepolia
128+
def testnet
107129
[
108130
"https://sepolia.infura.io/v3/#{env('INFURA_KEY')}",
109131
"https://go.getblock.io/#{env('GETBLOCK_SEPOILA_KEY')}"

0 commit comments

Comments
 (0)