Created
January 9, 2013 09:04
-
-
Save sashazykov/4491729 to your computer and use it in GitHub Desktop.
Revisions
-
sashazykov created this gist
Jan 9, 2013 .There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode charactersOriginal file line number Diff line number Diff line change @@ -0,0 +1,55 @@ require 'digest' class BitcoinAddressValidator < ActiveModel::EachValidator def validate_each(record, field, value) unless value.blank? || valid_bitcoin_address?(value) record.errors[field] << "Bitcoin address is invalid" end end private B58Chars = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz' B58Base = B58Chars.length def valid_bitcoin_address?(address) (address =~ /^[a-zA-Z1-9]{33,35}$/) and version(address) end def version(address) decoded = b58_decode(address, 25) version = decoded[0, 1] checksum = decoded[-4, decoded.length] vh160 = decoded[0, decoded.length - 4] hashed = (Digest::SHA2.new << (Digest::SHA2.new << vh160).digest).digest hashed[0, 4] == checksum ? version[0] : nil end def b58_decode(value, length) long_value = 0 index = 0 result = "" value.reverse.each_char do |c| long_value += B58Chars.index(c) * (B58Base ** index) index += 1 end while long_value >= 256 do div, mod = long_value.divmod 256 result = mod.chr + result long_value = div end result = long_value.chr + result if result.length < length result = 0.chr * (length - result.length) + result end result end end