Class: Rex::Proto::DNS::Server
- Inherits:
-
Object
- Object
- Rex::Proto::DNS::Server
- Includes:
- IO::GramServer
- Defined in:
- lib/rex/proto/dns/server.rb
Direct Known Subclasses
Defined Under Namespace
Classes: MockDnsClient
Constant Summary collapse
Instance Attribute Summary collapse
-
#cache ⇒ Rex::Proto::DNS::Server
Create DNS Server.
-
#fwd_res ⇒ Rex::Proto::DNS::Server
Create DNS Server.
-
#lock ⇒ Object
readonly
Returns the value of attribute lock.
-
#serve_tcp ⇒ Rex::Proto::DNS::Server
Create DNS Server.
-
#serve_udp ⇒ Rex::Proto::DNS::Server
Create DNS Server.
-
#sock_options ⇒ Object
readonly
Returns the value of attribute sock_options.
-
#start_cache ⇒ Rex::Proto::DNS::Server
Create DNS Server.
-
#tcp_sock ⇒ Object
readonly
Returns the value of attribute tcp_sock.
-
#udp_sock ⇒ Object
readonly
Returns the value of attribute udp_sock.
Attributes included from IO::GramServer
#dispatch_request_proc, #listener_thread, #send_response_proc
Class Method Summary collapse
-
.hardcore_alias(*args) ⇒ Object
Returns the hardcore alias for the DNS service.
Instance Method Summary collapse
-
#alias ⇒ Object
DNS server.
-
#default_dispatch_request(cli, data) ⇒ Object
Default DNS request dispatcher, attempts to find response records in cache or forwards request upstream.
-
#dispatch_request(cli, data) ⇒ Object
Process client request, handled with dispatch_request_proc if set.
-
#initialize(lhost = '0.0.0.0', lport = 53, udp = true, tcp = false, start_cache = true, res = nil, comm = nil, ctx = {}, dblock = nil, sblock = nil) ⇒ Server
constructor
A new instance of Server.
-
#monitor_listener ⇒ Object
protected
This method monitors the listener socket for new connections and calls the
on_client_connect
callback routine. -
#on_client_data(cli) ⇒ Object
protected
Processes request coming from client.
-
#running? ⇒ Boolean
Check if server is running.
-
#start ⇒ Object
Start the DNS server and cache.
-
#stop(flush_cache = false) ⇒ Object
Stop the DNS server and cache.
-
#switchns(ns = []) ⇒ Object
Switch DNS forwarders in resolver with thread safety.
Methods included from IO::GramServer
Constructor Details
#initialize(lhost = '0.0.0.0', lport = 53, udp = true, tcp = false, start_cache = true, res = nil, comm = nil, ctx = {}, dblock = nil, sblock = nil) ⇒ Server
Returns a new instance of Server.
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 |
# File 'lib/rex/proto/dns/server.rb', line 58 def initialize(lhost = '0.0.0.0', lport = 53, udp = true, tcp = false, start_cache = true, res = nil, comm = nil, ctx = {}, dblock = nil, sblock = nil) @serve_udp = udp @serve_tcp = tcp @sock_options = { 'LocalHost' => lhost, 'LocalPort' => lport, 'Context' => ctx, 'Comm' => comm } self.fwd_res = res.nil? ? Rex::Proto::DNS::Resolver.new(:comm => comm, :context => ctx) : res self.listener_thread = nil self.dispatch_request_proc = dblock self.send_response_proc = sblock self.start_cache = start_cache self.cache = Rex::Proto::DNS::Cache.new @lock = Mutex.new end |
Instance Attribute Details
#cache ⇒ Rex::Proto::DNS::Server
Create DNS Server
56 57 58 |
# File 'lib/rex/proto/dns/server.rb', line 56 def cache @cache end |
#fwd_res ⇒ Rex::Proto::DNS::Server
Create DNS Server
56 57 58 |
# File 'lib/rex/proto/dns/server.rb', line 56 def fwd_res @fwd_res end |
#lock ⇒ Object (readonly)
Returns the value of attribute lock.
57 58 59 |
# File 'lib/rex/proto/dns/server.rb', line 57 def lock @lock end |
#serve_tcp ⇒ Rex::Proto::DNS::Server
Create DNS Server
56 57 58 |
# File 'lib/rex/proto/dns/server.rb', line 56 def serve_tcp @serve_tcp end |
#serve_udp ⇒ Rex::Proto::DNS::Server
Create DNS Server
56 57 58 |
# File 'lib/rex/proto/dns/server.rb', line 56 def serve_udp @serve_udp end |
#sock_options ⇒ Object (readonly)
Returns the value of attribute sock_options.
57 58 59 |
# File 'lib/rex/proto/dns/server.rb', line 57 def @sock_options end |
#start_cache ⇒ Rex::Proto::DNS::Server
Create DNS Server
56 57 58 |
# File 'lib/rex/proto/dns/server.rb', line 56 def start_cache @start_cache end |
#tcp_sock ⇒ Object (readonly)
Returns the value of attribute tcp_sock.
57 58 59 |
# File 'lib/rex/proto/dns/server.rb', line 57 def tcp_sock @tcp_sock end |
#udp_sock ⇒ Object (readonly)
Returns the value of attribute udp_sock.
57 58 59 |
# File 'lib/rex/proto/dns/server.rb', line 57 def udp_sock @udp_sock end |
Class Method Details
.hardcore_alias(*args) ⇒ Object
Returns the hardcore alias for the DNS service
195 196 197 |
# File 'lib/rex/proto/dns/server.rb', line 195 def self.hardcore_alias(*args) "#{(args[0] || '')}-#{(args[1] || '')}-#{args[5] || ''}" end |
Instance Method Details
#alias ⇒ Object
DNS server.
202 203 204 |
# File 'lib/rex/proto/dns/server.rb', line 202 def alias "DNS Server" end |
#default_dispatch_request(cli, data) ⇒ Object
Default DNS request dispatcher, attempts to find response records in cache or forwards request upstream
160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 |
# File 'lib/rex/proto/dns/server.rb', line 160 def default_dispatch_request(cli,data) return if data.strip.empty? req = Packet.encode_drb(data) forward = req.dup # Find cached items, remove request from forwarded packet req.question.each do |ques| cached = self.cache.find(ques.qname, ques.qtype) if cached.empty? next else req.instance_variable_set(:@answer, (req.answer + cached).uniq) forward.question.delete(ques) end end # Forward remaining requests, cache responses if forward.question.count > 0 and @fwd_res forwarded = self.fwd_res.send(forward) req.instance_variable_set(:@answer, (req.answer + forwarded.answer).uniq) forwarded.answer.each do |ans| self.cache.cache_record(ans) end req.header.ra = true # Set recursion bit end # Finalize answers in response # Check for empty response prior to sending if req.answer.size < 1 req.header.rCode = Dnsruby::RCode::NOERROR end req.header.qr = true # Set response bit send_response(cli, req.data) end |
#dispatch_request(cli, data) ⇒ Object
Process client request, handled with dispatch_request_proc if set
146 147 148 149 150 151 152 |
# File 'lib/rex/proto/dns/server.rb', line 146 def dispatch_request(cli, data) if self.dispatch_request_proc self.dispatch_request_proc.call(cli,data) else default_dispatch_request(cli,data) end end |
#monitor_listener ⇒ Object (protected)
This method monitors the listener socket for new connections and calls the on_client_connect
callback routine.
212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 |
# File 'lib/rex/proto/dns/server.rb', line 212 def monitor_listener while true rds = [self.udp_sock] wds = [] eds = [self.udp_sock] r,_,_ = ::IO.select(rds,wds,eds,1) if (r != nil and r[0] == self.udp_sock) buf,host,port = self.udp_sock.recvfrom(65535) # Mock up a client object for sending back data cli = MockDnsClient.new(host, port, r[0]) dispatch_request(cli, buf) end end end |
#on_client_data(cli) ⇒ Object (protected)
Processes request coming from client
233 234 235 236 237 238 239 240 241 242 243 244 |
# File 'lib/rex/proto/dns/server.rb', line 233 def on_client_data(cli) begin data = cli.read(65535) raise ::EOFError if not data raise ::EOFError if data.empty? dispatch_request(cli, data) rescue EOFError => e self.tcp_socket.close_client(cli) if cli raise e end end |
#running? ⇒ Boolean
Check if server is running
93 94 95 |
# File 'lib/rex/proto/dns/server.rb', line 93 def running? self.listener_thread and self.listener_thread.alive? end |
#start ⇒ Object
Start the DNS server and cache
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 |
# File 'lib/rex/proto/dns/server.rb', line 100 def start if self.serve_udp @udp_sock = Rex::Socket::Udp.create(self.) self.listener_thread = Rex::ThreadFactory.spawn("UDPDNSServerListener", false) { monitor_listener } end if self.serve_tcp @tcp_sock = Rex::Socket::TcpServer.create(self.) self.tcp_sock.on_client_data_proc = Proc.new { |cli| on_client_data(cli) } self.tcp_sock.start if !self.serve_udp self.listener_thread = tcp_sock.listener_thread end end self.cache.start if self.start_cache end |
#stop(flush_cache = false) ⇒ Object
Stop the DNS server and cache
127 128 129 130 131 132 133 134 135 136 137 138 139 |
# File 'lib/rex/proto/dns/server.rb', line 127 def stop(flush_cache = false) ensure_close = [self.udp_sock, self.tcp_sock].compact begin self.listener_thread.kill if self.listener_thread.respond_to?(:kill) self.listener_thread = nil ensure while csock = ensure_close.shift csock.stop if csock.respond_to?(:stop) csock.close unless csock.respond_to?(:close) and csock.closed? end end self.cache.stop(flush_cache) end |
#switchns(ns = []) ⇒ Object
Switch DNS forwarders in resolver with thread safety
81 82 83 84 85 86 87 88 |
# File 'lib/rex/proto/dns/server.rb', line 81 def switchns(ns = []) if ns.respond_to?(:split) ns = [ns] end self.lock.synchronize do self.fwd_res.nameserver = ns end end |