erc20 0.0.20 → 0.1.0

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.
data/README.md CHANGED
@@ -54,6 +54,16 @@ eth = w.eth_balance(address)
54
54
  hex = w.eth_pay(private_key, to_address, amount)
55
55
  ```
56
56
 
57
+ To check the price of a gas unit and the expected cost of a payment:
58
+
59
+ ```ruby
60
+ # How many gas units required to send this payment:
61
+ units = w.gas_estimate(from, to, amount)
62
+
63
+ # What is the price of a gas unit, in gwei:
64
+ gwei = w.gas_price
65
+ ```
66
+
57
67
  To generate a new private key, use [eth](https://rubygems.org/gems/eth):
58
68
 
59
69
  ```ruby
@@ -83,6 +93,25 @@ You can use [squid-proxy] [Docker] image to set up your own [HTTP proxy] server.
83
93
  Of course, this library works with [Polygon], [Optimism],
84
94
  and other forks of [Etherium].
85
95
 
96
+ ## How to use in command line
97
+
98
+ This gem also provides a command line tool for sending ETH and ERC20 payments
99
+ and checking balances.
100
+
101
+ First, you install it, via [gem]:
102
+
103
+ ```bash
104
+ gem install erc20
105
+ ```
106
+
107
+ Then, run it:
108
+
109
+ ```bash
110
+ erc20 --help
111
+ ```
112
+
113
+ It should be obvious how to use it. If not, please, submit an issue.
114
+
86
115
  ## How to use in tests
87
116
 
88
117
  You can use `ERC20::FakeWallet` class that behaves exactly like
@@ -93,7 +122,7 @@ Also, it remembers all requests that were sent to it:
93
122
  require 'erc20'
94
123
  w = ERC20::FakeWallet.new
95
124
  w.pay(priv, address, 42_000)
