mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-13 22:50:23 -05:00
publish: Use 'x-raw-file' internal response header.
This adjusts the workaround for <http://bugs.gnu.org/21093> so that it's not limited to a single content-type. * guix/scripts/publish.scm (render-nar/cached): Add the 'x-raw-file' header on the response. (render-content-addressed-file): Likewise. (with-content-length): Remove the 'x-raw-file' header. (http-write): Instead of dispatching on 'application/octet-stream', check whether 'x-raw-file' is set to determine whether to spawn a thread.
This commit is contained in:
parent
06e3a5181e
commit
152b7beeac
1 changed files with 45 additions and 41 deletions
|
@ -544,11 +544,12 @@ (define* (render-nar/cached store cache request store-item
|
||||||
#:compression compression)))
|
#:compression compression)))
|
||||||
(if (file-exists? cached)
|
(if (file-exists? cached)
|
||||||
(values `((content-type . (application/octet-stream
|
(values `((content-type . (application/octet-stream
|
||||||
(charset . "ISO-8859-1"))))
|
(charset . "ISO-8859-1")))
|
||||||
;; XXX: We're not returning the actual contents, deferring
|
;; XXX: We're not returning the actual contents, deferring
|
||||||
;; instead to 'http-write'. This is a hack to work around
|
;; instead to 'http-write'. This is a hack to work around
|
||||||
;; <http://bugs.gnu.org/21093>.
|
;; <http://bugs.gnu.org/21093>.
|
||||||
cached)
|
(x-raw-file . ,cached))
|
||||||
|
#f)
|
||||||
(not-found request))))
|
(not-found request))))
|
||||||
|
|
||||||
(define (render-content-addressed-file store request
|
(define (render-content-addressed-file store request
|
||||||
|
@ -562,11 +563,12 @@ (define (render-content-addressed-file store request
|
||||||
#:recursive? #f)))
|
#:recursive? #f)))
|
||||||
(if (valid-path? store item)
|
(if (valid-path? store item)
|
||||||
(values `((content-type . (application/octet-stream
|
(values `((content-type . (application/octet-stream
|
||||||
(charset . "ISO-8859-1"))))
|
(charset . "ISO-8859-1")))
|
||||||
;; XXX: We're not returning the actual contents, deferring
|
;; XXX: We're not returning the actual contents,
|
||||||
;; instead to 'http-write'. This is a hack to work around
|
;; deferring instead to 'http-write'. This is a hack to
|
||||||
;; <http://bugs.gnu.org/21093>.
|
;; work around <http://bugs.gnu.org/21093>.
|
||||||
item)
|
(x-raw-file . ,item))
|
||||||
|
#f)
|
||||||
(not-found request)))
|
(not-found request)))
|
||||||
(not-found request)))
|
(not-found request)))
|
||||||
|
|
||||||
|
@ -622,9 +624,9 @@ (define (with-content-length response length)
|
||||||
"Return RESPONSE with a 'content-length' header set to LENGTH."
|
"Return RESPONSE with a 'content-length' header set to LENGTH."
|
||||||
(set-field response (response-headers)
|
(set-field response (response-headers)
|
||||||
(alist-cons 'content-length length
|
(alist-cons 'content-length length
|
||||||
(alist-delete 'content-length
|
(fold alist-delete
|
||||||
(response-headers response)
|
(response-headers response)
|
||||||
eq?))))
|
'(content-length x-raw-file)))))
|
||||||
|
|
||||||
(define-syntax-rule (swallow-EPIPE exp ...)
|
(define-syntax-rule (swallow-EPIPE exp ...)
|
||||||
"Swallow EPIPE errors raised by EXP..."
|
"Swallow EPIPE errors raised by EXP..."
|
||||||
|
@ -685,35 +687,37 @@ (define (http-write server client response body)
|
||||||
(swallow-zlib-error
|
(swallow-zlib-error
|
||||||
(close-port port))
|
(close-port port))
|
||||||
(values)))))
|
(values)))))
|
||||||
(('application/octet-stream . _)
|
|
||||||
;; Send a raw file in a separate thread.
|
|
||||||
(call-with-new-thread
|
|
||||||
(lambda ()
|
|
||||||
(set-thread-name "publish file")
|
|
||||||
(catch 'system-error
|
|
||||||
(lambda ()
|
|
||||||
(call-with-input-file (utf8->string body)
|
|
||||||
(lambda (input)
|
|
||||||
(let* ((size (stat:size (stat input)))
|
|
||||||
(response (write-response (with-content-length response
|
|
||||||
size)
|
|
||||||
client))
|
|
||||||
(output (response-port response)))
|
|
||||||
(if (file-port? output)
|
|
||||||
(sendfile output input size)
|
|
||||||
(dump-port input output))
|
|
||||||
(close-port output)
|
|
||||||
(values)))))
|
|
||||||
(lambda args
|
|
||||||
;; If the file was GC'd behind our back, that's fine. Likewise if
|
|
||||||
;; the client closes the connection.
|
|
||||||
(unless (memv (system-error-errno args)
|
|
||||||
(list ENOENT EPIPE ECONNRESET))
|
|
||||||
(apply throw args))
|
|
||||||
(values))))))
|
|
||||||
(_
|
(_
|
||||||
;; Handle other responses sequentially.
|
(match (assoc-ref (response-headers response) 'x-raw-file)
|
||||||
(%http-write server client response body))))
|
((? string? file)
|
||||||
|
;; Send a raw file in a separate thread.
|
||||||
|
(call-with-new-thread
|
||||||
|
(lambda ()
|
||||||
|
(set-thread-name "publish file")
|
||||||
|
(catch 'system-error
|
||||||
|
(lambda ()
|
||||||
|
(call-with-input-file file
|
||||||
|
(lambda (input)
|
||||||
|
(let* ((size (stat:size (stat input)))
|
||||||
|
(response (write-response (with-content-length response
|
||||||
|
size)
|
||||||
|
client))
|
||||||
|
(output (response-port response)))
|
||||||
|
(if (file-port? output)
|
||||||
|
(sendfile output input size)
|
||||||
|
(dump-port input output))
|
||||||
|
(close-port output)
|
||||||
|
(values)))))
|
||||||
|
(lambda args
|
||||||
|
;; If the file was GC'd behind our back, that's fine. Likewise if
|
||||||
|
;; the client closes the connection.
|
||||||
|
(unless (memv (system-error-errno args)
|
||||||
|
(list ENOENT EPIPE ECONNRESET))
|
||||||
|
(apply throw args))
|
||||||
|
(values))))))
|
||||||
|
(#f
|
||||||
|
;; Handle other responses sequentially.
|
||||||
|
(%http-write server client response body))))))
|
||||||
|
|
||||||
(define-server-impl concurrent-http-server
|
(define-server-impl concurrent-http-server
|
||||||
;; A variant of Guile's built-in HTTP server that offloads possibly long
|
;; A variant of Guile's built-in HTTP server that offloads possibly long
|
||||||
|
|
Loading…
Reference in a new issue