summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMathieu Othacehe <othacehe@gnu.org>2020-10-03 10:14:33 +0200
committerMathieu Othacehe <othacehe@gnu.org>2020-10-03 10:14:33 +0200
commitaf84a00818b53d80c1360ea7ed7eeb6e1e3bb4c6 (patch)
tree77365ca2e24dadf6a305bcf90aff59e8bbeadeca
parentd9c98705033de7b6c1a65d57ce1192bb25c68ac9 (diff)
downloadcuirass-af84a00818b53d80c1360ea7ed7eeb6e1e3bb4c6.tar
cuirass-af84a00818b53d80c1360ea7ed7eeb6e1e3bb4c6.tar.gz
Add a timeout on database worker query send.
The number of database worker is limited, and can introduce some starvation. Print a message each time a two seconds timeout expires while sending an SQL query to a worker. * src/cuirass/utils.scm (put-message-with-timeout): New procedure. (call-with-worker-thread): Rename timeout and timeout-proc arguments to receive-timeout and receive-timeout-proc arguments. Add two new arguments: send-timeout and send-timeout-proc. Use put-message-with-timeout to honor them. * src/cuirass/database.scm (with-db-worker-thread): Adapt accordingly and pass send-timeout and send-timeout-proc arguments.
-rw-r--r--src/cuirass/database.scm15
-rw-r--r--src/cuirass/utils.scm64
2 files changed, 65 insertions, 14 deletions
diff --git a/src/cuirass/database.scm b/src/cuirass/database.scm
index c904375..cc705cb 100644
--- a/src/cuirass/database.scm
+++ b/src/cuirass/database.scm
@@ -200,18 +200,25 @@ specified."
"Evaluate EXP... in the critical section corresponding to %DB-CHANNEL.
DB is bound to the argument of that critical section: the database
connection."
- (let ((timeout 5)
+ (let ((send-timeout 2)
+ (receive-timeout 5)
(caller-name (frame-procedure-name
(stack-ref (make-stack #t) 1))))
(call-with-worker-thread
(%db-channel)
(lambda (db) exp ...)
- #:timeout timeout
- #:timeout-proc
+ #:send-timeout send-timeout
+ #:send-timeout-proc
+ (lambda ()
+ (log-message
+ (format #f "No available database workers for ~a seconds."
+ (number->string send-timeout))))
+ #:receive-timeout receive-timeout
+ #:receive-timeout-proc
(lambda ()
(log-message
(format #f "Database worker unresponsive for ~a seconds (~a)."
- (number->string timeout)
+ (number->string receive-timeout)
caller-name))))))
(define-syntax-rule (with-db-registration-worker-thread db exp ...)
diff --git a/src/cuirass/utils.scm b/src/cuirass/utils.scm
index 7ce4b83..8eb0ed2 100644
--- a/src/cuirass/utils.scm
+++ b/src/cuirass/utils.scm
@@ -39,6 +39,7 @@
with-timeout
get-message-with-timeout
+ put-message-with-timeout
make-worker-thread-channel
call-with-worker-thread
@@ -163,23 +164,66 @@ it succeeds."
res)))
res)))
+
+(define* (put-message-with-timeout channel message
+ #:key
+ seconds
+ (retry? #t)
+ timeout-proc)
+ "Perform a put-operation sending MESSAGE on CHANNEL with a timeout set to
+SECONDS. If the timout expires and RETRY? is set to false, return 'timeout.
+If RETRY is true, call the TIMEOUT-PROC procedure on timeout and retry the
+put-operation until it succeeds."
+ (define (put-message*)
+ (perform-operation
+ (with-timeout
+ (wrap-operation (put-operation channel message) (const #t))
+ #:seconds seconds
+ #:wrap (const 'timeout))))
+
+ (let ((res (put-message*)))
+ (if retry?
+ (begin
+ (let loop ((res res))
+ (if (eq? res 'timeout)
+ (begin
+ (and timeout-proc (timeout-proc))
+ (loop (put-message*)))
+ res)))
+ res)))
+
(define* (call-with-worker-thread channel proc
#:key
- timeout
- timeout-proc)
+ send-timeout
+ send-timeout-proc
+ receive-timeout
+ receive-timeout-proc)
"Send PROC to the worker thread through CHANNEL. Return the result of PROC.
-If already in the worker thread, call PROC immediately. If TIMEOUT is set to
-a duration in seconds, TIMEOUT-PROC is called every time a delay of TIMEOUT
-seconds expires, without a response from the worker thread."
+If already in the worker thread, call PROC immediately.
+
+If SEND-TIMEOUT is set to a duration in seconds, SEND-TIMEOUT-PROC is called
+every time a delay of SEND-TIMEOUT seconds expires, when trying to send PROC
+to a worker thread.
+
+The same goes for RECEIVE-TIMEOUT and RECEIVE-TIMEOUT-PROC, except that the
+timer expires if there is no response from the database worker PROC was sent
+to."
(let ((args (%worker-thread-args)))
(if args
(apply proc args)
- (let ((reply (make-channel)))
- (put-message channel (cons reply proc))
- (match (if (and timeout (current-fiber))
+ (let* ((reply (make-channel))
+ (message (cons reply proc)))
+ (if (and send-timeout (current-fiber))
+ (put-message-with-timeout channel message
+ #:seconds send-timeout
+ #:timeout-proc send-timeout-proc)
+ (put-message channel message))
+ (match (if (and receive-timeout (current-fiber))
(get-message-with-timeout reply
- #:seconds timeout
- #:timeout-proc timeout-proc)
+ #:seconds
+ receive-timeout
+ #:timeout-proc
+ receive-timeout-proc)
(get-message reply))
(('worker-thread-error key args ...)
(apply throw key args))