96
- assert w.history.include?({ method: :pay, params: [priv, address, 42_000] })
125
+ assert w.history.include?({ method: :pay, priv:, address:, amount: 42_000 })
97
126
  ```
98
127
 
99
128
  ## How to contribute
@@ -112,6 +141,7 @@ bundle exec rake
112
141
 
113
142
  If it's clean and you don't see any error messages, submit your pull request.
114
143
 
144
+ [gem]: https://github.com/rubygems/rubygems
115
145
  [Etherium]: https://en.wikipedia.org/wiki/Ethereum
116
146
  [ERC20]: https://ethereum.org/en/developers/docs/standards/tokens/erc-20/
117
147
  [JSON-RPC]: https://ethereum.org/en/developers/docs/apis/json-rpc/
data/REUSE.toml ADDED
@@ -0,0 +1,31 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ # SPDX-License-Identifier: MIT
3
+
4
+ version = 1
5
+ [[annotations]]
6
+ path = [
7
+ "**/*.csv",
8
+ "**/*.jpg",
9
+ "**/*.json",
10
+ "**/*.md",
11
+ "**/*.pdf",
12
+ "**/*.png",
13
+ "**/*.svg",
14
+ "**/*.txt",
15
+ "**/*.vm",
16
+ "**/.gitignore",
17
+ "**/CNAME",
18
+ ".gitattributes",
19
+ ".gitignore",
20
+ ".gitleaksignore",
21
+ ".pdd",
22
+ ".xcop",
23
+ "Gemfile.lock",
24
+ "README.md",
25
+ "hardhat/.gitignore",
26
+ "hardhat/package.json",
27
+ "renovate.json",
28
+ ]
29
+ precedence = "override"
30
+ SPDX-FileCopyrightText = "Copyright (c) 2025 Yegor Bugayenko"
31
+ SPDX-License-Identifier = "MIT"
data/Rakefile CHANGED
@@ -1,24 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- # Copyright (c) 2025 Yegor Bugayenko
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.
3
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
4
+ # SPDX-License-Identifier: MIT
22
5
 
23
6
  require 'rubygems'
24
7
  require 'rake'
@@ -34,7 +17,7 @@ def version
34
17
  Gem::Specification.load(Dir['*.gemspec'].first).version
35
18
  end
36
19
 
37
- task default: %i[clean test rubocop yard]
20
+ task default: %i[clean test features rubocop yard]
38
21
 
39
22
  require 'rake/testtask'
40
23
  desc 'Run all unit tests'
@@ -59,6 +42,14 @@ RuboCop::RakeTask.new(:rubocop) do |task|
59
42
  task.requires << 'rubocop-rspec'
60
43
  end
61
44
 
45
+ require 'cucumber/rake/task'
46
+ Cucumber::Rake::Task.new(:features) do
47
+ Rake::Cleaner.cleanup_files(['coverage'])
48
+ end
49
+ Cucumber::Rake::Task.new(:'features:html') do |t|
50
+ t.profile = 'html_report'
51
+ end
52
+
62
53
  desc 'Run benchmark script'
63
54
  task :benchmark do
64
55
  ruby 'benchmarks/simple.rb'
data/bin/erc20 ADDED
@@ -0,0 +1,152 @@
1
+ #!/usr/bin/env ruby
2
+ # frozen_string_literal: true
3
+
4
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
5
+ # SPDX-License-Identifier: MIT
6
+
7
+ $stdout.sync = true
8
+
9
+ require 'backtrace'
10
+ require 'loog'
11
+ require 'slop'
12
+ require_relative '../lib/erc20'
13
+ require_relative '../lib/erc20/erc20'
14
+ require_relative '../lib/erc20/wallet'
15
+
16
+ begin
17
+ begin
18
+ opts = Slop.parse(ARGV, strict: true, help: true) do |o|
19
+ o.banner = "Usage (#{ERC20::VERSION}): erc20 [options] command [args]
20
+ Commands are:
21
+ key: Generate a new Etherium private key (64 symbols)
22
+ address: Turn private key into a public address (44 symbols)
23
+ price: Get current price of one gas unit, in gwei
24
+ pay: Send ERC20 payment
25
+ eth_pay: Send ETH payment
26
+ balance: Get ERC20 balance
27
+ eth_balance: Get ETH balance
28
+ Options are:"
29
+ o.string(
30
+ '--contract',
31
+ 'Public address of ERC20 contract',
32
+ default: ERC20::Wallet::USDT
33
+ )
34
+ o.integer(
35
+ '--chain',
36
+ 'Etherium chain ID',
37
+ default: 1
38
+ )
39
+ o.string(
40
+ '--host',
41
+ 'Host name of the provider',
42
+ default: 'eth.llamarpc.com'
43
+ )
44
+ o.string(
45
+ '--port',
46
+ 'TCP port of the provider',
47
+ default: 443
48
+ )
49
+ o.string(
50
+ '--http_path',
51
+ 'URL path for the HTTP RPC entry point of the provider',
52
+ default: '/'
53
+ )
54
+ o.string(
55
+ '--ws_path',
56
+ 'URL path for the Websockets entry point of the provider',
57
+ default: '/'
58
+ )
59
+ o.string(
60
+ '--proxy',
61
+ 'HTTP/S proxy for all requests, e.g. "localhost:3128"'
62
+ )
63
+ o.integer(
64
+ '--attempts',
65
+ 'How many times should we try before failing',
66
+ default: 1
67
+ )
68
+ o.bool(
69
+ '--dry',
70
+ 'Don\'t send a real payment, run in a read-only mode'
71
+ )
72
+ o.bool('--help', 'Read this: https://github.com/yegor256/erc20') do
73
+ puts o
74
+ exit
75
+ end
76
+ o.bool('--verbose', 'Print all possible debug messages')
77
+ end
78
+ rescue Slop::Error => e
79
+ raise e.message
80
+ end
81
+ raise 'Try --help' if opts.arguments.empty?
82
+ log = opts[:verbose] ? Loog::VERBOSE : Loog::REGULAR
83
+ wallet = ERC20::Wallet.new(
84
+ contract: opts[:contract],
85
+ host: opts[:host], port: opts[:port],
86
+ http_path: opts[:http_path], ws_path: opts[:ws_path],
87
+ log:
88
+ )
89
+ case opts.arguments[0]
90
+ when 'key'
91
+ puts Eth::Key.new.private_hex
92
+ when 'address'
93
+ puts Eth::Key.new(priv: opts.arguments[1]).address.to_s
94
+ when 'price'
95
+ puts wallet.gas_price
96
+ when 'balance'
97
+ address = opts.arguments[1]
98
+ raise 'Address is required' if address.nil?
99
+ log.debug("Checking ERC20 balance of #{address}")
100
+ tokens = wallet.balance(address)
101
+ log.debug("The balance is #{tokens} ERC20 tokens (#{tokens.to_f / 1_000_000} USDT)")
102
+ puts tokens
103
+ when 'eth_balance'
104
+ address = opts.arguments[1]
105
+ raise 'Address is required' if address.nil?
106
+ log.debug("Checking ETH balance of #{address}")
107
+ wei = wallet.eth_balance(address)
108
+ log.debug("The balance of #{address} is #{wei} wei (#{format('%0.4f', wei.to_f / 1_000_000_000_000_000_000)} ETH)")
109
+ puts wei
110
+ when 'pay'
111
+ pkey = opts.arguments[1]
112
+ raise 'Private key is required' if pkey.nil?
113
+ priv = Eth::Key.new(priv: pkey)
114
+ log.debug("Sending ERC20 tokens from #{priv.address.to_s}")
115
+ address = opts.arguments[2]
116
+ raise 'Address is required' if address.nil?
117
+ log.debug("Sending ERC20 tokens to #{address}")
118
+ amount = opts.arguments[3]
119
+ raise 'Amount argument is required' if amount.nil?
120
+ raise 'Amount is not valid' unless /^[0-9]+(usdt)?$/.match?(amount)
121
+ amount = amount.to_i if /^[0-9]+$/.match?(amount)
122
+ amount = (amount.gsub(/usdt^/, '').to_f * 1_000_000).to_i if /^[0-9]usdt+$/.match?(amount)
123
+ log.debug("Sending #{amount} ERC20 tokens")
124
+ puts wallet.pay(priv.private_hex, address, amount)
125
+ when 'eth_pay'
126
+ pkey = opts.arguments[1]
127
+ raise 'Private key is required' if pkey.nil?
128
+ priv = Eth::Key.new(priv: pkey)
129
+ log.debug("Sending ETH from #{priv.address.to_s}")
130
+ address = opts.arguments[2]
131
+ raise 'Address is required' if address.nil?
132
+ log.debug("Sending ETH to #{address}")
133
+ amount = opts.arguments[3]
134
+ raise 'Amount argument is required' if amount.nil?
135
+ raise "Amount #{amount.inspect} is not valid" unless /^[0-9]+(\.[0-9]+)?(eth|wei|gwei)?$/.match?(amount)
136
+ amount = amount.to_i if /^[0-9]+$/.match?(amount)
137
+ amount = amount.gsub(/wei^/, '').to_i if /[0-9]wei+$/.match?(amount)
138
+ amount = (amount.gsub(/gwei^/, '').to_f * 1_000_000_000).to_i if /[0-9]gwei+$/.match?(amount)
139
+ amount = (amount.gsub(/eth^/, '').to_f * 1_000_000_000_000_000_000).to_i if /[0-9]eth+$/.match?(amount)
140
+ log.debug("Sending #{amount} wei...")
141
+ puts wallet.eth_pay(priv.private_hex, address, amount)
142
+ else
143
+ raise "Command #{opts.arguments[0]} is not supported"
144
+ end
145
+ rescue StandardError => e
146
+ if opts[:verbose]
147
+ puts Backtrace.new(e)
148
+ else
149
+ puts "ERROR: #{e.message}"
150
+ end
151
+ exit(255)
152
+ end
data/erc20.gemspec CHANGED
@@ -1,24 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- # Copyright (c) 2025 Yegor Bugayenko
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.
3
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
4
+ # SPDX-License-Identifier: MIT
22
5
 
23
6
  require 'English'
24
7
  require_relative 'lib/erc20/erc20'
@@ -40,11 +23,13 @@ Gem::Specification.new do |s|
40
23
  s.homepage = 'http://github.com/yegor256/erc20.rb'
41
24
  s.files = `git ls-files`.split($RS)
42
25
  s.rdoc_options = ['--charset=UTF-8']
26
+ s.executables = s.files.grep(%r{^bin/}) { |f| File.basename(f) }
43
27
  s.extra_rdoc_files = ['README.md', 'LICENSE.txt']
44
28
  s.add_dependency 'eth', '>=0.5.13'
45
29
  s.add_dependency 'faye-websocket', '>=0.11.3'
46
30
  s.add_dependency 'json', '>=2.10.1'
47
31
  s.add_dependency 'jsonrpc-client', '>=0.1.4'
48
32
  s.add_dependency 'loog', '>0'
33
+ s.add_dependency 'slop', '>4'
49
34
  s.metadata['rubygems_mfa_required'] = 'true'
50
35
  end
@@ -0,0 +1,30 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ # SPDX-License-Identifier: MIT
3
+
4
+ Feature: Command Line Processing
5
+ As a simple ETH/ERC20 user I want to send payments
6
+
7
+ Scenario: Help can be printed
8
+ When I run bin/erc20 with "--help"
9
+ Then Exit code is zero
10
+ And Stdout contains "--help"
11
+
12
+ Scenario: Gas price price can be retrieved
13
+ When I run bin/erc20 with "price --attempts=4"
14
+ Then Exit code is zero
15
+
16
+ Scenario: ETH private key can be generated
17
+ When I run bin/erc20 with "key"
18
+ Then Exit code is zero
19
+
20
+ Scenario: ETH address can be created
21
+ When I run bin/erc20 with "address 46feba063e9b59a8ae0dba68abd39a3cb8f52089e776576d6eb1bb5bfec123d1"
22
+ Then Exit code is zero
23
+
24
+ Scenario: ERC20 balance can be checked
25
+ When I run bin/erc20 with "balance 0x7232148927F8a580053792f44D4d59d40Fd00ABD --verbose"
26
+ Then Exit code is zero
27
+
28
+ Scenario: ETH balance can be checked
29
+ When I run bin/erc20 with "eth_balance 0x7232148927F8a580053792f44D4d59d40Fd00ABD --verbose"
30
+ Then Exit code is zero
@@ -0,0 +1,27 @@
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ # SPDX-License-Identifier: MIT
3
+
4
+ Feature: Gem Package
5
+ As a source code writer I want to be able to
6
+ package the Gem into .gem file
7
+
8
+ Scenario: Gem can be packaged
9
+ Given It is Unix
10
+ Given I have a "execs.rb" file with content:
11
+ """
12
+ #!/usr/bin/env ruby
13
+ require 'rubygems'
14
+ spec = Gem::Specification::load('./spec.rb')
15
+ if spec.executables.empty?
16
+ fail 'no executables: ' + File.read('./spec.rb')
17
+ end
18
+ """
19
+ When I run bash with:
20
+ """
21
+ cd erc20
22
+ gem build erc20.gemspec
23
+ gem specification --ruby erc20-*.gem > ../spec.rb
24
+ cd ..
25
+ ruby execs.rb
26
+ """
27
+ Then Exit code is zero
@@ -0,0 +1,72 @@
1
+ # frozen_string_literal: true
2
+
3
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
4
+ # SPDX-License-Identifier: MIT
5
+
6
+ require 'tmpdir'
7
+ require 'slop'
8
+ require 'English'
9
+ require_relative '../../lib/erc20'
10
+
11
+ Before do
12
+ @cwd = Dir.pwd
13
+ @dir = Dir.mktmpdir('test')
14
+ FileUtils.mkdir_p(@dir)
15
+ Dir.chdir(@dir)
16
+ @opts =
17
+ Slop.parse ['-v'] do |o|
18
+ o.bool '-v', '--verbose'
19
+ end
20
+ end
21
+
22
+ After do
23
+ Dir.chdir(@cwd)
24
+ FileUtils.rm_rf(@dir)
25
+ end
26
+
27
+ Given(/^I have a "([^"]*)" file with content:$/) do |file, text|
28
+ FileUtils.mkdir_p(File.dirname(file)) unless File.exist?(file)
29
+ File.write(file, text.gsub('\\xFF', 0xFF.chr))
30
+ end
31
+
32
+ When(%r{^I run bin/erc20 with "([^"]*)"$}) do |arg|
33
+ home = File.join(File.dirname(__FILE__), '../..')
34
+ @stdout = `ruby -I#{home}/lib #{home}/bin/erc20 #{arg}`
35
+ @exitstatus = $CHILD_STATUS.exitstatus
36
+ end
37
+
38
+ Then(/^Stdout contains "([^"]*)"$/) do |txt|
39
+ raise "STDOUT doesn't contain '#{txt}':\n#{@stdout}" unless @stdout.include?(txt)
40
+ end
41
+
42
+ Then(/^Stdout is empty$/) do
43
+ raise "STDOUT is not empty:\n#{@stdout}" unless @stdout == ''
44
+ end
45
+
46
+ Then(/^Exit code is zero$/) do
47
+ raise "Non-zero exit #{@exitstatus}:\n#{@stdout}" unless @exitstatus.zero?
48
+ end
49
+
50
+ Then(/^Exit code is not zero$/) do
51
+ raise 'Zero exit code' if @exitstatus.zero?
52
+ end
53
+
54
+ When(/^I run bash with "([^"]*)"$/) do |text|
55
+ FileUtils.copy_entry(@cwd, File.join(@dir, 'erc20'))
56
+ @stdout = `#{text}`
57
+ @exitstatus = $CHILD_STATUS.exitstatus
58
+ end
59
+
60
+ When(/^I run bash with:$/) do |text|
61
+ FileUtils.copy_entry(@cwd, File.join(@dir, 'erc20'))
62
+ @stdout = `#{text}`
63
+ @exitstatus = $CHILD_STATUS.exitstatus
64
+ end
65
+
66
+ Given(/^It is Unix$/) do
67
+ pending if Gem.win_platform?
68
+ end
69
+
70
+ Given(/^It is Windows$/) do
71
+ pending unless Gem.win_platform?
72
+ end
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
4
+ # SPDX-License-Identifier: MIT
5
+
6
+ require 'simplecov'
7
+ require 'cucumber'
8
+ require_relative '../../lib/erc20'
data/hardhat/Dockerfile CHANGED
@@ -1,22 +1,5 @@
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.
1
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ # SPDX-License-Identifier: MIT
20
3
 
