glueby 0.4.4 → 0.5.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.
- checksums.yaml +4 -4
- data/README.md +463 -387
- data/glueby.gemspec +33 -33
- data/lib/generators/glueby/contract/templates/initializer.rb.erb +8 -8
- data/lib/generators/glueby/contract/templates/key_table.rb.erb +16 -16
- data/lib/generators/glueby/contract/templates/utxo_table.rb.erb +16 -16
- data/lib/glueby/block_syncer.rb +97 -97
- data/lib/glueby/configuration.rb +26 -2
- data/lib/glueby/contract/timestamp/syncer.rb +13 -13
- data/lib/glueby/contract/timestamp.rb +108 -102
- data/lib/glueby/contract/token.rb +270 -244
- data/lib/glueby/contract/tx_builder.rb +97 -30
- data/lib/glueby/fee_provider/tasks.rb +140 -140
- data/lib/glueby/fee_provider.rb +11 -0
- data/lib/glueby/internal/wallet/abstract_wallet_adapter.rb +151 -151
- data/lib/glueby/internal/wallet/active_record/utxo.rb +51 -51
- data/lib/glueby/internal/wallet/active_record_wallet_adapter/syncer.rb +13 -13
- data/lib/glueby/internal/wallet/active_record_wallet_adapter.rb +151 -151
- data/lib/glueby/internal/wallet/tapyrus_core_wallet_adapter.rb +186 -186
- data/lib/glueby/internal/wallet.rb +163 -162
- data/lib/glueby/railtie.rb +10 -9
- data/lib/glueby/utxo_provider/tasks.rb +135 -0
- data/lib/glueby/utxo_provider.rb +85 -0
- data/lib/glueby/version.rb +3 -3
- data/lib/glueby.rb +40 -39
- data/lib/tasks/glueby/block_syncer.rake +28 -28
- data/lib/tasks/glueby/contract/timestamp.rake +46 -39
- data/lib/tasks/glueby/fee_provider.rake +18 -18
- data/lib/tasks/glueby/utxo_provider.rake +18 -0
- metadata +5 -2
@@ -1,245 +1,271 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
require 'active_record'
|
3
|
-
|
4
|
-
module Glueby
|
5
|
-
module Contract
|
6
|
-
# This class represents custom token issued by application user.
|
7
|
-
# Application users can
|
8
|
-
# - issue their own tokens.
|
9
|
-
# - send to other users.
|
10
|
-
# - make the tokens disable.
|
11
|
-
#
|
12
|
-
# Examples:
|
13
|
-
#
|
14
|
-
# alice = Glueby::Wallet.create
|
15
|
-
# bob = Glueby::Wallet.create
|
16
|
-
#
|
17
|
-
# Use `Glueby::Internal::Wallet#receive_address` to generate the address of bob
|
18
|
-
# bob.internal_wallet.receive_address
|
19
|
-
# => '1CY6TSSARn8rAFD9chCghX5B7j4PKR8S1a'
|
20
|
-
#
|
21
|
-
# Issue
|
22
|
-
# token = Token.issue!(issuer: alice, amount: 100)
|
23
|
-
# token.amount(wallet: alice)
|
24
|
-
# => 100
|
25
|
-
#
|
26
|
-
# Send
|
27
|
-
# token.transfer!(sender: alice, receiver_address: '1CY6TSSARn8rAFD9chCghX5B7j4PKR8S1a', amount: 1)
|
28
|
-
# token.amount(wallet: alice)
|
29
|
-
# => 99
|
30
|
-
# token.amount(wallet: bob)
|
31
|
-
# => 1
|
32
|
-
#
|
33
|
-
# Burn
|
34
|
-
# token.burn!(sender: alice, amount: 10)
|
35
|
-
# token.amount(wallet: alice)
|
36
|
-
# => 89
|
37
|
-
# token.burn!(sender: alice)
|
38
|
-
# token.amount(wallet: alice)
|
39
|
-
# => 0
|
40
|
-
#
|
41
|
-
# Reissue
|
42
|
-
# token.reissue!(issuer: alice, amount: 100)
|
43
|
-
# token.amount(wallet: alice)
|
44
|
-
# => 100
|
45
|
-
#
|
46
|
-
class Token
|
47
|
-
include Glueby::Contract::TxBuilder
|
48
|
-
extend Glueby::Contract::TxBuilder
|
49
|
-
|
50
|
-
class << self
|
51
|
-
# Issue new token with specified amount and token type.
|
52
|
-
# REISSUABLE token can be reissued with #reissue! method, and
|
53
|
-
# NON_REISSUABLE and NFT token can not.
|
54
|
-
# Amount is set to 1 when the token type is NFT
|
55
|
-
#
|
56
|
-
# @param issuer [Glueby::Wallet]
|
57
|
-
# @param token_type [TokenTypes]
|
58
|
-
# @param amount [Integer]
|
59
|
-
# @return [Array<token, Array<tx>>] Tuple of tx array and token object
|
60
|
-
# @raise [InsufficientFunds] if wallet does not have enough TPC to send transaction.
|
61
|
-
# @raise [InvalidAmount] if amount is not positive integer.
|
62
|
-
# @raise [UnspportedTokenType] if token is not supported.
|
63
|
-
def issue!(issuer:, token_type: Tapyrus::Color::TokenTypes::REISSUABLE, amount: 1)
|
64
|
-
raise Glueby::Contract::Errors::InvalidAmount unless amount.positive?
|
65
|
-
|
66
|
-
txs, color_id = case token_type
|
67
|
-
when Tapyrus::Color::TokenTypes::REISSUABLE
|
68
|
-
issue_reissuable_token(issuer: issuer, amount: amount)
|
69
|
-
when Tapyrus::Color::TokenTypes::NON_REISSUABLE
|
70
|
-
issue_non_reissuable_token(issuer: issuer, amount: amount)
|
71
|
-
when Tapyrus::Color::TokenTypes::NFT
|
72
|
-
issue_nft_token(issuer: issuer)
|
73
|
-
else
|
74
|
-
raise Glueby::Contract::Errors::UnsupportedTokenType
|
75
|
-
end
|
76
|
-
|
77
|
-
[new(color_id: color_id), txs]
|
78
|
-
end
|
79
|
-
|
80
|
-
private
|
81
|
-
|
82
|
-
def issue_reissuable_token(issuer:, amount:)
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
tx = issuer
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
end
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
#
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
#
|
163
|
-
#
|
164
|
-
# @param sender [Glueby::Wallet] wallet to send this token
|
165
|
-
# @param
|
166
|
-
# @
|
167
|
-
# @
|
168
|
-
# @raise [
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
#
|
187
|
-
# @
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require 'active_record'
|
3
|
+
|
4
|
+
module Glueby
|
5
|
+
module Contract
|
6
|
+
# This class represents custom token issued by application user.
|
7
|
+
# Application users can
|
8
|
+
# - issue their own tokens.
|
9
|
+
# - send to other users.
|
10
|
+
# - make the tokens disable.
|
11
|
+
#
|
12
|
+
# Examples:
|
13
|
+
#
|
14
|
+
# alice = Glueby::Wallet.create
|
15
|
+
# bob = Glueby::Wallet.create
|
16
|
+
#
|
17
|
+
# Use `Glueby::Internal::Wallet#receive_address` to generate the address of bob
|
18
|
+
# bob.internal_wallet.receive_address
|
19
|
+
# => '1CY6TSSARn8rAFD9chCghX5B7j4PKR8S1a'
|
20
|
+
#
|
21
|
+
# Issue
|
22
|
+
# token = Token.issue!(issuer: alice, amount: 100)
|
23
|
+
# token.amount(wallet: alice)
|
24
|
+
# => 100
|
25
|
+
#
|
26
|
+
# Send
|
27
|
+
# token.transfer!(sender: alice, receiver_address: '1CY6TSSARn8rAFD9chCghX5B7j4PKR8S1a', amount: 1)
|
28
|
+
# token.amount(wallet: alice)
|
29
|
+
# => 99
|
30
|
+
# token.amount(wallet: bob)
|
31
|
+
# => 1
|
32
|
+
#
|
33
|
+
# Burn
|
34
|
+
# token.burn!(sender: alice, amount: 10)
|
35
|
+
# token.amount(wallet: alice)
|
36
|
+
# => 89
|
37
|
+
# token.burn!(sender: alice)
|
38
|
+
# token.amount(wallet: alice)
|
39
|
+
# => 0
|
40
|
+
#
|
41
|
+
# Reissue
|
42
|
+
# token.reissue!(issuer: alice, amount: 100)
|
43
|
+
# token.amount(wallet: alice)
|
44
|
+
# => 100
|
45
|
+
#
|
46
|
+
class Token
|
47
|
+
include Glueby::Contract::TxBuilder
|
48
|
+
extend Glueby::Contract::TxBuilder
|
49
|
+
|
50
|
+
class << self
|
51
|
+
# Issue new token with specified amount and token type.
|
52
|
+
# REISSUABLE token can be reissued with #reissue! method, and
|
53
|
+
# NON_REISSUABLE and NFT token can not.
|
54
|
+
# Amount is set to 1 when the token type is NFT
|
55
|
+
#
|
56
|
+
# @param issuer [Glueby::Wallet]
|
57
|
+
# @param token_type [TokenTypes]
|
58
|
+
# @param amount [Integer]
|
59
|
+
# @return [Array<token, Array<tx>>] Tuple of tx array and token object
|
60
|
+
# @raise [InsufficientFunds] if wallet does not have enough TPC to send transaction.
|
61
|
+
# @raise [InvalidAmount] if amount is not positive integer.
|
62
|
+
# @raise [UnspportedTokenType] if token is not supported.
|
63
|
+
def issue!(issuer:, token_type: Tapyrus::Color::TokenTypes::REISSUABLE, amount: 1)
|
64
|
+
raise Glueby::Contract::Errors::InvalidAmount unless amount.positive?
|
65
|
+
|
66
|
+
txs, color_id = case token_type
|
67
|
+
when Tapyrus::Color::TokenTypes::REISSUABLE
|
68
|
+
issue_reissuable_token(issuer: issuer, amount: amount)
|
69
|
+
when Tapyrus::Color::TokenTypes::NON_REISSUABLE
|
70
|
+
issue_non_reissuable_token(issuer: issuer, amount: amount)
|
71
|
+
when Tapyrus::Color::TokenTypes::NFT
|
72
|
+
issue_nft_token(issuer: issuer)
|
73
|
+
else
|
74
|
+
raise Glueby::Contract::Errors::UnsupportedTokenType
|
75
|
+
end
|
76
|
+
|
77
|
+
[new(color_id: color_id), txs]
|
78
|
+
end
|
79
|
+
|
80
|
+
private
|
81
|
+
|
82
|
+
def issue_reissuable_token(issuer:, amount:)
|
83
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
84
|
+
funding_tx = create_funding_tx(wallet: issuer, utxo_provider: utxo_provider)
|
85
|
+
script_pubkey = funding_tx.outputs.first.script_pubkey
|
86
|
+
color_id = Tapyrus::Color::ColorIdentifier.reissuable(script_pubkey)
|
87
|
+
|
88
|
+
ActiveRecord::Base.transaction(joinable: false, requires_new: true) do
|
89
|
+
# Store the script_pubkey for reissue the token.
|
90
|
+
Glueby::Contract::AR::ReissuableToken.create!(color_id: color_id.to_hex, script_pubkey: script_pubkey.to_hex)
|
91
|
+
|
92
|
+
funding_tx = issuer.internal_wallet.broadcast(funding_tx)
|
93
|
+
tx = create_issue_tx_for_reissuable_token(funding_tx: funding_tx, issuer: issuer, amount: amount)
|
94
|
+
tx = issuer.internal_wallet.broadcast(tx)
|
95
|
+
[[funding_tx, tx], color_id]
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
def issue_non_reissuable_token(issuer:, amount:)
|
100
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
101
|
+
funding_tx = create_funding_tx(wallet: issuer, utxo_provider: utxo_provider) if utxo_provider
|
102
|
+
funding_tx = issuer.internal_wallet.broadcast(funding_tx) if funding_tx
|
103
|
+
|
104
|
+
tx = create_issue_tx_for_non_reissuable_token(funding_tx: funding_tx, issuer: issuer, amount: amount)
|
105
|
+
tx = issuer.internal_wallet.broadcast(tx)
|
106
|
+
|
107
|
+
out_point = tx.inputs.first.out_point
|
108
|
+
color_id = Tapyrus::Color::ColorIdentifier.non_reissuable(out_point)
|
109
|
+
if funding_tx
|
110
|
+
[[funding_tx, tx], color_id]
|
111
|
+
else
|
112
|
+
[[tx], color_id]
|
113
|
+
end
|
114
|
+
end
|
115
|
+
|
116
|
+
def issue_nft_token(issuer:)
|
117
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
118
|
+
funding_tx = create_funding_tx(wallet: issuer, utxo_provider: utxo_provider) if utxo_provider
|
119
|
+
funding_tx = issuer.internal_wallet.broadcast(funding_tx) if funding_tx
|
120
|
+
|
121
|
+
tx = create_issue_tx_for_nft_token(funding_tx: funding_tx, issuer: issuer)
|
122
|
+
tx = issuer.internal_wallet.broadcast(tx)
|
123
|
+
|
124
|
+
out_point = tx.inputs.first.out_point
|
125
|
+
color_id = Tapyrus::Color::ColorIdentifier.nft(out_point)
|
126
|
+
if funding_tx
|
127
|
+
[[funding_tx, tx], color_id]
|
128
|
+
else
|
129
|
+
[[tx], color_id]
|
130
|
+
end
|
131
|
+
end
|
132
|
+
end
|
133
|
+
|
134
|
+
attr_reader :color_id
|
135
|
+
|
136
|
+
# Re-issue the token with specified amount.
|
137
|
+
# A wallet can issue the token only when it is REISSUABLE token.
|
138
|
+
# @param issuer [Glueby::Wallet]
|
139
|
+
# @param amount [Integer]
|
140
|
+
# @return [Array<String, tx>] Tuple of color_id and tx object
|
141
|
+
# @raise [InsufficientFunds] if wallet does not have enough TPC to send transaction.
|
142
|
+
# @raise [InvalidAmount] if amount is not positive integer.
|
143
|
+
# @raise [InvalidTokenType] if token is not reissuable.
|
144
|
+
# @raise [UnknownScriptPubkey] when token is reissuable but it doesn't know script pubkey to issue token.
|
145
|
+
def reissue!(issuer:, amount:)
|
146
|
+
raise Glueby::Contract::Errors::InvalidAmount unless amount.positive?
|
147
|
+
raise Glueby::Contract::Errors::InvalidTokenType unless token_type == Tapyrus::Color::TokenTypes::REISSUABLE
|
148
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
149
|
+
|
150
|
+
if validate_reissuer(wallet: issuer)
|
151
|
+
funding_tx = create_funding_tx(wallet: issuer, script: @script_pubkey, utxo_provider: utxo_provider)
|
152
|
+
funding_tx = issuer.internal_wallet.broadcast(funding_tx)
|
153
|
+
tx = create_reissue_tx(funding_tx: funding_tx, issuer: issuer, amount: amount, color_id: color_id)
|
154
|
+
tx = issuer.internal_wallet.broadcast(tx)
|
155
|
+
|
156
|
+
[color_id, tx]
|
157
|
+
else
|
158
|
+
raise Glueby::Contract::Errors::UnknownScriptPubkey
|
159
|
+
end
|
160
|
+
end
|
161
|
+
|
162
|
+
# Send the token to other wallet
|
163
|
+
#
|
164
|
+
# @param sender [Glueby::Wallet] wallet to send this token
|
165
|
+
# @param receiver_address [String] address to receive this token
|
166
|
+
# @param amount [Integer]
|
167
|
+
# @return [Array<String, tx>] Tuple of color_id and tx object
|
168
|
+
# @raise [InsufficientFunds] if wallet does not have enough TPC to send transaction.
|
169
|
+
# @raise [InsufficientTokens] if wallet does not have enough token to send.
|
170
|
+
# @raise [InvalidAmount] if amount is not positive integer.
|
171
|
+
def transfer!(sender:, receiver_address:, amount: 1)
|
172
|
+
raise Glueby::Contract::Errors::InvalidAmount unless amount.positive?
|
173
|
+
|
174
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
175
|
+
funding_tx = create_funding_tx(wallet: sender, utxo_provider: utxo_provider) if utxo_provider
|
176
|
+
funding_tx = sender.internal_wallet.broadcast(funding_tx) if funding_tx
|
177
|
+
|
178
|
+
tx = create_transfer_tx(funding_tx: funding_tx, color_id: color_id, sender: sender, receiver_address: receiver_address, amount: amount)
|
179
|
+
sender.internal_wallet.broadcast(tx)
|
180
|
+
[color_id, tx]
|
181
|
+
end
|
182
|
+
|
183
|
+
# Burn token
|
184
|
+
# If amount is not specified or 0, burn all token associated with the wallet.
|
185
|
+
#
|
186
|
+
# @param sender [Glueby::Wallet] wallet to send this token
|
187
|
+
# @param amount [Integer]
|
188
|
+
# @raise [InsufficientFunds] if wallet does not have enough TPC to send transaction.
|
189
|
+
# @raise [InsufficientTokens] if wallet does not have enough token to send transaction.
|
190
|
+
# @raise [InvalidAmount] if amount is not positive integer.
|
191
|
+
def burn!(sender:, amount: 0)
|
192
|
+
raise Glueby::Contract::Errors::InvalidAmount unless amount.positive?
|
193
|
+
|
194
|
+
utxo_provider = Glueby::UtxoProvider.new if Glueby.configuration.use_utxo_provider?
|
195
|
+
funding_tx = create_funding_tx(wallet: sender, utxo_provider: utxo_provider) if utxo_provider
|
196
|
+
funding_tx = sender.internal_wallet.broadcast(funding_tx) if funding_tx
|
197
|
+
|
198
|
+
tx = create_burn_tx(funding_tx: funding_tx, color_id: color_id, sender: sender, amount: amount)
|
199
|
+
sender.internal_wallet.broadcast(tx)
|
200
|
+
end
|
201
|
+
|
202
|
+
# Return balance of token in the specified wallet.
|
203
|
+
# @param wallet [Glueby::Wallet]
|
204
|
+
# @return [Integer] amount of utxo value associated with this token.
|
205
|
+
def amount(wallet:)
|
206
|
+
# collect utxo associated with this address
|
207
|
+
utxos = wallet.internal_wallet.list_unspent
|
208
|
+
_, results = collect_colored_outputs(utxos, color_id)
|
209
|
+
results.sum { |result| result[:amount] }
|
210
|
+
end
|
211
|
+
|
212
|
+
# Return token type
|
213
|
+
# @return [Tapyrus::Color::TokenTypes]
|
214
|
+
def token_type
|
215
|
+
color_id.type
|
216
|
+
end
|
217
|
+
|
218
|
+
# Return the script_pubkey of the token from ActiveRecord
|
219
|
+
# @return [String] script_pubkey
|
220
|
+
def script_pubkey
|
221
|
+
@script_pubkey ||= Glueby::Contract::AR::ReissuableToken.script_pubkey(@color_id.to_hex)
|
222
|
+
end
|
223
|
+
|
224
|
+
# Return serialized payload
|
225
|
+
# @return [String] payload
|
226
|
+
def to_payload
|
227
|
+
payload = +''
|
228
|
+
payload << @color_id.to_payload
|
229
|
+
payload << @script_pubkey.to_payload if script_pubkey
|
230
|
+
payload
|
231
|
+
end
|
232
|
+
|
233
|
+
# Restore token from payload
|
234
|
+
# @param payload [String]
|
235
|
+
# @return [Glueby::Contract::Token]
|
236
|
+
def self.parse_from_payload(payload)
|
237
|
+
color_id, script_pubkey = payload.unpack('a33a*')
|
238
|
+
color_id = Tapyrus::Color::ColorIdentifier.parse_from_payload(color_id)
|
239
|
+
if color_id.type == Tapyrus::Color::TokenTypes::REISSUABLE
|
240
|
+
raise ArgumentError, 'script_pubkey should not be empty' if script_pubkey.empty?
|
241
|
+
script_pubkey = Tapyrus::Script.parse_from_payload(script_pubkey)
|
242
|
+
Glueby::Contract::AR::ReissuableToken.create!(color_id: color_id.to_hex, script_pubkey: script_pubkey.to_hex)
|
243
|
+
end
|
244
|
+
new(color_id: color_id)
|
245
|
+
end
|
246
|
+
|
247
|
+
# Generate Token Instance
|
248
|
+
# @param color_id [String]
|
249
|
+
def initialize(color_id:)
|
250
|
+
@color_id = color_id
|
251
|
+
end
|
252
|
+
|
253
|
+
private
|
254
|
+
|
255
|
+
# Verify that wallet is the issuer of the reissuable token
|
256
|
+
# reutrn [Boolean]
|
257
|
+
def validate_reissuer(wallet:)
|
258
|
+
addresses = wallet.internal_wallet.get_addresses
|
259
|
+
addresses.each do |address|
|
260
|
+
decoded_address = Tapyrus.decode_base58_address(address)
|
261
|
+
pubkey_hash_from_address = decoded_address[0]
|
262
|
+
pubkey_hash_from_script = Tapyrus::Script.parse_from_payload(script_pubkey.chunks[2])
|
263
|
+
if pubkey_hash_from_address == pubkey_hash_from_script.to_s
|
264
|
+
return true
|
265
|
+
end
|
266
|
+
end
|
267
|
+
false
|
268
|
+
end
|
269
|
+
end
|
270
|
+
end
|
245
271
|
end
|