Class: IParty::MaxMind::Database
- Inherits:
-
Object
- Object
- IParty::MaxMind::Database
- Defined in:
- lib/iparty/max_mind/database.rb
Defined Under Namespace
Classes: Error, InvalidFileFormatError
Constant Summary collapse
- METADATA_BEGIN_MARKER =
"#{[0xAB, 0xCD, 0xEF].pack("C*")}MaxMind.com".encode("ascii-8bit", "ascii-8bit").freeze
- DATA_SECTION_SEPARATOR_SIZE =
16- SIZE_BASE_VALUES =
[0, 29, 285, 65_821].freeze
- POINTER_BASE_VALUES =
[0, 0, 2048, 526_336].freeze
Instance Attribute Summary collapse
-
#metadata ⇒ Object
readonly
Returns the value of attribute metadata.
Instance Method Summary collapse
- #cidr_from_long(long, net) ⇒ Object
- #close ⇒ Object
- #closed? ⇒ Boolean
-
#decode(base_pos, pos) ⇒ Object
rubocop:disable Metrics/CyclomaticComplexity – we could reduce this, sacrificing the neat overview.
- #decode_array(base_pos, pos, size) ⇒ Object
- #decode_bytes(base_pos, pos, size) ⇒ Object
-
#decode_double(base_pos, pos, size) ⇒ Object
rubocop:enable Metrics/CyclomaticComplexity.
- #decode_float(base_pos, pos, size) ⇒ Object
- #decode_int(base_pos, pos, size) ⇒ Object
- #decode_map(base_pos, pos, size) ⇒ Object
- #decode_pointer(base_pos, pos, ctrl) ⇒ Object
- #decode_uint(base_pos, pos, size) ⇒ Object
- #decode_utf8(base_pos, pos, size) ⇒ Object
-
#initialize(path, reader: EagerReader) ⇒ Database
constructor
A new instance of Database.
- #inspect ⇒ Object
- #lookup(addr, result_class: Result::Geo) ⇒ Object
- #read_record(node_no, flag) ⇒ Object
- #read_value(base_pos, pos, size) ⇒ Object
Constructor Details
#initialize(path, reader: EagerReader) ⇒ Database
Returns a new instance of Database.
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
# File 'lib/iparty/max_mind/database.rb', line 20 def initialize(path, reader: EagerReader) @path = path @data = reader.new(path) pos = @data.rindex(METADATA_BEGIN_MARKER) || raise(InvalidFileFormatError, "invalid file format") pos += METADATA_BEGIN_MARKER.size @metadata = decode(0, pos)[1] @ip_version = @metadata[:ip_version] @start_idx = @ip_version == 4 ? 96 : 0 @node_count = @metadata[:node_count] @node_byte_size = @metadata[:record_size] * 2 / 8 @search_tree_size = @node_count * @node_byte_size @data_section_start = @search_tree_size + DATA_SECTION_SEPARATOR_SIZE end |
Instance Attribute Details
#metadata ⇒ Object (readonly)
Returns the value of attribute metadata.
18 19 20 |
# File 'lib/iparty/max_mind/database.rb', line 18 def @metadata end |
Instance Method Details
#cidr_from_long(long, net) ⇒ Object
75 76 77 78 79 80 81 |
# File 'lib/iparty/max_mind/database.rb', line 75 def cidr_from_long(long, net) addr = IPAddr.new(long, long > (2**32) - 1 ? Socket::AF_INET6 : Socket::AF_INET) subnet_size = addr.ipv4? ? net - 96 + 1 : net + 1 subnet = IPAddr.new("#{addr}/#{subnet_size}") "#{subnet}/#{subnet_size}" end |
#close ⇒ Object
36 37 38 |
# File 'lib/iparty/max_mind/database.rb', line 36 def close @data.close end |
#closed? ⇒ Boolean
40 41 42 |
# File 'lib/iparty/max_mind/database.rb', line 40 def closed? @data.closed? end |
#decode(base_pos, pos) ⇒ Object
rubocop:disable Metrics/CyclomaticComplexity – we could reduce this, sacrificing the neat overview
105 106 107 108 109 110 111 112 113 114 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 |
# File 'lib/iparty/max_mind/database.rb', line 105 def decode base_pos, pos ctrl = @data[base_pos + pos].ord pos += 1 type = ctrl >> 5 if type == 1 # pointer decode_pointer(base_pos, pos, ctrl) else if type == 0 # extended type type = 7 + @data[base_pos + pos].ord pos += 1 end size = ctrl & 0x1f if size >= 29 byte_size = size - 29 + 1 val = read_value(base_pos, pos, byte_size) pos += byte_size size = val + SIZE_BASE_VALUES[byte_size] end # rubocop:disable Lint/DuplicateBranch -- readable order is more important case type when 2 # utf8 decode_utf8(base_pos, pos, size) when 3 # double decode_double(base_pos, pos, size) when 4 # bytes decode_bytes(base_pos, pos, size) when 5 # unsigned 16-bit int decode_uint(base_pos, pos, size) when 6 # unsigned 32-bit int decode_uint(base_pos, pos, size) when 7 # map decode_map(base_pos, pos, size) when 8 # signed 32-bit int decode_int(base_pos, pos, size) when 9 # unsigned 64-bit int decode_uint(base_pos, pos, size) when 10 # unsigned 128-bit int decode_uint(base_pos, pos, size) when 11 # array decode_array(base_pos, pos, size) when 12 # (deprecated) data cache container raise "TODO: (deprecated) data cache container format" when 13 # (deprecated) end marker [pos, nil] when 14 # boolean [pos, size != 0] when 15 # float decode_float(base_pos, pos, size) end # rubocop:enable Lint/DuplicateBranch end end |
#decode_array(base_pos, pos, size) ⇒ Object
197 198 199 200 201 202 203 |
# File 'lib/iparty/max_mind/database.rb', line 197 def decode_array base_pos, pos, size ary = Array.new(size) do pos, v = decode(base_pos, pos) v end [pos, ary] end |
#decode_bytes(base_pos, pos, size) ⇒ Object
166 167 168 |
# File 'lib/iparty/max_mind/database.rb', line 166 def decode_bytes base_pos, pos, size [pos + size, @data[base_pos + pos, size]] end |
#decode_double(base_pos, pos, size) ⇒ Object
rubocop:enable Metrics/CyclomaticComplexity
162 163 164 |
# File 'lib/iparty/max_mind/database.rb', line 162 def decode_double base_pos, pos, size [pos + size, @data[base_pos + pos, size].unpack1("G")] end |
#decode_float(base_pos, pos, size) ⇒ Object
180 181 182 |
# File 'lib/iparty/max_mind/database.rb', line 180 def decode_float base_pos, pos, size [pos + size, @data[base_pos + pos, size].unpack1("g")] end |
#decode_int(base_pos, pos, size) ⇒ Object
170 171 172 173 174 |
# File 'lib/iparty/max_mind/database.rb', line 170 def decode_int base_pos, pos, size v1 = @data[base_pos + pos, size].unpack1("N") bits = size * 8 [pos + size, (v1 & ~(1 << bits)) - (v1 & (1 << bits))] end |
#decode_map(base_pos, pos, size) ⇒ Object
205 206 207 208 209 210 211 212 213 |
# File 'lib/iparty/max_mind/database.rb', line 205 def decode_map base_pos, pos, size map = {} size.times do pos, k = decode(base_pos, pos) pos, v = decode(base_pos, pos) map[k.to_sym] = v end [pos, map] end |
#decode_pointer(base_pos, pos, ctrl) ⇒ Object
188 189 190 191 192 193 194 195 |
# File 'lib/iparty/max_mind/database.rb', line 188 def decode_pointer base_pos, pos, ctrl size = ((ctrl >> 3) & 0x3) + 1 v1 = ctrl & 0x7 v2 = read_value(base_pos, pos, size) pointer = (v1 << (8 * size)) + v2 + POINTER_BASE_VALUES[size] [pos + size, decode(base_pos, pointer)[1]] end |
#decode_uint(base_pos, pos, size) ⇒ Object
176 177 178 |
# File 'lib/iparty/max_mind/database.rb', line 176 def decode_uint base_pos, pos, size [pos + size, read_value(base_pos, pos, size)] end |
#decode_utf8(base_pos, pos, size) ⇒ Object
184 185 186 |
# File 'lib/iparty/max_mind/database.rb', line 184 def decode_utf8 base_pos, pos, size [pos + size, @data[base_pos + pos, size].encode("utf-8", "utf-8")] end |
#inspect ⇒ Object
44 45 46 |
# File 'lib/iparty/max_mind/database.rb', line 44 def inspect "#<#{self.class}:#{format("0x%x", object_id << 1)}: @path:#{@path} @metadata:#{@metadata}>" end |
#lookup(addr, result_class: Result::Geo) ⇒ Object
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 |
# File 'lib/iparty/max_mind/database.rb', line 48 def lookup(addr, result_class: Result::Geo) addr = IPAddr.new(addr) unless addr.is_a?(IPAddr) addr = IParty.config.local_ip_alias if IParty.config.local_ip_alias && addr.loopback? addr = IPAddr.new(addr) unless addr.is_a?(IPAddr) addr = addr.ipv4_compat if addr.ipv4? long = addr.is_a?(IParty::Address) ? addr.to_i(significant: true) : addr.to_i node_no = 0 (@start_idx...128).each do |i| flag = (long >> (127 - i)) & 1 next_node_no = read_record(node_no, flag) if next_node_no == 0 raise(InvalidFileFormatError, "invalid file format") elsif next_node_no >= @node_count pos = (next_node_no - @node_count) - DATA_SECTION_SEPARATOR_SIZE result = decode(@data_section_start, pos)[1] result[:network] = cidr_from_long(long, i) unless result.empty? return result_class.new(result) else node_no = next_node_no end end raise(InvalidFileFormatError, "invalid file format") end |
#read_record(node_no, flag) ⇒ Object
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 |
# File 'lib/iparty/max_mind/database.rb', line 83 def read_record(node_no, flag) rec_byte_size = @node_byte_size / 2 pos = @node_byte_size * node_no middle = @data[pos + rec_byte_size].ord if @node_byte_size.odd? if flag == 0 # left val = read_value(pos, 0, rec_byte_size) val += ((middle & 0xf0) << 20) if middle else # right val = read_value(pos + @node_byte_size - rec_byte_size, 0, rec_byte_size) val += ((middle & 0xf) << 24) if middle end val end |
#read_value(base_pos, pos, size) ⇒ Object
99 100 101 102 |
# File 'lib/iparty/max_mind/database.rb', line 99 def read_value(base_pos, pos, size) bytes = @data[base_pos + pos, size].unpack("C*") bytes.inject(0){|r, v| (r << 8) + v } end |