-
-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathdispatch.lisp
89 lines (77 loc) · 3.5 KB
/
dispatch.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
(in-package #:org.shirakumo.radiance.core)
(declaim (ftype (function (uri-dispatcher) function) dispatch-function))
(eval-when (:compile-toplevel :load-toplevel :execute)
(define-documentable uri-dispatcher (uri)
((name :initarg :name :accessor name)
(dispatch-function :initarg :dispatch-function :accessor dispatch-function)
(priority :initarg :priority :accessor priority))
(:default-initargs
:name (error "NAME required.")
:dispatch-function (error "DISPATCH-FUNCTION required.")
:priority NIL)
(:find-function uri-dispatcher)))
(defvar *uri-registry* (make-hash-table :test 'eql))
(defvar *uri-priority* (make-array 0 :element-type 'uri-dispatcher))
(defparameter *uri-fallback* (lambda ()
(cond
((boundp '*request*)
(error 'request-not-found :message "Reached dispatch fallback."))
(T
(error 'request-not-found :request NIL :message "Reached dispatch fallback.")))))
(declaim (type function *uri-fallback*))
(declaim (type (simple-array uri-dispatcher 1) *uri-priority*))
(defmethod print-object ((dispatcher uri-dispatcher) stream)
(print-unreadable-object (dispatcher stream :type T)
(format stream "~a " (name dispatcher))
(call-next-method)))
(defun uri-dispatcher (name)
(gethash name *uri-registry*))
(defun (setf uri-dispatcher) (dispatcher name)
(setf (gethash name *uri-registry*) dispatcher)
(rebuild-uri-priority)
dispatcher)
(defun remove-uri-dispatcher (name)
(remhash name *uri-registry*)
(rebuild-uri-priority)
name)
(defun list-uri-dispatchers ()
(coerce *uri-priority* 'list))
(defun uri-dispatcher> (a b)
(or (and (priority a)
(or (not (priority b))
(>= (priority a) (priority b))))
(and (not (priority b))
(uri> a b))))
(defun rebuild-uri-priority ()
(setf *uri-priority*
(sort (make-array (hash-table-count *uri-registry*)
:element-type 'uri-dispatcher
:initial-contents (loop for uri being the hash-values of *uri-registry*
collect uri))
#'uri-dispatcher>)))
(defmacro define-uri-dispatcher (name (uri &optional priority) &body body)
(let ((dispatcher (gensym "DISPATCHER")))
(check-for-shared-symbol name)
`(eval-when (:compile-toplevel :load-toplevel :execute)
(flet ((,dispatcher ()
,@body))
(setf (uri-dispatcher ',name)
(change-class (copy-uri ,uri)
'uri-dispatcher
:name ',name
:dispatch-function #',dispatcher
:priority ,priority
:matcher T
:documentation ,(form-fiddle:lambda-docstring
`(lambda () ,@body))))))))
;; TODO: Continuation based dispatching (CALL-NEXT-DISPATCHER)
;; Only do this if it doesn't slow down the common case.
(defun dispatch (uri)
(declare (optimize speed))
(loop for dispatcher across *uri-priority*
when (uri-matches uri dispatcher)
do (with-simple-restart (abort-handling "Abort the current handler and continue dispatching.")
(return (funcall (dispatch-function dispatcher))))
finally (return (funcall *uri-fallback*))))
(defun abort-handling ()
(invoke-restart 'abort-handling))