mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-15 23:48:07 -05:00
download: Add timeout parameter for connections.
* guix/build/download.scm (ensure-uri): New procedure. (current-http-proxy): New variable. (open-socket-for-uri): Copy from Guile commit aaea5b2, but add #:timeout parameter and use 'connect*' instead of 'connect'. (open-connection-for-uri): Add #:timeout parameter and pass it to 'open-socket-for-uri'.
This commit is contained in:
parent
4856700698
commit
60fd51222f
1 changed files with 58 additions and 18 deletions
|
@ -20,6 +20,7 @@
|
||||||
|
|
||||||
(define-module (guix build download)
|
(define-module (guix build download)
|
||||||
#:use-module (web uri)
|
#:use-module (web uri)
|
||||||
|
#:use-module (web http)
|
||||||
#:use-module ((web client) #:hide (open-socket-for-uri))
|
#:use-module ((web client) #:hide (open-socket-for-uri))
|
||||||
#:use-module (web response)
|
#:use-module (web response)
|
||||||
#:use-module (guix ftp-client)
|
#:use-module (guix ftp-client)
|
||||||
|
@ -277,26 +278,65 @@ (define (log level str)
|
||||||
(add-weak-reference record port)
|
(add-weak-reference record port)
|
||||||
record)))
|
record)))
|
||||||
|
|
||||||
(define (open-socket-for-uri uri)
|
(define (ensure-uri uri-or-string) ;XXX: copied from (web http)
|
||||||
"Return an open port for URI. This variant works around
|
(cond
|
||||||
<http://bugs.gnu.org/15368> which affects Guile's 'open-socket-for-uri' up to
|
((string? uri-or-string) (string->uri uri-or-string))
|
||||||
2.0.11 included."
|
((uri? uri-or-string) uri-or-string)
|
||||||
(define rmem-max
|
(else (error "Invalid URI" uri-or-string))))
|
||||||
;; The maximum size for a receive buffer on Linux, see socket(7).
|
|
||||||
"/proc/sys/net/core/rmem_max")
|
|
||||||
|
|
||||||
(define buffer-size
|
(define current-http-proxy
|
||||||
(if (file-exists? rmem-max)
|
;; XXX: Add a dummy definition for Guile < 2.0.10; this is used in
|
||||||
(call-with-input-file rmem-max read)
|
;; 'open-socket-for-uri'.
|
||||||
126976)) ;the default for Linux, per 'rmem_default'
|
(or (and=> (module-variable (resolve-interface '(web client))
|
||||||
|
'current-http-proxy)
|
||||||
|
variable-ref)
|
||||||
|
(const #f)))
|
||||||
|
|
||||||
(let ((s ((@ (web client) open-socket-for-uri) uri)))
|
(define* (open-socket-for-uri uri-or-string #:key timeout)
|
||||||
;; Work around <http://bugs.gnu.org/15368> by restoring a decent
|
"Return an open input/output port for a connection to URI. When TIMEOUT is
|
||||||
;; buffer size.
|
not #f, it must be a (possibly inexact) number denoting the maximum duration
|
||||||
(setsockopt s SOL_SOCKET SO_RCVBUF buffer-size)
|
in seconds to wait for the connection to complete; passed TIMEOUT, an
|
||||||
s))
|
ETIMEDOUT error is raised."
|
||||||
|
;; Includes a fix for <http://bugs.gnu.org/15368> which affects Guile's
|
||||||
|
;; 'open-socket-for-uri' up to 2.0.11 included, and uses 'connect*' instead
|
||||||
|
;; of 'connect'.
|
||||||
|
|
||||||
(define (open-connection-for-uri uri)
|
(define http-proxy (current-http-proxy))
|
||||||
|
(define uri (ensure-uri (or http-proxy uri-or-string)))
|
||||||
|
(define addresses
|
||||||
|
(let ((port (uri-port uri)))
|
||||||
|
(delete-duplicates
|
||||||
|
(getaddrinfo (uri-host uri)
|
||||||
|
(cond (port => number->string)
|
||||||
|
(else (symbol->string (uri-scheme uri))))
|
||||||
|
(if port
|
||||||
|
AI_NUMERICSERV
|
||||||
|
0))
|
||||||
|
(lambda (ai1 ai2)
|
||||||
|
(equal? (addrinfo:addr ai1) (addrinfo:addr ai2))))))
|
||||||
|
|
||||||
|
(let loop ((addresses addresses))
|
||||||
|
(let* ((ai (car addresses))
|
||||||
|
(s (with-fluids ((%default-port-encoding #f))
|
||||||
|
;; Restrict ourselves to TCP.
|
||||||
|
(socket (addrinfo:fam ai) SOCK_STREAM IPPROTO_IP))))
|
||||||
|
(catch 'system-error
|
||||||
|
(lambda ()
|
||||||
|
(connect* s (addrinfo:addr ai) timeout)
|
||||||
|
|
||||||
|
;; Buffer input and output on this port.
|
||||||
|
(setvbuf s _IOFBF)
|
||||||
|
;; If we're using a proxy, make a note of that.
|
||||||
|
(when http-proxy (set-http-proxy-port?! s #t))
|
||||||
|
s)
|
||||||
|
(lambda args
|
||||||
|
;; Connection failed, so try one of the other addresses.
|
||||||
|
(close s)
|
||||||
|
(if (null? (cdr addresses))
|
||||||
|
(apply throw args)
|
||||||
|
(loop (cdr addresses))))))))
|
||||||
|
|
||||||
|
(define* (open-connection-for-uri uri #:key timeout)
|
||||||
"Like 'open-socket-for-uri', but also handle HTTPS connections."
|
"Like 'open-socket-for-uri', but also handle HTTPS connections."
|
||||||
(define https?
|
(define https?
|
||||||
(eq? 'https (uri-scheme uri)))
|
(eq? 'https (uri-scheme uri)))
|
||||||
|
@ -319,7 +359,7 @@ (define https?
|
||||||
(thunk))
|
(thunk))
|
||||||
(thunk)))))))
|
(thunk)))))))
|
||||||
(with-https-proxy
|
(with-https-proxy
|
||||||
(let ((s (open-socket-for-uri uri)))
|
(let ((s (open-socket-for-uri uri #:timeout timeout)))
|
||||||
;; Buffer input and output on this port.
|
;; Buffer input and output on this port.
|
||||||
(setvbuf s _IOFBF %http-receive-buffer-size)
|
(setvbuf s _IOFBF %http-receive-buffer-size)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue