File: threaded-http-listener.lisp

package info (click to toggle)
araneida 0.90.1-dfsg-2
  • links: PTS
  • area: main
  • in suites: etch, etch-m68k
  • size: 700 kB
  • ctags: 643
  • sloc: lisp: 4,878; perl: 166; sh: 109; makefile: 34
file content (97 lines) | stat: -rw-r--r-- 3,767 bytes parent folder | download | duplicates (2)
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
90
91
92
93
94
95
96
97
(in-package :araneida)

;;; There is still work to be done abstracting out the commonality
;;; between this file and serve-event-http-listener.lisp

(defvar *handler-timeout* 60
  "Maximum number of seconds to spend on a request.  Applies both to
normal processing and to time spent in the debugger, should there be
an unhandled error")

(defun add-listener-thread (listener)
  (let ((thread (make-http-thread :last-hit 0)))
    (setf (http-thread-pid thread)
	  (host-run-thread
           'listener-thread
	   (lambda ()
	     (loop
	      (threaded-http-listener-accept-one-request listener thread)))))
    thread))

(defun threaded-http-listener-accept-one-request (listener thread)
  (handler-case
      (with-accept-flets 
	  (let ((s (accept listener)))
	    (host-with-timeout *handler-timeout*
			       (setf (http-thread-last-hit thread) nil)
			       (unwind-protect
				    (do-it listener s)
				 (setf (http-thread-last-hit thread) (get-universal-time))
				 (forcibly-close-stream s)))))
    (end-of-file () (let ((r (find-restart 'abort-response)))
		      (when r
			(invoke-restart r))))))

(defmethod start-listening ((listener threaded-http-listener)
			    &key (threads 5))
  (let ((socket
	 (host-make-listener-socket (http-listener-address listener)
                                    (http-listener-port listener))))
    (setf (http-listener-socket listener) socket)
    (dotimes (i threads)
      (push (add-listener-thread listener) (http-listener-threads listener)))
    (setf (http-listener-thread listener)
	  (host-run-thread
           'master-thread
	   (lambda ()
	     (loop (master-thread-one-iter listener)))))))

(defun master-thread-one-iter (listener)
  (let ((min (http-listener-min-spare listener))
	(max (http-listener-max-spare listener))
	(spares 0))
    ;; loop over the children.  count 1 for every stale thread
    ;; when spare-count > max, kill stale threads as they're
    ;; encountered when we get to the end, if spare-count < min,
    ;; start another if no change required, sleep a bit
    (dolist (this (http-listener-threads listener))
      (when (numberp (http-thread-last-hit this))
	(cond ((not (host-thread-alivep (http-thread-pid this)))
	       (format t ";; Thread!  Is!  Dead!  Ahaaaa (~A)~%"
		       (http-thread-pid this))
	       (setf (http-thread-pid this) nil))
	      ((> max spares)
	       #+nil
	       (format t "thread ~a last used ~A, ~A spare so far~%"
		       (http-thread-pid this)
		       (http-thread-last-hit this) spares)
	       (incf spares))
	      (t (format t ";; thread ~A last used ~A, killing~%"
			 (http-thread-pid this)
			 (http-thread-last-hit this))
		 (host-thread-kill (http-thread-pid this))
		 (setf (http-thread-pid this) nil)))))
    (setf (http-listener-threads listener)
	  (remove-if #'null (http-listener-threads listener)
		     :key #'http-thread-pid))
    (when (< spares min)
      (dotimes (i (- min spares))
	(format t ";; ~A spare threads < ~A, adding another~%" spares min)
	(push (add-listener-thread listener)
	      (http-listener-threads listener))))
    (when (<= min spares max)		; no change
      ;; (format t "no change ~%")
      (sleep 60))
    (sleep 1)))
	    
(defmethod stop-listening ((listener threaded-http-listener)
			   &key abort &allow-other-keys)
  (declare (ignore abort))		;FIXME we always abort.  not very nice
  (dolist (thread (http-listener-threads listener))
    (when (host-thread-alivep (http-thread-pid thread))
      (host-thread-kill (http-thread-pid thread))))
  (awhen (http-listener-thread listener) (host-thread-kill it))
  (host-close-socket (http-listener-socket listener))
  (setf (http-listener-threads listener) nil
	(http-listener-thread listener) nil
        (http-listener-socket listener) nil))