lib/eth/tx.rb
module Eth class Tx include RLP::Sedes::Serializable extend Sedes set_serializable_fields({ nonce: big_endian_int, gas_price: big_endian_int, gas_limit: big_endian_int, to: address, value: big_endian_int, data_bin: binary, v: big_endian_int, r: big_endian_int, s: big_endian_int }) attr_writer :signature def self.decode(data) data = Utils.hex_to_bin(data) if data.match(/\A(?:0x)?\h+\Z/) deserialize(RLP.decode data) end def initialize(params) fields = {v: 0, r: 0, s: 0}.merge params fields[:to] = Utils.normalize_address(fields[:to]) if params[:data] self.data = params.delete(:data) fields[:data_bin] = data_bin end serializable_initialize fields check_transaction_validity end def unsigned_encoded RLP.encode(unsigned, sedes: sedes) end def signing_data Utils.bin_to_prefixed_hex unsigned_encoded end def encoded RLP.encode self end def hex Utils.bin_to_prefixed_hex encoded end def sign(key) self.signature = key.sign(unsigned_encoded) vrs = Utils.v_r_s_for signature self.v = vrs[0] self.r = vrs[1] self.s = vrs[2] self end def to_h hash_keys.inject({}) do |hash, field| hash[field] = send field hash end end def from if signature public_key = OpenSsl.recover_compact(signature_hash, signature) Utils.public_key_to_address(public_key) if public_key end end def signature return @signature if @signature self.signature = [ Utils.int_to_base256(v), Utils.zpad_int(r), Utils.zpad_int(s), ].join if [v, r, s].all? end def hash "0x#{Utils.bin_to_hex Utils.keccak256_rlp(self)}" end alias_method :id, :hash def data_hex Utils.bin_to_prefixed_hex data_bin end def data_hex=(hex) self.data_bin = Utils.hex_to_bin(hex) end def data Eth.tx_data_hex? ? data_hex : data_bin end def data=(string) Eth.tx_data_hex? ? self.data_hex=(string) : self.data_bin=(string) end private def hash_keys keys = self.class.serializable_fields.keys keys.delete(:data_bin) keys + [:data] end def check_transaction_validity if [gas_price, gas_limit, value, nonce].max > UINT_MAX raise InvalidTransaction, "Values way too high!" elsif gas_limit < intrinsic_gas_used raise InvalidTransaction, "Gas limit too low" end end def intrinsic_gas_used num_zero_bytes = data_bin.count(BYTE_ZERO) num_non_zero_bytes = data_bin.size - num_zero_bytes Gas::GTXCOST + Gas::GTXDATAZERO * num_zero_bytes + Gas::GTXDATANONZERO * num_non_zero_bytes end def signature_hash Utils.keccak256 unsigned_encoded end def unsigned Tx.new to_h.merge(v: Eth.chain_id, r: 0, s: 0) end def sedes if Eth.prevent_replays? && !(Eth.replayable_v? v) self.class else UnsignedTx end end end UnsignedTx = Tx.exclude([:v, :r, :s]) end