# # Copyright 2018-2019 Yves Desharnais # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require "socket" require "fluent/plugin/input" require_relative 'parser_netflow_v5' require_relative 'parser_netflow_v9' require_relative 'netflowipfix_records' require_relative 'vash' module Fluent module Plugin class NetflowipfixInput < Fluent::Plugin::Input Fluent::Plugin.register_input("netflowipfix", self) include DetachMultiProcessMixin class PortConnection def initialize(bind, port, tag, cache_ttl, definitions, queuesleep, log) @bind = bind @port = port @tag = tag @cache_ttl = cache_ttl @definitions = definitions @eventQueue = Queue.new @udpQueue = Queue.new @queuesleep = queuesleep @log = log end def bind @bind end def port @port end def tag @tag end def start @thread_udp = UdpListenerThread.new(@bind, @port, @udpQueue, @tag, @log) @thread_parser = ParserThread.new(@udpQueue, @queuesleep, @eventQueue, @cache_ttl, @definitions, @log) @thread_udp.start @thread_parser.start end # def start def stop @thread_udp.close @thread_udp.join @thread_parser.close @thread_parser.join end # def stop def restartParser if !@thread_parser.nil? @thread_parser.close @thread_parser.join @thread_parser = nil end @thread_parser = ParserThread.new(@udpQueue, @queuesleep, @eventQueue, @cache_ttl, @definitions, @log) @thread_parser.start end def event_pop @eventQueue.pop end def event_queue_length @eventQueue.length end end #class PortConnection config_param :tag, :string config_param :port, :integer, default: nil config_param :bind, :string, :default => '0.0.0.0' config_param :queuesleep, :integer, default: 10 def configure(conf) super $log.debug "NetflowipfixInput::configure: #{@bind}:#{@port}" @@connections ||= {} if @@connections.nil? end @@connections[@port] = PortConnection.new(@bind, @port, @tag, @cache_ttl, @definitions, @queuesleep, log) log.debug "NetflowipfixInput::configure NB=#{@@connections.length}" @total = 0 end def start super $log.debug "NetflowipfixInput::start NB=#{@@connections.length}" if @@connections.nil? else @@connections.each do | port, conn | $log.debug "start listening UDP on #{conn.bind}:#{conn.port}" conn.start end end waitForEvents end def shutdown super $log.debug "NetflowipfixInput::shutdown NB=#{@@connections.length}" if @@connections.nil? else @@connections.each do | port, conn | $log.debug "shutdown listening UDP on #{conn.bind}:#{conn.port}" conn.stop end @@connections = nil end end def restartConnections @@connections.each do | port, conn | $log.debug "restart parser #{conn.bind}:#{conn.port}" conn.restartParser end before = GC.stat(:total_freed_objects) GC.start after = GC.stat(:total_freed_objects) end def waitForEvents timeStart = Time.now.getutc.to_i nb = 0 loop do @@connections.each do | port, conn | if (conn.event_queue_length > 0) $log.trace "waitForEvents: #{conn.bind}:#{conn.port} queue has #{conn.event_queue_length} elements" nbq = conn.event_queue_length loop do ar = conn.event_pop time = ar[0] record = ar[1] router.emit(conn.tag, EventTime.new(time.to_i), record) # Free up variables for garbage collection ar = nil time = nil record = nil nb = nb + 1 nbq = nbq - 1 break if nbq == 0 end end end # @log.trace "NetflowipfixInput::waitForEvents ObjectSpace.memsize_of(NetflowipfixInput)=#{ObjectSpace.memsize_of(self)}" if Time.now.getutc.to_i - timeStart > 600 # 300 = 5 min restartConnections timeStart = Time.now.getutc.to_i end # Garbage collection if nb >= 20 nb = 0 # debugSpace end before = GC.stat(:total_freed_objects) GC.start after = GC.stat(:total_freed_objects) # $log.trace "waitForEvents: sleep #{@queuesleep}" sleep(@queuesleep) end end private class UdpListenerThread def initialize(bind, port, udpQueue, tag, log) @port = port @udpQueue = udpQueue @udp_socket = UDPSocket.new @udp_socket.bind(bind, port) @total = 0 @tag = tag @log = log end def start @thread = Thread.new(&method(:run)) @log.trace "UdpListenerThread::start" end def close @udp_socket.close end def join @thread.join end def run nb = 0 loop do msg, sender = @udp_socket.recvfrom(4096) @total = @total + msg.length @log.trace "UdpListenerThread::recvfrom #{msg.length} bytes for #{@total} total on UDP/#{@port}" record = {} record["message"] = msg record["length"] = msg.length record["total"] = @total record["sender"] = sender record["port"] = @port # time = EventTime.new() time = Time.now.getutc @udpQueue << [time, record] # Garbage collection msg = nil sender = nil nb = nb + 1 if nb > 100 GC.start nb = 0 end end end end # class UdpListenerThread class ParserThread def initialize(udpQueue, queuesleep, eventQueue, cache_ttl, definitions, log) @udpQueue = udpQueue @queuesleep = queuesleep @eventQueue = eventQueue @log = log @parser_v5 = NetflowipfixInput::ParserNetflowv5.new @parser_v9 = NetflowipfixInput::ParserNetflowv9.new @parser_v10 = NetflowipfixInput::ParserIPfixv10.new @parser_v9.configure(cache_ttl, definitions) @parser_v10.configure(cache_ttl, definitions) end def start @thread = Thread.new(&method(:run)) @log.debug "ParserThread::start" end def close # Garbage collection @parser_v5 = nil @parser_v9 = nil @parser_v10 = nil GC.start end def join @thread.join end def run loop do if @udpQueue.length == 0 sleep(@queuesleep) else block = method(:emit) ar = @udpQueue.pop time = ar[0] msg = ar[1] payload = msg["message"] host = msg["sender"] version,_ = payload[0,2].unpack('n') @log.trace "ParserThread::pop #{@udpQueue.length} v#{version}" case version when 5 packet = NetflowipfixInput::Netflow5Packet.read(payload) @parser_v5.handle_v5(host, packet, block) when 9 packet = NetflowipfixInput::Netflow9Packet.read(payload) @parser_v9.handle_v9(host, packet, block) when 10 packet = NetflowipfixInput::Netflow10Packet.read(payload) @parser_v10.handle_v10(host, packet, block) else $log.warn "Unsupported Netflow version v#{version}: #{version.class}" end # case # Free up variables for garbage collection ar = @udpQueue.pop version = nil time = nil msg = nil payload = nil host = nil end end # loop do end # def run def emit(time, event, host = nil) if !host.nil? event["host"] = host end @eventQueue << [time, event] @log.trace "ParserThread::emit #{@eventQueue.length}" end # def emit end # class ParserThread end # class DnsCacheOuput end # module Plugin end # module Fluent