mirror of
https://github.com/ntop/ntopng.git
synced 2026-05-03 17:30:11 +00:00
452 lines
14 KiB
Lua
452 lines
14 KiB
Lua
--
|
|
-- (C) 2013-26 - ntop.org
|
|
--
|
|
-- Required modules for exporter site management
|
|
require "ntop_utils"
|
|
local json = require("dkjson")
|
|
|
|
-- Module definition - this module provides utilities for managing exporter sites
|
|
-- Exporter sites represent physical or logical locations of network flow exporters
|
|
local exporter_site_utils = {}
|
|
|
|
-- Redis cache keys configuration for persistent storage
|
|
local REDIS_HASH_NAME = "ntopng.prefs.exporter_sites" -- Stores all exporter sites as hash: id -> JSON
|
|
local REDIS_COUNTER_KEY = "ntopng.prefs.exporter_sites_counter" -- Auto-increment counter for site IDs
|
|
local flow_dev_exporter_sites_key = "ntopng.prefs.flow_dev_exporter_sites" -- Maps flow device IPs to site IDs
|
|
|
|
-- Configuration limits for exporter sites
|
|
local MAX_DESCRIPTION_SIZE = 256 -- Maximum character length for site descriptions
|
|
local MAX_PROFILES_NUM = 1024 -- Maximum number of exporter sites allowed in the system
|
|
|
|
-- Default site configuration - system reserved site used when no site is assigned
|
|
-- This site cannot be modified or deleted and serves as a fallback
|
|
local DEFAULT_SITE = {
|
|
id = "0", -- System reserved ID (always string "0")
|
|
name = "Default", -- Display name
|
|
description = "", -- Optional description
|
|
longitude = 0, -- Geographic coordinates (0,0 by default)
|
|
latitude = 0,
|
|
reserved = true -- Flag indicating this is a system-reserved site
|
|
}
|
|
|
|
-- ##############################################
|
|
-- Private Helper Functions
|
|
-- ##############################################
|
|
|
|
local iface_to_exporter = nil
|
|
local exporter_to_site = nil
|
|
local exporter_to_name = nil
|
|
|
|
--
|
|
-- Caches exporters information in memory
|
|
--
|
|
local function cache_exporters()
|
|
if((iface_to_exporter == nil) and ntop.isPro()) then
|
|
package.path = dirs.installdir .. "/pro/scripts/lua/modules/?.lua;" .. package.path
|
|
local snmp_cached_dev = require "snmp_cached_dev"
|
|
|
|
iface_to_exporter = {}
|
|
exporter_to_site = {}
|
|
exporter_to_name = {}
|
|
|
|
local ifstats = interface.getStats()
|
|
|
|
for interface_id, probe_list in pairs(ifstats.probes or {}) do
|
|
for probe_ip, probe_info in pairsByKeys(probe_list or {}) do
|
|
for exporter_ip, exporter_info in pairsByKeys(probe_info.exporters or {}) do
|
|
local ifaces = snmp_cached_dev:get_interfaces(exporter_ip)
|
|
local system = snmp_cached_dev:get_system(exporter_ip)
|
|
local ret = exporter_site_utils.getFlowDevExporterSite(exporter_ip)
|
|
local site_name = ret.name
|
|
|
|
for _,v in pairs(ifaces.interfaces) do
|
|
if(v.ip_addr ~= nil) then
|
|
for _,iface_ip in pairs(v.ip_addr) do
|
|
iface_to_exporter[iface_ip] = exporter_ip
|
|
end
|
|
end
|
|
end
|
|
|
|
exporter_to_site[exporter_ip] = site_name
|
|
exporter_to_name[exporter_ip] = system.system.name or exporter_ip
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
-- Validates all parameters for an exporter site before creation or modification
|
|
-- This comprehensive validation ensures data integrity and prevents duplicates
|
|
local function validate_site(name, description, latitude, longitude, existing_sites, ignore_name_duplication)
|
|
-- Step 1: Validate site name
|
|
if type(name) ~= "string" then
|
|
return false, "Invalid name"
|
|
end
|
|
|
|
-- Check name length constraints (1-16 characters)
|
|
if #name == 0 or #name > 16 then
|
|
return false, "Invalid name, max characters: 16"
|
|
end
|
|
|
|
-- Validate name format (alphanumeric only)
|
|
if not name:match("^[%w À-ÖØ-öø-ÿ]+$") then
|
|
return false, "Invalid name, illegal character"
|
|
end
|
|
|
|
-- Convert to lowercase for case-insensitive duplicate checking
|
|
local name_lower = name:lower()
|
|
|
|
-- Step 2: Validate description
|
|
if type(description) ~= "string" then
|
|
return false, "Invalid description"
|
|
end
|
|
|
|
-- Check description length limit
|
|
if #description > MAX_DESCRIPTION_SIZE then
|
|
return false, "Invalid description, max characters: 256"
|
|
end
|
|
|
|
-- Step 3: Validate geographic coordinates
|
|
if not tonumber(latitude) or not tonumber(longitude) then
|
|
return false, "Invalid coordinates"
|
|
end
|
|
|
|
-- Convert to numbers for range validation
|
|
latitude = tonumber(latitude)
|
|
longitude = tonumber(longitude)
|
|
|
|
-- Validate latitude range (-90 to 90 degrees)
|
|
if latitude < -90 or latitude > 90 then
|
|
return false, "Invalid latitude"
|
|
end
|
|
|
|
-- Validate longitude range (-180 to 180 degrees)
|
|
if longitude < -180 or longitude > 180 then
|
|
return false, "Invalid longitude"
|
|
end
|
|
|
|
-- Step 4: Check for duplicate site names (unless explicitly disabled for edits)
|
|
if not ignore_name_duplication then
|
|
for _, site in pairs(existing_sites) do
|
|
if site.name:lower() == name_lower then
|
|
return false, "Site " .. name .. " already exists"
|
|
end
|
|
end
|
|
end
|
|
|
|
-- All validation passed
|
|
return true
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Retrieves all exporter sites from Redis cache and prepares them for use
|
|
-- This function always includes the default site and merges it with user-defined sites
|
|
local function get_sites_from_cache()
|
|
local sites_list = {}
|
|
|
|
-- Always include the default site as ID "0"
|
|
sites_list["0"] = exporter_site_utils.get_default_site()
|
|
|
|
-- Retrieve all user-defined sites from Redis
|
|
local current_defined_sites = ntop.getHashAllCache(REDIS_HASH_NAME) or {}
|
|
|
|
-- Process each site JSON string from Redis
|
|
for _, site in pairs(current_defined_sites) do
|
|
-- Decode JSON string to Lua table
|
|
local uncompressed_json = json.decode(site) or nil
|
|
if uncompressed_json then
|
|
-- Store site using its string ID as key for easy lookup
|
|
sites_list[tostring(uncompressed_json.id)] = uncompressed_json
|
|
end
|
|
end
|
|
|
|
return sites_list
|
|
end
|
|
|
|
-- ##############################################
|
|
-- Public API Functions
|
|
-- ##############################################
|
|
|
|
-- Returns the system default exporter site
|
|
-- Used as fallback when no site is assigned to a flow device
|
|
function exporter_site_utils.get_default_site()
|
|
return DEFAULT_SITE
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Associates a flow device (exporter) with a specific site
|
|
-- Creates or updates the mapping in Redis, or removes it if site_id is invalid
|
|
function exporter_site_utils.setFlowDevExporterSite(flowdev_ip, exporter_site_id)
|
|
if (exporter_site_id) and tonumber(exporter_site_id) then
|
|
-- Store the association: flow device IP -> site ID (as string)
|
|
ntop.setHashCache(flow_dev_exporter_sites_key, flowdev_ip, tostring(exporter_site_id))
|
|
else
|
|
-- Remove association if site_id is nil or invalid
|
|
ntop.delHashCache(flow_dev_exporter_sites_key, flowdev_ip)
|
|
end
|
|
|
|
-- Refresh the cached site ID in C++ for all interfaces
|
|
local old_ifid = interface.getId()
|
|
for _, ifname in pairs(interface.getIfNames() or {}) do
|
|
interface.select(ifname)
|
|
interface.refreshFlowDeviceSiteId(flowdev_ip)
|
|
end
|
|
interface.select(old_ifid)
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Retrieves the exporter site associated with a specific flow device
|
|
-- Returns the default site if no association exists or if the association is invalid
|
|
function exporter_site_utils.getFlowDevExporterSite(flowdev_ip)
|
|
-- Look up site ID for this flow device from Redis
|
|
local exporter_site_id = ntop.getHashCache(flow_dev_exporter_sites_key, flowdev_ip)
|
|
|
|
-- Get all available sites
|
|
local exporter_sites = get_sites_from_cache()
|
|
|
|
-- Check if we have a valid site ID for this device
|
|
if not isEmptyString(exporter_site_id) then
|
|
-- Look up the site by its ID
|
|
local site = exporter_sites[tostring(exporter_site_id)]
|
|
if site then
|
|
return site -- Return the associated site
|
|
end
|
|
end
|
|
|
|
-- Fallback: return default site if no valid association exists
|
|
return exporter_site_utils.get_default_site()
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Returns all exporter sites as a sorted array for display purposes
|
|
-- Sites are sorted by ID in ascending order, with default site always included
|
|
function exporter_site_utils.getExporterSites()
|
|
local exporter_sites = get_sites_from_cache()
|
|
|
|
local result = {}
|
|
|
|
-- Iterate through sites sorted by ID (ascending order)
|
|
for id, site in pairsByKeys(exporter_sites, asc) do
|
|
local record = {}
|
|
record["id"] = tostring(site.id) -- Ensure ID is string
|
|
record["name"] = site.name -- Site display name
|
|
record["description"] = site.description -- Optional description
|
|
record["latitude"] = site.latitude -- Geographic coordinates
|
|
record["longitude"] = site.longitude
|
|
record["reserved"] = site.reserved -- System-reserved flag
|
|
|
|
-- Add to result array
|
|
result[#result + 1] = record
|
|
end
|
|
|
|
return result
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Edits an existing exporter site with new parameters
|
|
-- Performs validation and updates the site in Redis storage
|
|
function exporter_site_utils.editExporterSite(id, name, description, latitude, longitude)
|
|
-- Get current sites for validation
|
|
local existing_sites = get_sites_from_cache()
|
|
|
|
-- Validate and normalize the site ID
|
|
if id and tonumber(id) then
|
|
id = tostring(id) -- Convert to string for consistency
|
|
else
|
|
return false, "Invalid ID"
|
|
end
|
|
|
|
-- Ensure the site exists
|
|
if not existing_sites[id] then
|
|
return false, "Invalid Site"
|
|
end
|
|
|
|
local old_site = existing_sites[id]
|
|
|
|
-- Handle empty coordinate values (default to 0)
|
|
if isEmptyString(latitude) then
|
|
latitude = 0
|
|
end
|
|
if isEmptyString(longitude) then
|
|
longitude = 0
|
|
end
|
|
|
|
-- Skip duplicate name check if the name hasn't changed (edit vs rename scenario)
|
|
local ignore_name_duplication = old_site.name == name
|
|
|
|
-- Validate all input parameters
|
|
local res, msg = validate_site(name, description, latitude, longitude, existing_sites, ignore_name_duplication)
|
|
|
|
if res then
|
|
-- Delete old entry first to ensure clean update
|
|
ntop.delHashCache(REDIS_HASH_NAME, id)
|
|
|
|
-- Create updated site object
|
|
local site_json = {
|
|
id = tostring(id),
|
|
name = name,
|
|
description = description,
|
|
latitude = latitude,
|
|
longitude = longitude
|
|
}
|
|
|
|
-- Store updated site in Redis
|
|
ntop.setHashCache(REDIS_HASH_NAME, id, json.encode(site_json))
|
|
else
|
|
return res, msg -- Return validation error
|
|
end
|
|
|
|
local success_msg = "Site edited successfully"
|
|
return true, success_msg
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Creates a new exporter site with auto-generated ID
|
|
-- Validates input, checks system limits, and stores in Redis
|
|
function exporter_site_utils.addExporterSite(name, description, latitude, longitude)
|
|
-- Get current site counter from Redis (or default to 1)
|
|
local current_count = tonumber(ntop.getCache(REDIS_COUNTER_KEY)) or 1
|
|
|
|
-- Check system limit before proceeding
|
|
if current_count + 1 > MAX_PROFILES_NUM then
|
|
return false, "Adding a site would exceed maximum limit (" .. MAX_PROFILES_NUM .. "). Current: " .. current_count
|
|
end
|
|
|
|
-- Get existing sites for validation
|
|
local existing_sites = get_sites_from_cache()
|
|
|
|
-- Handle empty coordinate values
|
|
if isEmptyString(latitude) then
|
|
latitude = 0
|
|
end
|
|
if isEmptyString(longitude) then
|
|
longitude = 0
|
|
end
|
|
|
|
-- Validate all input parameters
|
|
local res, msg = validate_site(name, description, latitude, longitude, existing_sites, false)
|
|
|
|
if res then
|
|
-- Generate new site ID (use current counter value)
|
|
local site_id = tostring(current_count)
|
|
|
|
-- Create site object
|
|
local site_json = {
|
|
id = site_id,
|
|
name = name,
|
|
description = description,
|
|
latitude = latitude,
|
|
longitude = longitude
|
|
}
|
|
|
|
-- Store new site in Redis
|
|
ntop.setHashCache(REDIS_HASH_NAME, site_id, json.encode(site_json))
|
|
|
|
-- Increment counter for next site
|
|
ntop.setCache(REDIS_COUNTER_KEY, current_count + 1)
|
|
else
|
|
return res, msg -- Return validation error
|
|
end
|
|
|
|
local success_msg = "Site added successfully"
|
|
return true, success_msg
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Deletes an exporter site by ID
|
|
-- Note: Does not check if the site is currently in use by any flow devices
|
|
function exporter_site_utils.deleteExporterSite(id)
|
|
-- Get current sites to verify existence
|
|
local existing_sites = get_sites_from_cache()
|
|
|
|
-- Validate and normalize ID
|
|
if id then
|
|
id = tostring(id)
|
|
else
|
|
return false, "Invalid ID"
|
|
end
|
|
|
|
-- Check if site exists before deletion
|
|
if existing_sites[id] then
|
|
-- Remove site from Redis
|
|
ntop.delHashCache(REDIS_HASH_NAME, id)
|
|
else
|
|
return false, "Invalid Site"
|
|
end
|
|
|
|
local success_msg = "Site deleted successfully"
|
|
return true, success_msg
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
function exporter_site_utils.map_exporter_ip(exp_ip)
|
|
local site
|
|
|
|
if ntop.isPro() then
|
|
cache_exporters()
|
|
|
|
site = exporter_to_site[exp_ip]
|
|
|
|
if(site == nil) then
|
|
for k,v in pairs(iface_to_exporter) do
|
|
if(v == exp_ip) then
|
|
exp_ip = k
|
|
site = exporter_to_site(k)
|
|
break
|
|
end
|
|
end
|
|
end
|
|
else
|
|
site = DEFAULT_SITE.name
|
|
end
|
|
|
|
return exp_ip, site, exporter_to_name[exp_ip] or exp_ip
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
function exporter_site_utils.map_host_to_exporter_ip(host_ip)
|
|
local site
|
|
local exp_ip
|
|
|
|
if ntop.isPro() then
|
|
cache_exporters()
|
|
|
|
-- tprint(iface_to_exporter)
|
|
exp_ip = iface_to_exporter[host_ip]
|
|
|
|
if(exp_ip ~= nil) then
|
|
site = exporter_to_site[exp_ip]
|
|
|
|
if(site == nil) then
|
|
for k,v in pairs(iface_to_exporter) do
|
|
if(v == exp_ip) then
|
|
exp_ip = k
|
|
site = exporter_to_site(k)
|
|
break
|
|
end
|
|
end
|
|
end
|
|
else
|
|
exp_ip = host_ip
|
|
end
|
|
else
|
|
site = DEFAULT_SITE.name
|
|
end
|
|
|
|
return exp_ip, site, exporter_to_name[exp_ip] or exp_ip
|
|
end
|
|
|
|
-- ##############################################
|
|
|
|
-- Export the module for use in other Lua files
|
|
return exporter_site_utils
|