196 lines
7.9 KiB
Common Lisp
196 lines
7.9 KiB
Common Lisp
;;;; backplane-dns.lisp
|
|
|
|
(in-package #:backplane-dns)
|
|
|
|
(defvar *dns-store* nil)
|
|
|
|
(defun symbolize (str) (-> str string-upcase (intern :KEYWORD)))
|
|
|
|
(defclass dns-request ()
|
|
((hostname :initarg :hostname)
|
|
(domain :initarg :domain)
|
|
(msg-id :initarg :msg-id
|
|
:reader msg-id)))
|
|
|
|
(defclass request-change-ipv4 (dns-request)
|
|
((ip-address :initarg :ip-address)))
|
|
|
|
(defclass request-change-ipv6 (dns-request)
|
|
((ip-address :initarg :ip-address)))
|
|
|
|
(defclass request-change-sshfp (dns-request)
|
|
((sshfp :initarg :sshfp)))
|
|
|
|
(defclass unknown-dns-request (dns-request)
|
|
((request-type :initarg :request-type
|
|
:reader request-type)))
|
|
|
|
(defparameter *hostname-rx*
|
|
"(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])")
|
|
|
|
(define-condition backplane-dns-error (error)
|
|
((msg :initarg :msg :initform nil :reader error-msg)))
|
|
|
|
(define-condition invalid-hostname (backplane-dns-error)
|
|
((hostname :initarg :hostname :reader invalid-hostname)))
|
|
|
|
(let ((hostname-extractor-rx
|
|
(cl-ppcre:create-scanner
|
|
`(:SEQUENCE :START-ANCHOR "host-"
|
|
(:REGISTER (:REGEX ,*hostname-rx*))
|
|
#\@
|
|
(:REGEX ,*hostname-rx*) :END-ANCHOR))))
|
|
(defun sender-hostname (sender)
|
|
(if-let ((hostname (cl-ppcre:register-groups-bind (extracted-hostname nil)
|
|
(hostname-extractor-rx sender)
|
|
extracted-hostname)))
|
|
hostname
|
|
(error 'invalid-hostname
|
|
:hostname sender
|
|
:msg (format nil "unable to extract hostname from ~A, expecting format host-<hostname>"
|
|
sender)))))
|
|
|
|
(defmethod backplane-server:handle-message ((message dns-request))
|
|
(handler-case
|
|
(if-let ((store *dns-store*))
|
|
(-> message
|
|
(handle-dns-message store)
|
|
(handle-dns-response (msg-id message)))
|
|
(make-error :msg "dns store is not initialized"
|
|
:msg-id (msg-id message)))
|
|
(domain-name-missing (err)
|
|
(make-error :msg (format nil "missing domain name: ~A"
|
|
(missing-domain err))
|
|
:msg-id (msg-id message)))
|
|
(invalid-ip (err)
|
|
(make-error :msg (error-msg err)
|
|
:msg-id (msg-id message)))
|
|
(invalid-sshfp (err)
|
|
(make-error :msg (format nil "invalid ssh fingerprint: ~A"
|
|
(invalid-sshfp err))
|
|
:msg-id (msg-id message)))
|
|
(error (e)
|
|
(declare (ignorable e))
|
|
(make-error :msg (format nil "an unknown error occurred: ~A"
|
|
e)
|
|
:msg-id (msg-id message)))))
|
|
|
|
(defgeneric handle-dns-message (message store))
|
|
|
|
(defmethod handle-dns-message ((message request-change-ipv4) store)
|
|
(with-slots (hostname domain ip-address) message
|
|
(backplane-dns-store:set-ipv4 store domain hostname ip-address)
|
|
(make-instance 'dns-success
|
|
:msg (format nil "successfully set ipv4 for ~A.~A to ~A"
|
|
hostname domain ip-address))))
|
|
|
|
(defmethod handle-dns-message ((message request-change-ipv6) store)
|
|
(with-slots (hostname domain ip-address) message
|
|
(backplane-dns-store:set-ipv6 store domain hostname ip-address)
|
|
(make-instance 'dns-success
|
|
:msg (format nil "successfully set ipv6 for ~A.~A to ~A"
|
|
hostname domain ip-address))))
|
|
|
|
(defmethod handle-dns-message ((message request-change-sshfp) store)
|
|
(with-slots (hostname domain sshfp) message
|
|
(backplane-dns-store:set-sshfp store domain hostname sshfp)
|
|
(make-instance 'dns-success
|
|
:msg (format nil "successfully set sshfps for ~A.~A"
|
|
hostname domain))))
|
|
|
|
(defmethod handle-dns-message ((message unknown-dns-request) store)
|
|
(make-instance 'dns-error
|
|
:msg (format nil "unknown request to the dns service: ~A"
|
|
(request-type message))))
|
|
|
|
(defgeneric parse-dns-message (sender request message msg-id)
|
|
(:documentation "Parse a DNS service message of type REQUEST"))
|
|
|
|
(defmethod parse-dns-message (sender (request (eql :CHANGE_IPV4)) message msg-id)
|
|
(make-instance 'request-change-ipv4
|
|
:hostname (sender-hostname sender)
|
|
:domain (cdr (assoc :DOMAIN message))
|
|
:ip-address (cdr (assoc :IP message))
|
|
:msg-id msg-id))
|
|
|
|
(defmethod parse-dns-message (sender (request (eql :CHANGE_IPV6)) message msg-id)
|
|
(make-instance 'request-change-ipv6
|
|
:hostname (sender-hostname sender)
|
|
:domain (cdr (assoc :DOMAIN message))
|
|
:ip-address (cdr (assoc :IP message))
|
|
:msg-id msg-id))
|
|
|
|
(defmethod parse-dns-message (sender (request (eql :CHANGE_SSHFP)) message msg-id)
|
|
(make-instance 'request-change-sshfp
|
|
:hostname (sender-hostname sender)
|
|
:domain (cdr (assoc :DOMAIN message))
|
|
:sshfp (cdr (assoc :SSHFP message))
|
|
:msg-id msg-id))
|
|
|
|
(defmethod parse-dns-message (sender request message msg-id)
|
|
(make-instance 'unknown-dns-request
|
|
:request-type request
|
|
:msg-id msg-id))
|
|
|
|
(defmethod backplane-server:parse-message (sender (service (eql :DNS)) api-version message msg-id)
|
|
(parse-dns-message sender (symbolize (cdr (assoc :REQUEST message))) message msg-id))
|
|
|
|
(defclass dns-response ()
|
|
((msg :initarg :msg :reader msg)))
|
|
(defclass dns-success (dns-response) ())
|
|
(defclass dns-error (dns-response) ())
|
|
|
|
(defgeneric handle-dns-response (resp msg-id))
|
|
(defmethod handle-dns-response ((resp dns-success) msg-id)
|
|
(make-success :msg (msg resp) :msg-id msg-id))
|
|
(defmethod handle-dns-response ((resp dns-error) msg-id)
|
|
(make-error :msg (msg resp) :msg-id msg-id))
|
|
|
|
(defun backplane-dns-listen (&key
|
|
xmpp-host
|
|
xmpp-username
|
|
xmpp-password
|
|
db-host
|
|
db-port
|
|
db-name
|
|
db-username
|
|
db-password
|
|
db-use-ssl)
|
|
(let ((postmodern:*ignore-unknown-columns* t)
|
|
(cl-postgres:*query-log* *standard-output*)
|
|
(*dns-store* (make-instance 'backplane-dns-store:postgres-dns-store)))
|
|
(postmodern:with-connection (list db-name db-username db-password db-host (port db-port) (use-ssl db-use-ssl))
|
|
(with-backplane (backplane (backplane-connect xmpp-host xmpp-username xmpp-password))
|
|
(start-listening backplane)))))
|
|
|
|
(defun read-file-line (filename)
|
|
(let ((input (open filename :if-does-not-exist nil)))
|
|
(if input
|
|
(read-line input)
|
|
(uiop:die 1 "unable to read file: ~A" filename))))
|
|
|
|
(defun getenv-or-fail (env-var &optional default)
|
|
(let ((value (uiop:getenv env-var)))
|
|
(if (null value)
|
|
(if default
|
|
default
|
|
(uiop:die 1 "unable to find required env var: ~A" env-var))
|
|
value)))
|
|
|
|
(defun getenv-flag (env-var)
|
|
(let ((value (uiop:getenv env-var)))
|
|
(if value t nil)))
|
|
|
|
(defun start-listener-with-env ()
|
|
(backplane-dns-listen
|
|
:xmpp-host (getenv-or-fail "FUDO_DNS_BACKPLANE_XMPP_HOSTNAME")
|
|
:xmpp-username (getenv-or-fail "FUDO_DNS_BACKPLANE_XMPP_USERNAME")
|
|
:xmpp-password (read-file-line (getenv-or-fail "FUDO_DNS_BACKPLANE_XMPP_PASSWORD_FILE"))
|
|
|
|
:db-name (getenv-or-fail "FUDO_DNS_BACKPLANE_DATABASE_NAME")
|
|
:db-username (getenv-or-fail "FUDO_DNS_BACKPLANE_DATABASE_USERNAME")
|
|
:db-host (getenv-or-fail "FUDO_DNS_BACKPLANE_DATABASE_HOSTNAME")
|
|
:db-port (getenv-or-fail "FUDO_DNS_BACKPLANE_DATABASE_PORT" 5432)
|
|
:db-password (read-file-line (getenv-or-fail "FUDO_DNS_BACKPLANE_DATABASE_PASSWORD_FILE"))
|
|
:db-use-ssl (getenv-flag "FUDO_DNS_BACKPLANE_DATABASE_USE_SSL")))
|