celluloid/celluloid-dns

View on GitHub
lib/celluloid/dns/server.rb

Summary

Maintainability
A
3 hrs
Test Coverage
# 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