Codebase list dnscat2 / 9b31863 server / libs / socketer.rb
9b31863

Tree @9b31863 (Download .tar.gz)

socketer.rb @9b31863raw · history · blame

##
# socketer.rb
# Created November 27, 2015
# By Ron Bowes
#
# See: LICENSE.md
#
# This is a fairly simple wrapper around TCPSocket that makes the interface
# a bit more ruby-esque.
##

require 'socket'

class Socketer
  attr_reader :lhost, :lport

  BUFFER = 65536

  class Session
    attr_reader :host, :port

    def initialize(client, callbacks = {})
      @client     = client
      @on_connect = callbacks[:on_connect]
      @on_data    = callbacks[:on_data]
      @on_error   = callbacks[:on_error]
      @host       = client.peeraddr[2]
      @port       = client.peeraddr[1]

      # Do the 'new connection' callback
      if(@on_connect)
        @on_connect.call(self, client.peeraddr[2], client.peeraddr[1])
      end
    end

    def _handle_exception(e, msg)
      if(@on_error)
        @on_error.call(self, "Error #{msg}: #{e}", e)
      end
      stop!()
    end

    def ready!()
      @thread = Thread.new() do
        begin
          loop do
            data = @client.recv(BUFFER)

            if(data.nil? || data.length == 0)
              raise(IOError, "Connection closed")
            end

            if(@on_data)
              @on_data.call(self, data)
            end
          end
        rescue Exception => e
          begin
            _handle_exception(e, "receiving data")
          rescue Exception => e
            puts("Error in exception handler; please don't do that :)")
            puts(e)
            puts(e.backtrace)
          end
        end
      end
    end

    def stop!()
      if(@thread)
        @thread.exit()
      end

      @client.close()
    end

    def send(data)
      begin
        @client.write(data)
      rescue Exception => e
        _handle_exception(e, "sending data")

        stop!()
      end
    end
  end

  def initialize(socket, thread, lhost, lport)
    @socket = socket
    @thread = thread
    @lhost  = lhost
    @lport  = lport
  end

  def to_s()
    return "Tunnel listening on %s:%d" % [@lhost, @lport]
  end

  def kill()
    begin
      @thread.exit()
      @thread.join()
      @socket.close()
    rescue
      # Ignore exceptions
    end
  end

  def Socketer._handle_exception(e, msg, callbacks)
    if(callbacks[:on_error])
      callbacks[:on_error].call(self, "Error #{msg}: #{e}", e)
    end
  end

  def Socketer.listen(host, port, callbacks = {})
    # Create the socket right away so we'll know if it fails
    #puts("Listening on #{host}:#{port}...")
    s = TCPServer.new(host, port)

    return Socketer.new(s, Thread.new() do
      begin
        loop do
          Session.new(s.accept(), callbacks)
        end
      rescue StandardError => e
        Socketer._handle_exception(e, "connecting to #{host}:#{port}", callbacks)
      end
    end, host, port)
  end

  def Socketer.connect(host, port, callbacks = {})
    #puts("Connecting to #{host}:#{port}...")

    return Thread.new() do
      begin
        s = TCPSocket.new(host, port)
        if(s)
          Session.new(s, callbacks)
        else
          if(callbacks[:on_error])
            callbacks[:on_error].call(nil, "Couldn't connect to #{host}:#{port}!")
          end
        end
      rescue Exception => e
        Socketer._handle_exception(e, "connecting to #{host}:#{port}", callbacks)
      end
    end
  end
end