lib/celluloid/dns/server.rb in celluloid-dns-0.0.1 vs lib/celluloid/dns/server.rb in celluloid-dns-0.17.3

- old
+ new

@@ -1,27 +1,151 @@ -module Celluloid - module DNS - class Server - # Maximum UDP packet we'll accept - MAX_PACKET_SIZE = 512 - - include Celluloid::IO - - def initialize(addr, port, &block) - @block = block - - # Create a non-blocking Celluloid::IO::UDPSocket - @socket = UDPSocket.new - @socket.bind(addr, port) - - async.run - end - - def run - loop do - data, (_, port, addr) = @socket.recvfrom(MAX_PACKET_SIZE) - @block.call Request.new(addr, port, @socket, data) - end - end - end - end -end \ No newline at end of file +# Copyright, 2009, 2012, by Samuel G. D. Williams. <http://www.codeotaku.com> +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. + +require 'celluloid/io' + +require_relative 'transaction' +require_relative 'logger' + +module Celluloid::DNS + class Server + include Celluloid::IO + + # The default server interfaces + DEFAULT_INTERFACES = [[:udp, "0.0.0.0", 53], [:tcp, "0.0.0.0", 53]] + + # Instantiate a server with a block + # + # server = Server.new do + # match(/server.mydomain.com/, IN::A) do |transaction| + # transaction.respond!("1.2.3.4") + # end + # end + # + def initialize(options = {}) + @logger = options[:logger] || Celluloid.logger + @interfaces = options[:listen] || DEFAULT_INTERFACES + + @origin = options[:origin] || '.' + end + + # Records are relative to this origin: + attr_accessor :origin + + attr_accessor :logger + + # Fire the named event as part of running the server. + def fire(event_name) + end + + finalizer def stop + # Celluloid.logger.debug(self.class.name) {"-> Shutdown..."} + + fire(:stop) + + # Celluloid.logger.debug(self.class.name) {"<- Shutdown..."} + end + + # Give a name and a record type, try to match a rule and use it for processing the given arguments. + def process(name, resource_class, transaction) + raise NotImplementedError.new + end + + # Process an incoming DNS message. Returns a serialized message to be sent back to the client. + def process_query(query, options = {}, &block) + start_time = Time.now + + # Setup response + response = Resolv::DNS::Message::new(query.id) + response.qr = 1 # 0 = Query, 1 = Response + response.opcode = query.opcode # Type of Query; copy from query + response.aa = 1 # Is this an authoritative response: 0 = No, 1 = Yes + response.rd = query.rd # Is Recursion Desired, copied from query + response.ra = 0 # Does name server support recursion: 0 = No, 1 = Yes + response.rcode = 0 # Response code: 0 = No errors + + transaction = nil + + begin + query.question.each do |question, resource_class| + begin + question = question.without_origin(@origin) + + @logger.debug {"<#{query.id}> Processing question #{question} #{resource_class}..."} + + transaction = Transaction.new(self, query, question, resource_class, response, options) + + transaction.process + rescue Resolv::DNS::OriginError + # This is triggered if the question is not part of the specified @origin: + @logger.debug {"<#{query.id}> Skipping question #{question} #{resource_class} because #{$!}"} + end + end + rescue StandardError => error + @logger.error "<#{query.id}> Exception thrown while processing #{transaction}!" + Celluloid::DNS.log_exception(@logger, error) + + response.rcode = Resolv::DNS::RCode::ServFail + end + + end_time = Time.now + @logger.debug {"<#{query.id}> Time to process request: #{end_time - start_time}s"} + + return response + end + + # Setup all specified interfaces and begin accepting incoming connections. + def run + @logger.info "Starting Celluloid::DNS server (v#{Celluloid::DNS::VERSION})..." + + fire(:setup) + + # Setup server sockets + @interfaces.each do |spec| + if spec.is_a?(BasicSocket) + spec.do_not_reverse_lookup + protocol = spec.getsockopt(Socket::SOL_SOCKET, Socket::SO_TYPE).unpack("i")[0] + ip = spec.local_address.ip_address + port = spec.local_address.ip_port + + case protocol + when Socket::SOCK_DGRAM + @logger.info "<> Attaching to pre-existing UDP socket #{ip}:#{port}" + link UDPSocketHandler.new(self, Celluloid::IO::Socket.try_convert(spec)) + when Socket::SOCK_STREAM + @logger.info "<> Attaching to pre-existing TCP socket #{ip}:#{port}" + link TCPSocketHandler.new(self, Celluloid::IO::Socket.try_convert(spec)) + else + raise ArgumentError.new("Unknown socket protocol: #{protocol}") + end + elsif spec[0] == :udp + @logger.info "<> Listening on #{spec.join(':')}" + link UDPHandler.new(self, spec[1], spec[2]) + elsif spec[0] == :tcp + @logger.info "<> Listening on #{spec.join(':')}" + link TCPHandler.new(self, spec[1], spec[2]) + else + raise ArgumentError.new("Invalid connection specification: #{spec.inspect}") + end + end + + fire(:start) + end + end +end