fort/bin/lua/wipf/util/ip.lua

132 lines
3.0 KiB
Lua
Raw Normal View History

2015-01-19 06:40:38 +00:00
-- WIPF IP Utilities
2015-02-01 08:34:35 +00:00
local ffi = require "ffi"
2015-01-19 12:24:15 +00:00
local bit = require"bit"
2015-01-19 06:40:38 +00:00
local sys = require"sys"
local sock = require"sys.sock"
2015-01-26 04:55:17 +00:00
local util_ip = {}
-- Convert IPv4 ranges
2015-01-19 12:24:15 +00:00
do
2015-02-01 08:34:35 +00:00
local function to_uint32(num)
return tonumber(ffi.cast("uint32_t", num))
end
-- sort and try to merge ranges
local function iprange_map_merge(map)
-- fill temporary "from" range
local tmp_from, tmp_count = {}, 0
for from in pairs(map) do
tmp_count = tmp_count + 1
tmp_from[tmp_count] = from
end
table.sort(tmp_from)
-- try to merge ranges
local prev_from, prev_to
for i = 1, tmp_count do
local from = tmp_from[i]
local to = map[from]
2015-02-01 08:34:35 +00:00
if prev_from and from <= prev_to then -- collides with previous?
if to > prev_to then
map[prev_from], prev_to = to, to
end
tmp_from[i] = nil
else
prev_from, prev_to = from, to
end
end
-- fill "from" & "to" ranges
local iprange_from, iprange_to, count = {}, {}, 0
for i = 1, tmp_count do
local from = tmp_from[i]
if from then
count = count + 1
iprange_from[count] = from
iprange_to[count] = map[from]
end
end
iprange_from.n, iprange_to.n = count, count
return iprange_from, iprange_to
end
2015-01-25 16:29:02 +00:00
local function parse_address_mask(line)
local from, sep, mask = string.match(line, "([%d%.]+)%s*([/%-])%s*(%S+)")
if not from then
return
end
local from_ip = sock.inet_pton(from, true)
if not from_ip then
return
end
local to_ip
if sep == '-' then -- e.g. "127.0.0.0-127.255.255.255"
to_ip = sock.inet_pton(mask, true)
if not to_ip or from_ip > to_ip then
return
end
elseif sep == '/' then -- e.g. "127.0.0.0/24"
local nbits = tonumber(mask)
if nbits > 32 or nbits < 0 then
return
elseif nbits == 32 then
to_ip = 0xFFFFFFFF
else
to_ip = bit.bor(from_ip, bit.lshift(1, nbits) - 1)
2015-02-01 08:34:35 +00:00
to_ip = to_uint32(to_ip)
2015-01-25 16:29:02 +00:00
end
end
return from_ip, to_ip
end
2015-01-26 04:55:17 +00:00
-- Convert IPv4 ranges in text to 'from_ip4' & 'to_ip4' arrays with numbers
function util_ip.ip4range_to_numbers(text)
local iprange_map = {}
local line_no = 0
2015-01-19 12:24:15 +00:00
2015-01-20 10:16:56 +00:00
for line in string.gmatch(text, "%s*([^\n]+)") do
local from, to = parse_address_mask(line)
2015-01-19 12:24:15 +00:00
line_no = line_no + 1
2015-01-19 12:24:15 +00:00
if from then
iprange_map[from] = to
elseif string.find(line, "%S") then
return nil, line_no
2015-01-19 12:24:15 +00:00
end
end
return iprange_map_merge(iprange_map)
2015-01-19 12:24:15 +00:00
end
2015-01-19 06:40:38 +00:00
end
2015-02-01 08:34:35 +00:00
-- Convert IPv4 ranges to text from 'from_ip4' & 'to_ip4' arrays with numbers
function util_ip.ip4range_from_numbers(iprange_from, iprange_to)
local text = ""
local n = iprange_from.n
for i = 1, n do
local from_ip, to_ip = iprange_from[i], iprange_to[i]
local from = sock.inet_ntop(from_ip)
local to = sock.inet_ntop(to_ip)
text = text .. from .. "-" .. to .. "\n"
end
return text
end
2015-01-19 06:40:38 +00:00
2015-01-26 04:55:17 +00:00
return util_ip