-- -- (C) 2013-24 - ntop.org -- local dirs = ntop.getDirs() package.path = dirs.installdir .. "/scripts/lua/modules/?.lua;" .. package.path if ((dirs.scriptdir ~= nil) and (dirs.scriptdir ~= "")) then package.path = dirs.scriptdir .. "/lua/modules/?.lua;" .. package.path end local top_sites_update local snmp_utils if ntop.isPro() then package.path = dirs.installdir .. "/scripts/lua/pro/modules/?.lua;" .. package.path package.path = dirs.installdir .. "/pro/scripts/callbacks/?.lua;" .. package.path snmp_utils = require "snmp_utils" top_sites_update = require("host_sites_update") end require "db_utils" require "lua_utils" require "prefs_utils" local json = require "dkjson" local template = require "template_utils" local format_utils = require "format_utils" local top_talkers_utils = require "top_talkers_utils" local internals_utils = require "internals_utils" local page_utils = require("page_utils") local ui_utils = require("ui_utils") local auth = require "auth" local behavior_utils = require("behavior_utils") local graph_utils = require "graph_utils" local recording_utils = require "recording_utils" local companion_interface_utils = require "companion_interface_utils" local storage_utils = require "storage_utils" local have_nedge = ntop.isnEdge() local sites_granularities = nil local show_zmq_encryption_public_key = false if ntop.isPro() then shaper_utils = require("shaper_utils") end sendHTTPContentTypeHeader('text/html') page = _GET["page"] ifid = _GET["ifid"] -- parse interface names and possibly fall back to the selected interface: -- priority goes to the interface id if ifid ~= nil and ifid ~= "" then if_name = getInterfaceName(ifid) else -- fall-back to the default interface if_name = ifname ifid = interface.name2id(ifname) end interface.select(ifname) ifname_clean = "iface_" .. tostring(ifid) msg = "" local is_packet_interface = interface.isPacketInterface() local is_pcap_dump = interface.isPcapDumpInterface() local ifstats = interface.getStats() local is_sub_interface = interface.isSubInterface() -- tprint(ifstats.stats.hosts_rcvd_only .. " / " .. ifstats.stats.hosts) if page == "syslog_producers" and not ifstats.isSyslog then page = "overview" end local is_mirrored_traffic = false local is_mirrored_traffic_pref = string.format("ntopng.prefs.ifid_%d.is_traffic_mirrored", interface.getId()) if not ntop.isnEdge() and is_packet_interface then is_mirrored_traffic = ternary(ntop.getPref(is_mirrored_traffic_pref) == '1', true, false) end local service_map_available = false local periodicity_map_available = false service_map_available, periodicity_map_available = behavior_utils.mapsAvailable() local disaggregation_criterion_key = "ntopng.prefs.dynamic_sub_interfaces.ifid_" .. tostring(ifid) .. ".mode" local host_threshold_rules_key = "ntopng.prefs.ifid_" .. tostring(ifid) .. ".host_threshold_rules" local host_ts_available = areHostTimeseriesEnabled() local charts_available = areInterfaceTimeseriesEnabled() local zmq_charts_available = charts_available and not interface.isView() function percentage(value, total) if (total > 0) then local pctg = round((value * 100) / total, 2) if (pctg > 0) then return (" [ " .. pctg .. " % ] ") end end return ("") end function inline_input_form(name, placeholder, tooltip, value, can_edit, input_opts, input_class, measure_unit) if (can_edit) then print( ']] if (measure_unit) then print([[]] .. i18n(measure_unit) .. [[]]) end else if (value ~= nil) then print(value) end end end function override_stats(stats, overr_stats) local new_stats = stats -- override stats with the values calculated from the latest user reset for k, v in pairs(overr_stats) do new_stats[k] = v end return new_stats end -- this is a user-browseable page, so we must return counters from -- the latest reset as the user may have chosen to reset statistics at some point if ifstats.stats and ifstats.stats_since_reset then ifstats.stats = override_stats(ifstats.stats, ifstats.stats_since_reset) end if ifstats.zmqRecvStats and ifstats.zmqRecvStats_since_reset then ifstats.zmqRecvStats = override_stats(ifstats.zmqRecvStats, ifstats.zmqRecvStats_since_reset) end local probes_stats = ifstats.probes or {} if interface.isView() then local view_id = interface.getId() local zmq_stats = {} local exporters_stats = {} for interface_name, _ in pairsByKeys(interface.getIfNames() or {}) do interface.select(interface_name) if interface.viewedBy() == view_id then local tmp = interface.getStats() if tmp.stats and tmp.stats_since_reset then tmp.stats = override_stats(tmp.stats, tmp.stats_since_reset) end if tmp.zmqRecvStats and tmp.zmqRecvStats_since_reset then tmp.zmqRecvStats = override_stats(tmp.zmqRecvStats, tmp.zmqRecvStats_since_reset) end for k, v in pairs(tmp.probes or {}) do probes_stats[k] = v end for k, v in pairs(tmp.exporters or {}) do if not exporters_stats[k] then exporters_stats[k] = {} end for key_stat, value_stat in pairs(v) do exporters_stats[k][key_stat] = value_stat + (exporters_stats[k][key_stat] or 0) end end for k, v in pairs(tmp.zmqRecvStats or {}) do zmq_stats[k] = (zmq_stats[k] or 0) + v end end end ifstats.zmqRecvStats = zmq_stats ifstats.exporters = exporters_stats interface.select(ifname) -- Go back to the View interface end local ext_interfaces = {} -- refresh traffic recording availability as one may have installed n2disk -- with a running instance of ntopng recording_utils.checkAvailable() if recording_utils.isAvailable() and recording_utils.isSupportedZMQInterface(ifid) then ext_interfaces = recording_utils.getExtInterfaces(ifid) end if (isAdministrator()) then if (page == "config") and (not table.empty(_POST)) then local custom_name = _POST["custom_name"] if starts(custom_name, "tcp:__") then -- Was mangled by sanitization custom_name = "tcp://" .. string.sub(custom_name, 7) end ntop.setCache('ntopng.prefs.ifid_' .. ifstats.id .. '.name', custom_name) local ifspeed_cache = 'ntopng.prefs.ifid_' .. ifstats.id .. '.speed' if isEmptyString(_POST["ifSpeed"]) then ntop.delCache(ifspeed_cache) else ntop.setCache(ifspeed_cache, _POST["ifSpeed"]) end if is_packet_interface then if _POST["gw_macs"] ~= nil then -- Gw MAC addresses - used to compute traffic direction local gw_macs_set = getGwMacsSet(ifstats.id) ntop.delCache(gw_macs_set) for _, mac in pairs(split(_POST["gw_macs"] or "", ",")) do mac = trimSpace(mac) if not isEmptyString(mac) then ntop.setMembersCache(gw_macs_set, mac) end end interface.reloadGwMacs() end end if not ntop.isnEdge() and is_packet_interface then if _POST["is_mirrored_traffic"] == "1" then is_mirrored_traffic = true else is_mirrored_traffic = false end ntop.setPref(is_mirrored_traffic_pref, ternary(is_mirrored_traffic == true, '1', '0')) interface.updateTrafficMirrored() end local sf = tonumber(_POST["scaling_factor"]) if (sf == nil) then sf = 1 end ntop.setCache("ntopng.prefs.iface_" .. tostring(ifid) .. '.scaling_factor', tostring(sf)) interface.loadScalingFactorPrefs() end end page_utils.print_header_and_set_active_menu_entry(page_utils.menu_entries.interface, { ifname = getHumanReadableInterfaceName(if_name) }) dofile(dirs.installdir .. "/scripts/lua/inc/menu.lua") -- print(msg) if _SERVER["REQUEST_METHOD"] == "POST" and _POST["companion_interface"] ~= nil then companion_interface_utils.setCompanion(ifstats.id, _POST["companion_interface"]) end if _SERVER["REQUEST_METHOD"] == "POST" and _POST["disaggregation_criterion"] ~= nil then if _POST["disaggregation_criterion"] == "none" then ntop.delCache(disaggregation_criterion_key) else ntop.setCache(disaggregation_criterion_key, _POST["disaggregation_criterion"]) end end if _SERVER["REQUEST_METHOD"] == "POST" and _POST["host_threshold_rules"] ~= nil then if _POST["host_threshold_rules"] == "none" then ntop.delCache(host_threshold_rules_key) else ntop.setCache(host_threshold_rules_key, _POST["host_threshold_rules"]) end end if _SERVER["REQUEST_METHOD"] == "POST" and not isEmptyString(_POST["traffic_recording_provider"]) then local prev_provider = recording_utils.getCurrentTrafficRecordingProvider(ifstats.id) -- if the current provider is the builtin ntopng and we are changing to another provider -- then it may be necessary to stop the builtin ntopng if prev_provider == "ntopng" and _POST["traffic_recording_provider"] ~= "ntopng" then recording_utils.stop(ifstats.id) ntop.setCache('ntopng.prefs.ifid_' .. ifstats.id .. '.traffic_recording.enabled', "false") end recording_utils.setCurrentTrafficRecordingProvider(ifstats.id, _POST["traffic_recording_provider"]) end local has_traffic_recording_page = (recording_utils.isAvailable() and (is_packet_interface or (ifstats.isView) -- Allows view interfaces (e.g., views of ZMQ) or ((recording_utils.isSupportedZMQInterface(ifid) and not table.empty(ext_interfaces))) or (recording_utils.getCurrentTrafficRecordingProvider(ifid) ~= "ntopng"))) local dismiss_recording_providers_reminder = recording_utils.isExternalProvidersReminderDismissed(ifstats.id) local url = ntop.getHttpPrefix() .. '/lua/if_stats.lua?ifid=' .. ifid -- Added global javascript variable, in order to disable the refresh of pie chart in case -- of historical interface print('\n\n') local short_name = getHumanReadableInterfaceName(ifname) local title = i18n("interface") .. ": " .. shortenCollapse(short_name) if (ntop.isPro()) then sites_granularities = top_sites_update.getGranularitySites(nil, nil, ifid, true) end page_utils.print_navbar(title, url, { { hidden = only_historical, active = page == "overview" or page == nil, page_name = "overview", label = "" }, { hidden = not is_packet_interface or ntop.isnEdge() or interface.isView(), active = page == "networks", page_name = "networks", label = i18n("networks") }, { hidden = have_nedge or not ifstats or ifstats.stats.packets == 0 or ntop.isnEdge(), active = page == "packets", page_name = "packets", label = i18n("packets") }, { hidden = not ifstats or not ifstats["has_macs"] or ntop.isnEdge(), active = page == "DSCP", page_name = "DSCP", label = i18n("dscp") }, { active = page == "ndpi", page_name = "ndpi", label = i18n("applications_long") }, { active = page == "ndpi_categories", page_name = "ndpi_categories", label = i18n("categories") }, { hidden = have_nedge, active = page == "ICMP", page_name = "ICMP", label = i18n("icmp") }, { hidden = not ifstats or not ifstats["has_macs"] or ntop.isnEdge(), active = page == "ARP", page_name = "ARP", label = i18n("arp") }, { hidden = (sites_granularities == nil or table.len(sites_granularities) == 0), active = page == "sites", page_name = "sites", label = i18n("sites_page.sites") }, { hidden = not charts_available, active = page == "historical", page_name = "historical", label = "" }, { hidden = have_nedge or not ifstats or table.empty(ifstats.profiles), active = page == "trafficprofiles", page_name = "trafficprofiles", label = "" }, { hidden = not has_traffic_recording_page, active = page == "traffic_recording", page_name = "traffic_recording", label = "" }, { hidden = not areAlertsEnabled() or not auth.has_capability(auth.capabilities.alerts), active = page == "alerts", page_name = "alerts", url = ntop.getHttpPrefix() .. "/lua/alert_stats.lua?&page=interface", label = "" }, { hidden = not hasTrafficReport(), active = page == "traffic_report", page_name = "traffic_report", label = "" }, { hidden = not isAdministrator() or is_pcap_dump, active = page == "config", page_name = "config", label = "" }, { active = page == "internals", page_name = "internals", label = "" }, { hidden = have_nedge or not isAdministrator() or not ntop.isEnterpriseM() or is_sub_interface or ifstats.isView, active = page == "sub_interfaces", page_name = "sub_interfaces", label = "" }, { hidden = have_nedge or not isAdministrator() or not ifstats.isSyslog, active = page == "syslog_producers", page_name = "syslog_producers", label = "" }, { hidden = not isAdministrator() or not ifstats.has_macs or have_nedge, active = page == "unassigned_pool_devices", page_name = "unassigned_pool_devices", label = "" }, { hidden = not isAdministrator() or is_pcap_dump or interface.isZMQInterface(), active = page == "dhcp", page_name = "dhcp", label = "" }, { hidden = (not periodicity_map_available), page_name = "periodicity_map", url = ntop.getHttpPrefix() .. "/lua/pro/enterprise/network_maps.lua?map=periodicity_map", label = "" }, { hidden = (not service_map_available), page_name = "service_map", url = ntop.getHttpPrefix() .. "/lua/pro/enterprise/network_maps.lua?map=service_map", label = "" } }) print(template.gen("modal_confirm_dialog.html", { dialog = { id = "reset_stats_dialog", action = "resetCounters(false)", title = i18n("reset_if_title"), message = i18n("reset_if_message"), confirm = i18n("reset"), confirm_button = "btn-danger" } })) print(template.gen("modal_confirm_dialog.html", { dialog = { id = "reset_drops_dialog", action = "resetCounters(true)", title = i18n("reset_drops_if_title"), message = i18n("reset_drops_if_message"), confirm = i18n("reset"), confirm_button = "btn-danger" } })) if ((page == "overview") or (page == nil)) then local tags = { ifid = ifstats.id } print("
") print("\n") print("\n") if isAdministrator() and (not is_pcap_dump and ifstats["type"] ~= "netfilter") then print("\n") end local nprobe_interface, nprobe_version, nprobe_probe_ip, nprobe_probe_public_ip, nprobe_edition, nprobe_license, nprobe_maintenance = {}, {}, {}, {}, {}, {}, {} local tot_num_nprobes = table.len(probes_stats or {}) local max_num_nprobes = 4 local cur_num_nprobes = 0 if tot_num_nprobes > 0 then print("") local msg = "" if ntop.isEnterpriseM() then msg = i18n("if_stats_overview.remote_probe_collecting_from_x_devices", { num = tot_num_nprobes, url = ntop.getHttpPrefix() .. "/lua/pro/enterprise/nprobe.lua" }) else msg = i18n("if_stats_overview.remote_probe_collecting_from_x_devices_no_link", { num = tot_num_nprobes, }) end print("") print("") end if cur_num_nprobes > 0 then local max_items_per_row = 3 local cur_i = 0 print("") if not interface.isView() then if not isEmptyString(ifstats["remote_pps"]) or not isEmptyString(ifstats["remote_bps"]) then if cur_i >= max_items_per_row then print(""); cur_i = 0 end print("") print('') cur_i = cur_i + 1 end end if ifstats["timeout.lifetime"] or 0 > 0 then if cur_i >= max_items_per_row then print(""); cur_i = 0 end print( "") print("") cur_i = cur_i + 1 end if (ifstats["timeout.idle"] or 0) > 0 then if cur_i >= max_items_per_row then print(""); cur_i = 0 end print("") cur_i = cur_i + 1 end if not isEmptyString(ifstats["probe.local_time"]) and not isEmptyString(ifstats["probe.remote_time"]) then local tdiff = math.abs(ifstats["probe.local_time"] - ifstats["probe.remote_time"]) if tdiff > 0 then if cur_i >= max_items_per_row then print(""); cur_i = 0 end print("") cur_i = cur_i + 1 end end local has_drops_export_queue_full = (tonumber(ifstats["zmq.drops.export_queue_full"]) and tonumber(ifstats["zmq.drops.export_queue_full"]) > 0) local has_drops_flow_collection_drops = (tonumber(ifstats["zmq.drops.flow_collection_drops"]) or 0) > 0 local has_drops_flow_collection_udp_socket_drops = (tonumber( ifstats["zmq.drops.flow_collection_udp_socket_drops"]) or 0) > 0 local has_remote_drops = (has_drops_export_queue_full or has_drops_flow_collection_drops) if has_drops_export_queue_full then local span_class = ' ' if cur_i >= max_items_per_row then print(""); cur_i = 0 end print("") print("") cur_i = cur_i + 1 end if has_drops_flow_collection_drops then local span_class = ' ' if cur_i >= max_items_per_row then print(""); cur_i = 0 end print( "") print("") cur_i = cur_i + 1 end if has_drops_flow_collection_udp_socket_drops then local span_class = ' ' if cur_i >= max_items_per_row then print(""); cur_i = 0 end print("") print("") cur_i = cur_i + 1 end if cur_i < max_items_per_row then -- 7 is the default colspan, then for each element we subtract two columns print(string.format("", 7 - cur_i * 2)) end print("") end local is_physical_iface = is_packet_interface and (not is_pcap_dump) local label = getHumanReadableInterfaceName(ifstats.name) local s if ((not isEmptyString(label)) and (label ~= ifstats.name)) then s = label .. " (" .. ifstats.name .. ")" else s = ifstats.name end if ((isAdministrator()) and (not is_pcap_dump)) then s = s .. " " end print('\n') print("") show_zmq_encryption_public_key = (ifstats.encryption and ifstats.encryption.public_key and isAdministrator()) if show_zmq_encryption_public_key == true then print("\n") end if is_physical_iface and not ifstats.isView then print("") print("\n") local speed_key = 'ntopng.prefs.ifid_' .. tostring(interface.name2id(ifname)) .. '.speed' local speed = ntop.getCache(speed_key) if (tonumber(speed) == nil) then speed = ifstats.speed end print("") print("") end if (not hasAllowedNetworksSet()) and ((ifstats.num_alerts_engaged > 0) or (ifstats.num_dropped_alerts > 0)) then print("") local warning = " " print("\n") print("\n") end label = i18n("pkts") print [[
" .. i18n("if_stats_overview.id") .. "" .. ifstats.id .. " ") if (ifstats.description ~= ifstats.name) then print(" (" .. ifstats.description .. ")") end print("
" .. i18n("if_stats_overview.state") .. "") state = toggleTableButton("", "", i18n("if_stats_overview.active"), "1", "primary", i18n("if_stats_overview.paused"), "0", "primary", "toggle_local", "ntopng.prefs.ifid_" .. tostring(ifid) .. "_not_idle") if (state == "0") then on_state = true else on_state = false end interface.setInterfaceIdleState(on_state) print("
" .. i18n("if_stats_overview.remote_probe") .. "" .. msg .. "
" .. i18n("if_stats_overview.probe_throughput") .. "' .. format_utils.bitsToSize(ifstats["remote_bps"]) .. '') print(' [' .. format_utils.pktsToSize(ifstats["remote_pps"]) .. ']
") print(i18n("if_stats_overview.probe_timeout_lifetime") .. " ") print("" .. i18n("if_stats_overview.note_lifetime_timeout") .. "
") if ((ifstats["timeout.collected_lifetime"] ~= nil) and (ifstats["timeout.collected_lifetime"] > 0)) then -- We're in collector mode on the nProbe side print(" " .. secondsToTime(ifstats["timeout.lifetime"]) .. " [" .. i18n("if_stats_overview.remote_flow_lifetime") .. ": " .. secondsToTime(ifstats["timeout.collected_lifetime"]) .. "]") else -- Modern nProbe in non-flow collector mode or old nProbe print(secondsToTime(ifstats["timeout.lifetime"])) end print("
" .. i18n("if_stats_overview.probe_timeout_idle") .. "" .. secondsToTime(ifstats["timeout.idle"]) .. "
" .. i18n("if_stats_overview.remote_probe_time") .. " " .. "") if (tdiff > 10) then print("") end print(formatValue(tdiff) .. " sec") if (tdiff > 10) then print("") end print("
" .. i18n("if_stats_overview.probe_zmq_drops_export_queue_full") .. " " .. formatValue(ifstats["zmq.drops.export_queue_full"]) .. "
") print(i18n("if_stats_overview.probe_zmq_drops_flow_collection_drops") .. " ") print("" .. i18n("if_stats_overview.note_probe_zmq_drops_flow_collection_drops") .. "
" .. formatValue(ifstats["zmq.drops.flow_collection_drops"]) .. "
" .. i18n("if_stats_overview.probe_zmq_drops_flow_collection_udp_socket_drops") .. " " .. formatValue(ifstats["zmq.drops.flow_collection_udp_socket_drops"]) .. "
' .. i18n("name") .. '

') print(s) if (ifstats.mac and ifstats.mac ~= "00:00:00:00:00:00") then print(" [" .. ifstats.mac .. "]"); end print('

" .. i18n("if_stats_overview.family") .. "") if (ifstats.type == "zmq") then print("ZMQ") else print(ifstats.type) end if (ifstats.inline) then print(" " .. i18n("if_stats_overview.in_path_interface")) end if (ifstats.has_traffic_directions) then print(" " .. i18n("if_stats_overview.has_traffic_directions") .. " ") end print("
" .. i18n("if_stats_overview.zmq_encryption_public_key") .. "" .. i18n("if_stats_overview.zmq_encryption_alias") .. "") print("") print("") print("
" .. i18n("if_stats_overview.note") .. ":
  • " .. i18n("if_stats_overview.zmq_encryption_public_key_note", { key = "<key>" }) .. "") local zmq_endpoint = ifstats.name local probe_mode = "" if endswith(zmq_endpoint, 'c') then zmq_endpoint = string.sub(zmq_endpoint, 1, -2) probe_mode = " --zmq-probe-mode" end print("
  • nprobe --zmq " .. zmq_endpoint .. probe_mode .. " --zmq-encryption-key '" .. i18n("if_stats_overview.zmq_encryption_alias") .. "' ...") print("
"); print("
" .. i18n("mtu") .. "" .. ifstats.mtu .. " " .. i18n("bytes") .. "" .. i18n("speed") .. "" .. bitsToSize(speed * 1000000) .. "
" .. ternary(ifstats.num_alerts_engaged > 0, warning, "") .. i18n("show_alerts.engaged_alerts") .. ternary(charts_available, " ", "") .. "" .. formatValue(ifstats.num_alerts_engaged) .. " " .. ternary(ifstats.num_dropped_alerts > 0, warning, "") .. i18n("show_alerts.dropped_alerts") .. " " .. ternary(charts_available, " ", "") .. "" .. formatValue(ifstats.num_dropped_alerts) .. "
]] print [[]] print [[ ]] if (ifstats.type ~= "zmq") then print [[ ]] else print [[ ]] end print [[ \n") if (ifstats.zmqRecvStats ~= nil and table.len(ifstats.zmqRecvStats) > 0) then print("\n") local tot_flows = (ifstats.zmqRecvStats.flows or 0) + (ifstats.zmqRecvStats.dropped_flows or 0) if tot_flows == 0 then tot_flows = 1 end local pctg_dropped_flows = (((ifstats.zmqRecvStats.dropped_flows or 0) * 100) / tot_flows) or 0 local tot_zmq_msg_rcvd = (ifstats.zmqRecvStats.zmq_msg_rcvd or 0) + (ifstats.zmqRecvStats.zmq_msg_drops or 0) if tot_zmq_msg_rcvd == 0 then tot_zmq_msg_rcvd = 1 end local pctg_dropped_zmq_msg = string.format("%.2f", (((ifstats.zmqRecvStats.zmq_msg_drops or 0) * 100) / tot_flows) or 0) print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") end if (ifstats.dbStats ~= nil) then print("\n") local tot_flows = ifstats.dbStats.flows + ifstats.dbStats.dropped_flows local pctg_dropped_flows = ((ifstats.dbStats.dropped_flows * 100) / (tot_flows or 1)) or 0 print("") print("") print( "") print("") print("") end print("\n") print("") print("") print("") print("\n") print("") if not is_sub_interface then print("") else print(i18n("if_stats_overview.dropped_flows") .. "" .. "") for _, v in pairs(ifstats.exporters or {}) do drops = drops + v["num_drops"] end end print("\n") end else print("") end print("") if (ifstats.has_traffic_directions) then local tx = ifstats.traffic_sent_since_reset local rx = ifstats.traffic_rcvd_since_reset local tx_pkts = ifstats.packets_sent_since_reset local rx_pkts = ifstats.packets_rcvd_since_reset local tot = rx + tx print("") print("") local tx_ratio = 0 local rx_ratio = 0 if tot > 0 then tx_ratio = (tx * 100 / tot) rx_ratio = (rx * 100 / tot) end print('') print("") end if interface.isSyslogInterface() then -- Syslog Stats print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") print("") -- Additional stats (e.g. Suricata) local external_json_stats = ntop.getCache("ntopng.prefs.ifid_" .. tostring(ifid) .. ".external_stats") if not isEmptyString(external_json_stats) then local external_stats = json.decode(external_json_stats) if external_stats ~= nil then local external_stats_title = i18n("external_stats.title") if external_stats.i18n_title then external_stats_title = i18n(external_stats.i18n_title) external_stats.i18n_title = nil end print("\n") for key, value in pairsByKeys(external_stats, asc) do print("") print( "") print("") print("") end end end end local function printExportStats(db_type, db_label) local export_count = ifstats.stats_since_reset[db_type].flow_export_count or 0 local export_rate = ifstats.stats_since_reset[db_type].flow_export_rate or 0 local export_drops = ifstats.stats_since_reset[db_type].flow_export_drops or 0 local export_drops_pct = 0 if export_drops > 0 or export_count > 0 then export_drops_pct = export_drops / (export_count + export_drops) * 100 end print("\n") print("") print("") print("") print("") local span_danger = "" if export_drops > 0 then span_danger = ' class="badge bg-danger"' end print("") if not is_packet_interface then print("") print("") else print("") end print("") end if not ifstats.isViewed then if ntop.isClickHouseEnabled() then printExportStats('db', 'ClickHouse') end if prefs.is_dump_flows_to_es_enabled then printExportStats('es', 'ElasticSearch') end if prefs.is_dump_flows_to_kafka_enabled then printExportStats('kafka', 'Kafka') end if prefs.is_dump_flows_to_syslog_enabled then printExportStats('syslog', 'Syslog') end end local an = ifstats.anomalies.tot_num_anomalies or {} local tot_an = 0 if table.len(an) > 0 then tot_an = an.local_hosts + an.remote_hosts end if (tot_an > 0) then -- TODO: Add JSON update print("") print("") print("") print("") print("\n") -- Cell for future usage end if isAdministrator() and ifstats.isView == false then -- Traffic recording if has_traffic_recording_page then print("\n") -- Traffic Recording stats local stats = recording_utils.stats(ifstats.id) local first_epoch = stats['FirstDumpedEpoch'] local last_epoch = stats['LastDumpedEpoch'] local start_time = stats['StartEpoch'] if first_epoch ~= nil then print("\n") end if start_time ~= nil then print("\n") end -- Smart Recording stats stats = recording_utils.smartStats(ifstats.id) first_epoch = stats['FirstDumpedEpoch'] last_epoch = stats['LastDumpedEpoch'] if first_epoch ~= nil then print("\n") end end -- Storage utilization local ts_utils = require "ts_utils_core" local storage_info = storage_utils.interfaceStorageInfo(ifid) local storage_items = {} if storage_info then if ts_utils.getDriverName() == "rrd" then if storage_info.rrd ~= nil and storage_info.rrd > 0 then table.insert(storage_items, { title = i18n("if_stats_overview.rrd_timeseries"), value = storage_info.rrd, class = "primary" }) end end if storage_info.flows ~= nil and storage_info.flows > 0 then table.insert(storage_items, { title = i18n("flows"), value = storage_info.flows, class = "info" }) end if storage_info.pcap ~= nil and storage_info.pcap > 0 then local link = nil if recording_utils.isAvailable(ifstats.id) then link = ntop.getHttpPrefix() .. "/lua/if_stats.lua?ifid=" .. ifid .. "&page=traffic_recording" end table.insert(storage_items, { title = i18n("traffic_recording.packet_dumps"), value = storage_info.pcap, class = "warning", link = link }) end if #storage_items > 0 then print("\n") end end end if ntop.isPcapDownloadAllowed() and ifstats.isView == false and not is_sub_interface and is_packet_interface then print("\n") end if isAdministrator() then print("") print("") print("\n") end if have_nedge and ifstats.type == "netfilter" and ifstats.netfilter then local st = ifstats.netfilter print("") local span_class = '' if st.nfq.queue_pct > 80 then span_class = "class='badge bg-danger'" end print("") print("") print("") print("") print("") print("") print("") print("") end print [[ ]] print("
]] print(i18n("if_stats_overview.traffic_breakdown")) print [[
" .. i18n("if_stats_overview.zmq_rx_statistics") .. "
" .. i18n("if_stats_overview.collected_flows") .. ternary(zmq_charts_available, " ", "") .. "" .. formatValue(ifstats.zmqRecvStats.flows) .. " " .. i18n("if_stats_overview.discarded_flows") .. "" .. formatValue(ifstats.zmqRecvStats.dropped_flows) .. " [ " .. pctg_dropped_flows .. " % ] " .. "
" .. i18n("if_stats_overview.zmq_message_rcvd") .. ternary(zmq_charts_available, " ", "") .. "" .. formatValue(ifstats.zmqRecvStats.zmq_msg_rcvd) .. " " .. i18n("if_stats_overview.zmq_message_drops") .. ternary(zmq_charts_available, " ", "") .. "" .. formatValue(ifstats.zmqRecvStats.zmq_msg_drops) .. " [ " .. pctg_dropped_zmq_msg .. " % ] " .. " " .. i18n("if_stats_overview.zmq_avg_msg_flows") .. ":
" .. i18n("if_stats_overview.interface_rx_updates") .. "" .. formatValue(ifstats.zmqRecvStats.events) .. "" .. i18n("if_stats_overview.sflow_counter_updates") .. "" .. formatValue(ifstats.zmqRecvStats.counters) .. "
" .. i18n("if_stats_overview.db_statistics") .. "
" .. i18n("if_stats_overview.processed_records") .. "" .. formatValue(ifstats.dbStats.flows) .. " " .. i18n("if_stats_overview.discarded_records") .. "" .. formatValue(ifstats.dbStats.dropped_flows) .. " [ " .. pctg_dropped_flows .. " % ] " .. "
" .. i18n("if_stats_overview.traffic_statistics") .. "
" .. i18n("report.traffic_anomalies") .. ternary(charts_available, " ", "") .. "" .. i18n("report.traffic_anomalies_local_hosts") .. "" .. formatValue(ifstats.anomalies.num_local_hosts_anomalies) .. " " .. i18n("report.traffic_anomalies_remote_hosts") .. "" .. formatValue(ifstats.anomalies.num_remote_hosts_anomalies) .. "
" .. i18n("report.total_traffic") .. ternary(charts_available, " ", "") .. "" .. bytesToSize(ifstats.stats.bytes) .. " [" .. formatValue(ifstats.stats.packets) .. " " .. label .. "] ") print(" ") local drops = 0 local probe_total_packets = 0 local probe_total_packet_drops = 0 for _, v in pairs(ifstats.probes or {}) do for _, v1 in pairs(v or {}) do for _, v2 in pairs(v or {}) do if (v2["packets.total"] ~= nil) then probe_total_packets = probe_total_packets + v2["packets.total"] end if (v2["packets.drops"] ~= nil) then probe_total_packet_drops = probe_total_packet_drops + v2["packets.drops"] end end end end if not ifstats.zmqRecvStats or table.len(ifstats.zmqRecvStats) == 0 then drops = ifstats.stats.drops print(i18n("if_stats_overview.dropped_packets") .. ternary(charts_available, " ", "") .. "") local drops = 0 if interface.isView() then local zmq_stats = {} for interface_name, _ in pairsByKeys(interface.getIfNames() or {}) do interface.select(interface_name) local tmp = interface.getStats() for k, v in pairs(tmp.exporters or {}) do drops = v["num_drops"] + drops end end interface.select(ifname) -- Go back to the View interface elseif (ifstats) then drops = ifstats.stats_since_reset.drops end if (drops > 0) then print('') end print(formatValue(drops) .. " " .. ternary((not ifstats.zmqRecvStats or table.len(ifstats.zmqRecvStats) == 0), i18n("pkts"), i18n("flows"))) if ((ifstats.stats.packets + drops) > 0) then local pctg = round((drops * 100) / (ifstats.stats.packets + drops), 2) if (pctg > 0) then print(" [ " .. pctg .. " % ] ") end end if (drops > 0) then print('') end print(" \n") if ifstats.zmqRecvStats and table.len(ifstats.zmqRecvStats) > 0 then -- Packets print(" " .. i18n("if_stats_overview.dropped_probe_packets") .. " ") local drop_pctg = 0 if (probe_total_packets > 0) then drop_pctg = (probe_total_packet_drops * 100) / probe_total_packets end print(string.format("%.1f", drop_pctg) .. "\n") print("") print("" .. i18n("if_stats_overview.note") .. ": " .. i18n("if_stats_overview.note_drop_ifstats_dynamic") .. "") print("
" .. i18n("http_page.traffic_sent") .. ternary(charts_available, " ", "") .. "" .. bytesToSize(tx) .. " [" .. formatValue(tx_pkts) .. " " .. label .. "] " .. i18n("http_page.traffic_received") .. ternary(charts_available, " ", "") .. "" .. bytesToSize(rx) .. " [" .. formatValue(rx_pkts) .. " " .. label .. "]
' .. i18n("sent") .. '
') print('
' .. i18n("received") .. '
" .. i18n("if_stats_overview.syslog_statistics") .. "
" .. i18n("if_stats_overview.collected_logs") .. "" .. formatValue(ifstats.syslog.tot_events) .. "" .. i18n("if_stats_overview.dispatched_logs") .. "" .. formatValue(ifstats.syslog.dispatched) .. "" .. i18n("if_stats_overview.unhandled_logs") .. "" .. formatValue(ifstats.syslog.unhandled) .. "
" .. i18n("if_stats_overview.malformed_logs") .. "" .. formatValue(ifstats.syslog.malformed) .. "" .. i18n("if_stats_overview.host_correlations") .. "" .. formatValue(ifstats.syslog.host_correlations) .. "" .. i18n("if_stats_overview.alert_events") .. "" .. formatValue(ifstats.syslog.alerts) .. "
" .. external_stats_title .. "
" .. ternary(i18n("external_stats." .. key), i18n("external_stats." .. key), key) .. "" .. ternary(type(value) == "number", formatValue(value), value) .. "
") if prefs.is_dump_flows_runtime_enabled then print(db_label .. " " .. i18n("if_stats_overview.flows_export_statistics")) else print(i18n("if_stats_overview.export_disabled")) end print("
" .. i18n("if_stats_overview.dumped_flows") .. ternary(charts_available, " ", "") .. "" .. formatValue(export_count) .. "") if export_rate == nil then export_rate = 0 end print(" [" .. formatValue(round(export_rate, 2)) .. " fps] ") print(i18n("if_stats_overview.dropped_export_to_db") .. ternary(charts_available, " ", "") .. "" .. formatValue(export_drops) .. " ") print("[" .. formatValue(round(export_drops_pct, 2)) .. "%]" .. i18n("if_stats_overview.direct_mode") .. "" .. ternary(prefs.is_dump_flows_direct_enabled, i18n("enabled"), i18n("disabled")) .. "
" .. i18n("if_stats_overview.counter_anomalies") .. "" .. i18n("total") .. ": " .. formatValue(tot_an) .. "" .. i18n("local_hosts") .. ": " .. formatValue(an.local_hosts) .. "" .. i18n("remote_hosts") .. ": " .. formatValue(an.remote_hosts) .. " 
" .. i18n("traffic_recording.traffic_recording") .. "
" .. i18n("traffic_recording.dump_window") .. "") if first_epoch ~= nil and last_epoch ~= nil and first_epoch > 0 and last_epoch > 0 then print(formatEpoch(first_epoch) .. " - " .. formatEpoch(last_epoch)) else print(i18n("traffic_recording.no_file")) end print("
" .. i18n("traffic_recording.active_since") .. "" .. formatEpoch(start_time)) if (start_time ~= nil) and (first_epoch ~= nil) and (first_epoch > 0) and (start_time > first_epoch) then print(' - ') print(i18n("traffic_recording.missing_data_msg")) end print("
" .. i18n("traffic_recording.smart_window") .. "") if first_epoch ~= nil and last_epoch ~= nil and first_epoch > 0 and last_epoch > 0 then print(formatEpoch(first_epoch) .. " - " .. formatEpoch(last_epoch)) else print(i18n("traffic_recording.no_file")) end print("
" .. i18n("traffic_recording.storage_utilization") .. "") print(graph_utils.stackedProgressBars(storage_info.total, storage_items, nil, bytesToSize)) print("
" .. i18n("live_capture.live_capture") .. " ") local live_traffic_utils = require("live_traffic_utils") live_traffic_utils.printLiveTrafficForm(interface.getId()) print("
" .. i18n("if_stats_overview.reset_counters") .. "") print( ' ') print( '') print("
" .. i18n("if_stats_overview.nf_queue_total") .. "" .. string.format("%s [%s %%]", formatValue(st.nfq.queue_total), formatValue(st.nfq.queue_pct)) .. " " .. i18n("if_stats_overview.nf_handle_packet_failed") .. "" .. formatValue(st.failures.handle_packet) .. " " .. i18n("if_stats_overview.nf_enobufs") .. "" .. formatValue(st.failures.no_buffers) .. "
") print(i18n("if_stats_overview.conntrack_flow_entries")) print("") print("" .. formatValue(st.nfq.num_conntrack_entries) .. "
]] print(i18n("if_stats_overview.note") .. ":

" .. i18n("if_stats_overview.note_packets")) print [[

\n") print("
") -- close of table responsive elseif page == "networks" and is_packet_interface then print("") if (ifstats.ip_addresses ~= "") then local tokens = split(ifstats.ip_addresses, ",") if (tokens ~= nil) then print("") end else print("") end local has_ghost_networks = false local ghost_icon = '' if ifstats.bcast_domains and table.len(ifstats.bcast_domains) > 0 then print("") else print("") end print("
" .. i18n("ip_address") .. "
  • ") local addresses = {} for _, s in pairs(tokens) do t = string.split(s, "/") host = interface.getHostInfo(t[1]) if (host ~= nil) then addresses[#addresses + 1] = hostinfo2detailshref(host, nil, t[1]) .. "/" .. t[2] else addresses[#addresses + 1] = s end end print(table.concat(addresses, "\n
  • ")) print("
" .. i18n("ip_address") .. "") print(i18n("if_stats_networks.no_ip_addresses_read")) print("
" .. i18n("broadcast_domain") .. "") local bcast_domains = {} for bcast_domain, domain_info in pairsByKeys(ifstats.bcast_domains) do bcast_domain = string.format("%s", ntop.getHttpPrefix(), bcast_domain, bcast_domain) if domain_info.ghost_network then has_ghost_networks = true bcast_domain = bcast_domain .. ' ' .. ghost_icon end bcast_domains[#bcast_domains + 1] = bcast_domain end if #bcast_domains > 0 then print("
    ") for _, bcast_domain in ipairs(bcast_domains) do print("
  • " .. bcast_domain .. "
  • ") end print("
") end print( ' ') print("
" .. i18n("broadcast_domain") .. "") print(i18n('if_stats_networks.no_broadcast_domains')) print("
") print(ui_utils.render_notes({ { content = i18n("if_stats_networks.note_iface_addresses") }, { content = i18n("if_stats_networks.note_iface_bcast_domains") }, { content = i18n("if_stats_networks.note_ghost_bcast_domains", { ghost_icon = ghost_icon }), hidden = not has_ghost_networks } })) elseif ((page == "packets")) then local nedge_hidden = ternary(have_nedge, 'class="hidden"', '') print [[ ]] print("\n") print("\n") print("\n") if (ifstats.type ~= "zmq") then print [[]] end print [[
" .. i18n("packets_page.tcp_packets_analysis") .. "" .. i18n("packets_page.retransmissions") .. "" .. formatPackets(ifstats.tcpPacketStats.retransmissions) .. percentage(ifstats.tcpPacketStats.retransmissions, ifstats.stats.packets) .. "
" .. i18n("packets_page.out_of_order") .. "" .. formatPackets(ifstats.tcpPacketStats.out_of_order) .. percentage(ifstats.tcpPacketStats.out_of_order, ifstats.stats.packets) .. "
" .. i18n("packets_page.lost") .. "" .. formatPackets(ifstats.tcpPacketStats.lost) .. percentage(ifstats.tcpPacketStats.lost, ifstats.stats.packets) .. "
]] print(i18n("packets_page.size_distribution")) print [[
]] print(i18n("packets_page.version_vs_flags_distribution")) print [[

]] elseif (page == "DSCP") then print [[
]] print(i18n("dscp_page.statistics")) print [[
]] elseif (page == "ndpi") then local context = { ifid = ifid, historical_available = hasClickHouseSupport(), csrf = ntop.getRandomCSRFValue(), uptime = ntop.getUptime(), l7_timeseries_enabled = areInterfaceL7TimeseriesEnabled(ifid), iface_ts_id = getIfacenDPITsName() } local json_context = json.encode(context) template.render("pages/vue_page.template", { vue_page_name = "PageTopInterfaceApplications", page_context = json_context }) elseif (page == "ndpi_categories") then local context = { ifid = ifid, historical_available = hasClickHouseSupport(), csrf = ntop.getRandomCSRFValue(), uptime = ntop.getUptime(), l7_timeseries_enabled = areInterfaceCategoriesTimeseriesEnabled(ifid) } local json_context = json.encode(context) template.render("pages/vue_page.template", { vue_page_name = "PageTopInterfaceCategories", page_context = json_context }) elseif (page == "ICMP") then print [[

]] print(i18n("icmp_page.icmp_message")) print [[]] print(i18n("icmp_page.icmp_type")) print [[]] print(i18n("icmp_page.icmp_code")) print [[]] print(i18n("packets")) print [[
]] print(i18n("icmp_page.icmp_message")) print [[]] print(i18n("icmp_page.icmp_type")) print [[]] print(i18n("icmp_page.icmp_code")) print [[]] print(i18n("packets")) print [[
]] elseif (page == "ARP") then local endpoint = string.format(ntop.getHttpPrefix() .. "/lua/rest/v2/get/interface/arp.lua?ifid=%s", ifId) local context = { json = json, template = template, sites = { endpoint = endpoint } } print(template.gen("pages/arp.template", context)) elseif (page == "sites") then if not prefs.are_top_talkers_enabled then local msg = i18n("sites_page.top_sites_not_enabled_message", { url = ntop.getHttpPrefix() .. "/lua/admin/prefs.lua?tab=protocols" }) print("
" .. msg .. "
") elseif table.len(sites_granularities) > 0 then local endpoint = string.format(ntop.getHttpPrefix() .. "/lua/pro/rest/v2/get/interface/top/sites.lua?ifid=%s", ifid) local context = { json = json, template = template, sites = { endpoint = endpoint, ifid = ifid, granularities = sites_granularities, default_granularity = "current" } } print(template.gen("pages/top_sites.template", context)) else local msg = i18n("sites_page.top_sites_not_seen") print("
" .. msg .. "
") end elseif (page == "historical") then local source_value_object = { ifid = interface.getId() } graph_utils.drawNewGraphs(source_value_object) elseif (page == "trafficprofiles") then print("\n") print("\n") for pname, pbytes in pairsByKeys(ifstats.profiles, asc) do local trimmed = trimSpace(pname) local statschart_icon = '' if areInterfaceTimeseriesEnabled(ifid) then statschart_icon = '' end print("\n") end print [[
" .. i18n("traffic_profiles.profile_name") .. "" .. i18n("chart") .. "" .. i18n("traffic") .. "
" .. pname .. "" .. statschart_icon .. "" .. bytesToSize(pbytes) .. "
]] elseif (page == "traffic_recording" and has_traffic_recording_page) then local master_ifid = interface.getMasterInterfaceId() if not dismiss_recording_providers_reminder then print('
' .. i18n('traffic_recording.msg_external_providers_detected', { url = ntop.getHttpPrefix() .. "/lua/if_stats.lua?page=config" }) .. '
') print [[ ]] end local tab = _GET["tab"] or "config" local recording_enabled = recording_utils.isEnabled(master_ifid) -- config tab is only shown when the recording service is managed by ntopng -- otherwise it is assumed that the user is managing the service manually with n2disk local config_enabled = (recording_utils.getCurrentTrafficRecordingProvider(master_ifid) == "ntopng") if tab == "config" and not config_enabled then if recording_enabled then tab = "status" else tab = "" end end if (tab == "status" or tab == "jobs") and not recording_enabled then if config_enabled then tab = "config" else tab = "" end end if (_SERVER["REQUEST_METHOD"] == "POST") and (tab == "config") then recording_enabled = not isEmptyString(_POST["record_traffic"]) end print('') print('
') if recording_enabled and tab == "status" then dofile(dirs.installdir .. "/scripts/lua/inc/traffic_recording_status.lua") elseif recording_enabled and ntop.isEnterpriseM() and tab == "jobs" then dofile(dirs.installdir .. "/scripts/lua/inc/traffic_recording_jobs.lua") elseif config_enabled and tab == "config" then -- config, default dofile(dirs.installdir .. "/scripts/lua/inc/traffic_recording_config.lua") end print('
') elseif (page == "config") then if (not isAdministrator()) then return end local messages = {} -- Flow dump check local interface_flow_dump = true if prefs.is_dump_flows_enabled then interface_flow_dump = (ntop.getPref("ntopng.prefs.ifid_" .. interface.getId() .. ".is_flow_dump_disabled") ~= "1") if _SERVER["REQUEST_METHOD"] == "POST" then local new_value = (_POST["interface_flow_dump"] == "1") if new_value ~= interface_flow_dump then local info = ntop.getInfo() local conf_utils = require "configuration_utils" -- Value changed interface_flow_dump = new_value ntop.setPref("ntopng.prefs.ifid_" .. interface.getId() .. ".is_flow_dump_disabled", ternary(interface_flow_dump, "0", "1")) conf_utils.set_restart_required() end end end if not table.empty(messages) then printMessageBanners(messages) print("
") end print [[
]] if ((not is_pcap_dump) and (ifstats.name ~= nil) and (ifstats.name ~= "dummy")) then -- Custom name print [[ ]] -- Interface speed if not have_nedge then print [[ ]] end end if not have_nedge then -- Scaling factor if is_packet_interface and not have_nedge then local label = ntop.getCache("ntopng.prefs.iface_" .. tostring(ifid) .. ".scaling_factor") if ((label == nil) or (label == "")) then label = "1" end print [[ ]] end end local serialize_by_mac local serialize_by_mac_key = string.format("ntopng.prefs.ifid_%u.serialize_local_broadcast_hosts_as_macs", interface.getId()) if not (is_mirrored_traffic) and not interface.isZMQInterface() then if (_POST["lbd_hosts_as_macs"] ~= nil) then serialize_by_mac = _POST["lbd_hosts_as_macs"] if ntop.getPref(serialize_by_mac_key) ~= serialize_by_mac then ntop.setPref(serialize_by_mac_key, serialize_by_mac) interface.updateLbdIdentifier() end else serialize_by_mac = ntop.getPref(serialize_by_mac_key) end else -- In case of mirrored traffic or ZMQ Interface, force the key to IP ntop.setPref(serialize_by_mac_key, false) interface.updateLbdIdentifier() end -- LBD identifier print [[ ]] -- Automatic Reports if isAdministrator() and ntop.isEnterpriseL() then package.path = dirs.installdir .. "/pro/scripts/lua/enterprise/modules/?.lua;" .. package.path local reports_utils = require "reports_utils" if toboolean(ntop.getPref("ntopng.prefs.automatic_reports_enabled")) then local automatic_reports_creation if _SERVER["REQUEST_METHOD"] == "POST" then automatic_reports_creation = toboolean(_POST["automatic_reports_creation"]) reports_utils.toggle_automatic_reports(ifid, automatic_reports_creation) else automatic_reports_creation = reports_utils.automatic_reports_enabled(ifid) end print [[]] end end -- per-interface Top-Talkers generation local interface_top_talkers_creation = true if _SERVER["REQUEST_METHOD"] == "POST" then if _POST["interface_top_talkers_creation"] ~= "1" then interface_top_talkers_creation = false top_talkers_utils.disableTop(interface.getId()) else top_talkers_utils.enableTop(interface.getId()) end else if not top_talkers_utils.areTopEnabled(interface.getId()) then interface_top_talkers_creation = false end end print [[]] -- Flow dump if prefs.is_dump_flows_enabled then print [[]] end -- Mirrored Traffic if not ntop.isnEdge() and is_packet_interface then print [[]] -- Gw Macs for Address-Based Traffic Directions local rv = ntop.getMembersCache(getGwMacsSet(ifstats.id)) or {} local members = {} -- impose sort order for _, addr in pairsByValues(rv, asc) do members[#members + 1] = addr end local gw_macs = table.concat(members, ",") print([[ ]]) end -- per-interface Network Discovery if interface.isDiscoverableInterface() then local discover = require "discover_utils" local interface_network_discovery = true if _SERVER["REQUEST_METHOD"] == "POST" then if _POST["interface_network_discovery"] ~= "1" then interface_network_discovery = false end ntop.setPref(discover.getInterfaceNetworkDiscoveryEnabledKey(interface.getId()), tostring(interface_network_discovery)) else interface_network_discovery = ntop.getPref( discover.getInterfaceNetworkDiscoveryEnabledKey(interface.getId())) if interface_network_discovery == "false" then interface_network_discovery = false end end print [[]] end if not is_sub_interface then local cur_companion = companion_interface_utils.getCurrentCompanion(ifstats.id) local companions = companion_interface_utils.getAvailableCompanions() if table.len(companions) > 1 then print [[ ]] end end if has_traffic_recording_page then local cur_provider = recording_utils.getCurrentTrafficRecordingProvider(ifstats.id) local providers = recording_utils.getAvailableTrafficRecordingProviders() -- only 1 provider means there's only the default ntopng -- so no need to show this extra menu entry if table.len(providers) > 1 then print [[ ]] end end if not is_sub_interface and not have_nedge then local cur_mode = ntop.getCache(disaggregation_criterion_key) if isEmptyString(cur_mode) then cur_mode = "none" end local labels = { i18n("prefs.none"), i18n("prefs.vlan"), i18n("prefs.probe_ip_address"), i18n("prefs.ingress_egress_flow_interface"), i18n("prefs.ingress_flow_interface"), i18n("prefs.ingress_vrf_id"), i18n("prefs.probe_ip_and_ingress_iface_idx") } local values = {} if is_packet_interface then values = { "none", "vlan" } else values = { "none", "vlan", "probe_ip", "iface_idx", "ingress_iface_idx", "ingress_vrf_id", "probe_ip_and_ingress_iface_idx" } end print [[ ]] -- Show dynamic traffic in the master interface local show_dyn_iface_traffic = false local show_dyn_iface_traffic_pref = string.format("ntopng.prefs.ifid_%d.show_dynamic_interface_traffic", interface.getId()) if _SERVER["REQUEST_METHOD"] == "POST" then if _POST["show_dyn_iface_traffic"] == "1" then show_dyn_iface_traffic = true end ntop.setPref(show_dyn_iface_traffic_pref, ternary(show_dyn_iface_traffic == true, '1', '0')) interface.updateDynIfaceTrafficPolicy() else show_dyn_iface_traffic = ternary(ntop.getPref(show_dyn_iface_traffic_pref) == '1', true, false) end print [[]] -- Show Push Host Filters to PF_RING toggle local push_host_filters = false local push_host_filters_pref = string.format("ntopng.prefs.ifid_%d.push_host_filters_to_pfring", interface.getId()) if _SERVER["REQUEST_METHOD"] == "POST" then if _POST["push_host_filters"] == "1" then push_host_filters = true end ntop.setPref(push_host_filters_pref, ternary(push_host_filters == true, '1', '0')) interface.updatePushFiltersSettings() else push_host_filters = ternary(ntop.getPref(push_host_filters_pref) == '1', true, false) end print [[]] end print [[
]] print(i18n("if_stats_config.custom_name")) print [[ ]] local label = getHumanReadableInterfaceName(ifstats.name) inline_input_form("custom_name", "Custom Name", i18n("if_stats_config.custom_name_popup_msg"), label, isAdministrator(), 'autocorrect="off" spellcheck="false"') print [[
]] print(i18n("if_stats_config.interface_speed")) print [[ ]] local ifspeed = getInterfaceSpeed(ifstats.id) inline_input_form("ifSpeed", "Interface Speed", i18n("if_stats_config.interface_speed_popup_msg"), ifspeed, isAdministrator(), 'type="number" min="1"', "d-inline-block", "if_stats_config.interface_speed_measure_unit") print [[
]] print(i18n("if_stats_config.scaling_factor")) print [[ ]] inline_input_form("scaling_factor", "Scaling Factor", i18n("if_stats_config.scaling_factor_popup_msg"), label, isAdministrator(), 'type="number" min="1" step="1"') print [[
]] print(i18n("prefs.toggle_host_tskey_title")) print [[ ]] print [[]] print [[
]] print(i18n("notes")) print [[

]] print(i18n("if_stats_config.local_broadcast_notes")) print [[

]] print(i18n("if_stats_config.automatic_reports_creation")) print [[ ]] print(template.gen("on_off_switch.html", { id = "automatic_reports_creation", checked = automatic_reports_creation })) print [[
]] print(i18n("if_stats_config.interface_top_talkers_creation")) print [[ ]] print(template.gen("on_off_switch.html", { id = "interface_top_talkers_creation", checked = interface_top_talkers_creation })) print [[
]] print(i18n("if_stats_config.dump_flows_to_database")) print [[ ]] print(template.gen("on_off_switch.html", { id = "interface_flow_dump", checked = interface_flow_dump })) print [[
]] print(i18n("if_stats_config.is_mirrored_traffic")) print [[ ]] print(template.gen("on_off_switch.html", { id = "is_mirrored_traffic", checked = is_mirrored_traffic })) print [[
]] .. i18n("if_stats_config.gw_macs") .. [[ ]]) print('') print([[
]] .. i18n("notes") .. [[

]] .. i18n("if_stats_config.gw_macs_description") .. [[

]] print(i18n("if_stats_config.interface_network_discovery")) print [[ ]] print(template.gen("on_off_switch.html", { id = "interface_network_discovery", checked = interface_network_discovery })) print [[
]] print(i18n("if_stats_config.companion_interface")) print [[
]] print(i18n("traffic_recording.traffic_recording_provider")) print [[
]] print(i18n("prefs.dynamic_interfaces_creation_title")) print [[ ]] print([[
]] .. i18n("notes") .. [[

]] .. i18n("prefs.dynamic_interfaces_creation_description") .. [[

]] .. i18n("prefs.dynamic_interfaces_creation_note_0") .. [[

]] .. i18n("prefs.dynamic_interfaces_creation_note_4") .. [[

]] .. i18n("prefs.dynamic_interfaces_creation_note_1") .. [[

]] .. (not is_packet_interface and i18n("prefs.dynamic_interfaces_creation_note_2") or '') .. [[

]] .. (not is_packet_interface and i18n("prefs.dynamic_interfaces_creation_note_3") or '') .. [[

]]) print [[
]] print(i18n("if_stats_config.show_dyn_iface_traffic")) print [[ ]] print(template.gen("on_off_switch.html", { id = "show_dyn_iface_traffic", checked = show_dyn_iface_traffic })) print [[
]] print(i18n("if_stats_config.toggle_push_host_filters")) print [[ ]] local queue = "pfring." .. interface.getId() .. ".filter.host.queue" print(template.gen("on_off_switch.html", { id = "push_host_filters", checked = push_host_filters, label = "" .. i18n("if_stats_config.toggle_push_host_filters_queue") .. ": " .. queue .. "" })) print [[


]] elseif (page == "internals") then internals_utils.printInternals(ifid, true --[[ hash tables ]], true --[[ periodic activities ]], true --[[ checks]], true --[[ queues --]]) print [[ ]] elseif (page == "snmp_bind") then if ((not hasSnmpDevices(ifstats.id)) or (not is_packet_interface)) then return end local snmp_host = _POST["ip"] local snmp_interface = _POST["snmp_port_idx"] or "" if (snmp_host ~= nil) then -- snmp_host can be empty snmp_utils.set_snmp_bound_interface(ifstats.id, snmp_host, snmp_interface) else local value = snmp_utils.get_snmp_bound_interface(ifstats.id) if value ~= nil then snmp_host = value.snmp_device snmp_interface = value.snmp_port end end local snmp_devices = snmp_utils.get_snmp_devices(ifstats.id) print [[
]] print [[
]] print(i18n("snmp.snmp_device")) print [[ ]] print(i18n("snmp.view_device")) print [[
]] print(i18n("snmp.snmp_interface")) print [[
]] print(i18n("snmp.note") .. ":") print [[
]] print(i18n("snmp.bound_interface_description")) print [[ ]] elseif (page == "sub_interfaces") then if (isAdministrator() and ntop.isEnterpriseM()) then dofile(dirs.installdir .. "/pro/scripts/lua/enterprise/sub_interfaces.lua") end elseif (page == "syslog_producers") then if (isAdministrator()) then dofile(dirs.installdir .. "/scripts/lua/syslog_producers.lua") end elseif (page == "unassigned_pool_devices") then dofile(dirs.installdir .. "/scripts/lua/unknown_devices.lua") elseif (page == "dhcp") then dofile(dirs.installdir .. "/scripts/lua/admin/dhcp.lua") elseif page == "traffic_report" then package.path = dirs.installdir .. "/pro/scripts/lua/enterprise/?.lua;" .. package.path local traffic_report = require "traffic_report" traffic_report.generate_traffic_report() end print(" ]] print [[ ]] dofile(dirs.installdir .. "/scripts/lua/inc/footer.lua") if show_zmq_encryption_public_key == true then print [[ ]] end print [[ ]]