(in-package :sb-bsd-sockets)
#|| Name Service
Presently name service is implemented by calling whatever gethostbyname(2) uses. This may be any or all of /etc/hosts, NIS, DNS, or something completely different. Typically it's controlled by /etc/nsswitch.conf
Direct links to the asynchronous resolver(3) routines would be nice to have eventually, so that we can do DNS lookups in parallel with other things |# (defclass host-ent () ((name :initarg :name :accessor host-ent-name) (aliases :initarg :aliases :accessor host-ent-aliases) (address-type :initarg :type :accessor host-ent-address-type) ; presently always AF_INET (addresses :initarg :addresses :accessor host-ent-addresses))) (defmethod host-ent-address ((host-ent host-ent)) (car (host-ent-addresses host-ent))) ;(define-condition host-not-found-error (socket-error)) ; host unknown ;(define-condition no-address-error (socket-error)) ; valid name but no IP address ;(define-condition no-recovery-error (socket-error)) ; name server error ;(define-condition try-again-error (socket-error)) ; temporary (defun get-host-by-name (host-name) "Returns a HOST-ENT instance for HOST-NAME or throws some kind of condition. HOST-NAME may also be an IP address in dotted quad notation or some other weird stuff - see gethostbyname(3) for grisly details." (let ((h (sockint::gethostbyname host-name))) (make-host-ent h))) (defun get-host-by-address (address) "Returns a HOST-ENT instance for ADDRESS, which should be a vector of (integer 0 255), or throws some kind of error. See gethostbyaddr(3) for grisly details." (let ((packed-addr (sockint::allocate-in-addr))) (loop for i from 0 to 3 do (setf (sockint::in-addr-addr packed-addr i) (elt address i))) (make-host-ent (sb-sys:without-gcing (sockint::gethostbyaddr (sockint::array-data-address packed-addr) 4 sockint::af-inet))))) (defun make-host-ent (h) (if (sockint::foreign-nullp h) (name-service-error "gethostbyname")) (let* ((local-h (sockint::foreign-vector h 1 sockint::size-of-hostent)) (length (sockint::hostent-length local-h)) (aliases (loop for i = 0 then (1+ i) for al = (sb-sys:sap-ref-sap (sb-sys:int-sap (sockint::hostent-aliases local-h)) (* i 4)) until (= (sb-sys:sap-int al) 0) collect (sb-c-call::%naturalize-c-string al))) (address0 (sb-sys:sap-ref-sap (sb-sys:int-sap (sockint::hostent-addresses local-h)) 0)) (addresses (loop for i = 0 then (+ length i) for ad = (sb-sys:sap-ref-32 address0 i) while (> ad 0) collect (sockint::foreign-vector (sb-sys:sap+ address0 i) 1 length)))) (make-instance 'host-ent :name (sb-c-call::%naturalize-c-string (sb-sys:int-sap (sockint::hostent-name local-h))) :type (sockint::hostent-type local-h) :aliases aliases :addresses addresses))) ;;; The remainder is my fault - gw (defvar *name-service-errno* 0 "The value of h_errno, after it's been fetched from Unix-land by calling GET-NAME-SERVICE-ERRNO") (defun name-service-error (where) (get-name-service-errno) ;; Comment next to NETDB_INTERNAL in netdb.h says "See errno.". ;; This special case treatment hasn't actually been tested yet. (if (= *name-service-errno* sockint::NETDB-INTERNAL) (socket-error where) (let ((condition (condition-for-name-service-errno *name-service-errno*))) (error condition :errno *name-service-errno* :syscall where)))) (define-condition name-service-error (condition) ((errno :initform nil :initarg :errno :reader name-service-error-errno) (symbol :initform nil :initarg :symbol :reader name-service-error-symbol) (syscall :initform "an unknown location" :initarg :syscall :reader name-service-error-syscall)) (:report (lambda (c s) (let ((num (name-service-error-errno c))) (format s "Name service error in \"~A\": ~A (~A)" (name-service-error-syscall c) (or (name-service-error-symbol c) (name-service-error-errno c)) (get-name-service-error-message num)))))) (defmacro define-name-service-condition (symbol name) `(progn (define-condition ,name (name-service-error) ((symbol :reader name-service-error-symbol :initform (quote ,symbol)))) (push (cons ,symbol (quote ,name)) *conditions-for-name-service-errno*))) (defparameter *conditions-for-name-service-errno* nil) (define-name-service-condition sockint::NETDB-INTERNAL netdb-internal-error) (define-name-service-condition sockint::NETDB-SUCCESS netdb-success-error) (define-name-service-condition sockint::HOST-NOT-FOUND host-not-found-error) (define-name-service-condition sockint::TRY-AGAIN try-again-error) (define-name-service-condition sockint::NO-RECOVERY no-recovery-error) ;; this is the same as the next one ;;(define-name-service-condition sockint::NO-DATA no-data-error) (define-name-service-condition sockint::NO-ADDRESS no-address-error) (defun condition-for-name-service-errno (err) (or (cdr (assoc err *conditions-for-name-service-errno* :test #'eql)) 'name-service)) (defun get-name-service-errno () (setf *name-service-errno* (sb-alien:alien-funcall (sb-alien:extern-alien "get_h_errno" (function integer))))) #-solaris (progn #+sbcl (sb-alien:define-alien-routine "hstrerror" sb-c-call:c-string (errno integer)) #+cmu (alien:def-alien-routine "hstrerror" sb-c-call:c-string (errno integer)) (defun get-name-service-error-message (num) (hstrerror num)) )