21
4
  FROM node:22
22
5
 
@@ -1,4 +1,6 @@
1
+ // SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
1
2
  // SPDX-License-Identifier: MIT
3
+
2
4
  pragma solidity ^0.8.28;
3
5
 
4
6
  contract Foo {
@@ -1,22 +1,5 @@
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.
1
+ // SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ // SPDX-License-Identifier: MIT
20
3
 
21
4
  require("@nomicfoundation/hardhat-toolbox");
22
5
 
@@ -1,22 +1,5 @@
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.
1
+ // SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
2
+ // SPDX-License-Identifier: MIT
20
3
 
21
4
  import { buildModule } from "@nomicfoundation/hardhat-ignition/modules";
22
5
 
data/lib/erc20/erc20.rb CHANGED
@@ -1,24 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- # Copyright (c) 2025 Yegor Bugayenko
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 NONINFINGEMENT. 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.
3
+ # SPDX-FileCopyrightText: Copyright (c) 2025 Yegor Bugayenko
4
+ # SPDX-License-Identifier: MIT
22
5
 
23
6
  # This module makes manipulations with Etherium ERC20 tokens
24
7
  # as simple as they can be, if you have a provider of
@@ -42,5 +25,5 @@
42
25
  # License:: MIT
43
26
  module ERC20
44
27
  # Current version of the gem (changed by the +.rultor.yml+ on every release)
45
- VERSION = '0.0.20'
28
+ VERSION = '0.1.0'
46
29
  end