lib/packetman/filter.rb
module Packetman
class Filter
include ConfigMethods
attr_accessor :input
def initialize(input)
self.input = input
yield config if block_given?
end
def self.bit_length(num)
case num
when /^0x/
$'.length * bit_density(16)
else
nil
end
end
def self.bit_density(radix=config.radix)
(radix.nil?) ? 8 : Math.log2(radix).ceil
end
def map_chr
shift_and_pad(input.scan(/./).map{ |chr| yield chr }.join)
end
def shift_and_pad(bin_str)
#shift
bin_str.ljust(target_bit_length, '0').
#pad
rjust(target_bit_length + config.offset_bits % 8, '0')
end
def target_bit_length
((input.length*self.class.bit_density + config.offset_bits)/8.to_f).ceil*8 - config.offset_bits
end
# Mask for 1 character of current radix
def radix_mask
("1"*self.class.bit_density).to_i(2)
end
# Mask string for _chr_ substituting wildcards as necessary
def mask_chr(chr)
if chr == config.wildcard
0
else
radix_mask
end.to_s(2).rjust(self.class.bit_density, '0')
end
# Converts the `chr` from `config.radix` to binary, substituting wildcards as necessary
#
# @param chr [String] character to convert to binary
# @return [String] binary string
def bin_chr(chr)
if config.radix
chr = '0' if chr == config.wildcard
raise "invalid character '#{chr}' for radix=#{config.radix}" if chr.downcase != chr.to_i(config.radix).to_s(config.radix).downcase
chr.to_i(config.radix)
else
if chr == config.wildcard
0
else
chr.ord
end
end.to_s(2).rjust(self.class.bit_density, '0')
end
def mask_hex
hex_encode(map_chr{ |c| mask_chr(c) })
end
def search_hex
hex_encode(map_chr{ |c| bin_chr(c) })
end
# Transform _bin_str_ to array of 32, 16, and 8 bit hex encoded strings
def hex_encode(bin_str)
bin_str.reverse.scan(/.{1,4}/).map{ |chunk|
chunk.reverse.to_i(2).to_s(16)
}.reverse.join.scan(/.{8}|.{4}|.{2}/).map{ |hex|
hex.prepend('0x')
}
end
def clauses
start_bit = 0
[].tap do |filter|
search_hex.zip(mask_hex).each do |search, mask|
filter << Packetman::Clause.new(search, mask, start_bit)
start_bit += self.class.bit_length(search)
end
end
end
def to_s
clauses.map{ |clause| clause.to_s }.join(' && ')
end
end
end