chibi-scheme/lib/chibi/net/http.scm

306 lines
11 KiB
Scheme

;; http.scm -- http client
;; Copyright (c) 2009-2017 Alex Shinn. All rights reserved.
;; BSD-style license: http://synthcode.com/license.txt
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; string utils
(define (string-char-index str c . o)
(let ((start (if (pair? o) (car o) 0))
(end (if (and (pair? o) (pair? (cdr o))) (cadr o) (string-length str))))
(let lp ((i start))
(cond
((= i end) #f)
((eq? c (string-ref str i)) i)
(else (lp (+ i 1)))))))
(define (string-split str ch)
(let ((len (string-length str)))
(let lp ((i 0) (res '()))
(let ((j (string-char-index str ch i)))
(if j
(lp (+ j 1) (cons (substring str i j) res))
(reverse (cons (substring str i len) res)))))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; client utils
(define http-user-agent "chibi")
(define http-redirect-limit 10)
(define http-chunked-buffer-size 4096)
(define http-chunked-size-limit 409600)
(define (string-scan str ch . o)
(let ((start (if (pair? o) (car o) 0))
(end (if (and (pair? o) (pair? (cdr o))) (cadr o) (string-length str))))
(let lp ((i start))
(and (< i end)
(if (eqv? ch (string-ref str i))
i
(lp (+ i 1)))))))
(define (http-parse-response line)
(let* ((len (string-length line))
(i (or (string-scan line #\space 0 len) len))
(j (or (string-scan line #\space (+ i 1) len) len))
(n (and (< i j) (string->number (substring line (+ i 1) j)))))
(if (not (integer? n))
(error "bad response" line i j)
(list (substring line 0 i)
n
(if (>= j len) "" (substring line (+ j 1) len))))))
(define (http-wrap-chunked-input-port in)
(define (read-chunk in)
(let* ((line (read-line in))
(n (and (string? line) (string->number line 16))))
(cond
((not (and (integer? n) (<= 0 n http-chunked-size-limit)))
(error "invalid chunked size line" line))
((zero? n) "")
(else (read-bytevector n in)))))
(make-generated-binary-input-port
(lambda () (read-chunk in))))
(define (http-generate-boundary)
(string-append "chibi-boundary-"
(number->string (random-integer 1000000000))))
;; A body can be a raw string or bytevector, or an alist of name/value
;; pairs.
(define (http-send-body headers body out)
(cond
((string? body)
(display body out))
((bytevector? body)
(write-bytevector body out))
((pair? body)
(let* ((ctype (cond ((or (assq 'Content-Type headers)
(assq 'content-type headers))
=> (lambda (x)
(mime-parse-content-type (cdr x))))
(else #f)))
(boundary (and ctype (assq-ref (cdr ctype) 'boundary))))
(for-each
(lambda (x)
(display "\r\n--" out)
(if boundary (display boundary out))
(let* ((content
(if (pair? (cdr x))
(cond ((assq 'value (cdr x)) => cdr)
((assq 'file (cdr x)) =>
(lambda (x)
(port->bytevector
(open-binary-input-file (cdr x)))))
(else (error "unknown content: " x)))
(cdr x)))
(content-type
(cond ((and (pair? (cdr x))
(or (assq 'content-type (cdr x))
(assq 'Content-Type (cdr x))))
=> cdr)
((string? content) "text/plain")
(else "application/octet-stream"))))
(display "\r\nContent-Disposition: form-data; name=\"" out)
(display (car x) out)
(display "\"" out)
(cond ((and (pair? (cdr x)) (assq 'file (cdr x)))
=> (lambda (x)
(display "; filename=\"" out)
(display (cdr x) out)
(display "\"" out))))
(display "\r\nContent-Type: " out)
(display content-type out)
(display "\r\n\r\n" out)
(http-send-body headers content out)))
body)
(display "\r\n--" out)
(if boundary (display boundary out))
(display "--\r\n" out)))
(body
(error "unknown body" body))))
(define (http-call-method method url in-headers body limit)
(if (<= limit 0)
(error "http-get: redirect limit reached" (uri->string url))
(let* ((uri (if (uri? url) url (string->uri url)))
(host (and uri (uri-host uri))))
(if (not host)
(error "invalid url" url)
(let* ((io (open-net-io
host
(or (uri-port uri)
(if (eq? 'https (uri-scheme uri)) 443 80))
(assq-ref in-headers 'blocking)))
(in (cadr io))
(out (car (cddr io))))
(display method out)
(display " " out)
(display (or (uri-path uri) "/") out)
(cond
((uri-query uri)
(display "?" out)
(display (uri-query uri) out)))
(display " HTTP/1.0\r\n" out)
(display "Host: " out) (display host out) (display "\r\n" out)
(cond
((not (assq-ref in-headers 'user-agent))
(display "User-Agent: " out)
(display http-user-agent out)
(display "\r\n" out)))
(for-each
(lambda (x)
(cond
((not (eq? 'blocking (car x)))
(display (car x) out) (display ": " out)
(display (cdr x) out) (display "\r\n" out))))
in-headers)
(display "Connection: close\r\n\r\n" out)
(http-send-body in-headers body out)
(flush-output-port out)
(let* ((resp (http-parse-response (read-line in)))
(headers (mime-headers->list in))
(status (quotient (cadr resp) 100)))
(case status
((2)
(let ((enc (assq-ref headers 'transfer-encoding)))
(cond
((equal? enc "chunked")
(cons headers (http-wrap-chunked-input-port in)))
(else
(cons headers in)))))
((3)
(close-input-port in)
(close-output-port out)
(let ((url2 (assq-ref headers 'location)))
(if url2
(http-get/raw url2 in-headers (- limit 1))
(error "redirect with no location header" url url2))))
(else
(close-input-port in)
(close-output-port out)
(error "couldn't retrieve url" (uri->string url) resp)))))))))
(define (http-get/raw url headers limit)
(http-call-method 'GET url headers #f limit))
(define (http-get/headers url . headers)
(http-get/raw url
(if (pair? headers) (car headers) '())
http-redirect-limit))
(define (http-get url . headers)
(cdr (apply http-get/headers url headers)))
(define (http-head url . headers)
(car (http-call-method 'HEAD url
(if (pair? headers) (car headers) '()) #f
http-redirect-limit)))
(define (http-post url body . o)
(let* ((headers (if (pair? o) (car o) '()))
(headers
(if (or (assq 'content-type headers)
(assq 'Content-Type headers))
headers
(let ((boundary (http-generate-boundary)))
`((Content-Type . ,(string-append
"multipart/form-data; boundary="
boundary))
,@headers))))
(body
(let ((out (open-output-bytevector)))
(http-send-body headers body out)
(get-output-bytevector out)))
(headers
(if (or (assq 'content-length headers)
(assq 'Content-Length headers))
headers
`((Content-Length . ,(bytevector-length body))
,@headers))))
(cdr (http-call-method 'POST url headers body http-redirect-limit))))
(define (http-put url body . headers)
(cdr (http-call-method 'PUT url
(if (pair? headers) (car headers) '()) body
http-redirect-limit)))
(define (http-delete url . headers)
(cdr (http-call-method 'DELETE url
(if (pair? headers) (car headers) '()) #f
http-redirect-limit)))
(define (call-with-input-url url proc)
(let* ((p (http-get url))
(res (proc p)))
(close-input-port p)
res))
(define (call-with-input-url/headers url proc)
(let* ((h+p (http-get/headers url))
(res (proc (car h+p) (cdr h+p))))
(close-input-port (cdr h+p))
res))
(define (with-input-from-url url thunk)
(let ((p (http-get url)))
(let ((res (parameterize ((current-input-port p)) (thunk))))
(close-input-port p)
res)))
(define (http-get-to-file url path)
(call-with-input-url url
(lambda (in)
(let ((out (open-binary-output-file path)))
(let lp ()
(let ((c (read-u8 in)))
(cond ((not (eof-object? c))
(write-u8 c out)
(lp)))))))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; server utils
;;> Read and parse a request line.
(define (http-parse-request . o)
(let ((line (string-split
(read-line (if (pair? o) (car o) (current-input-port)) 4096))))
(cons (string->symbol (car line)) (cdr line))))
;;> Parse a form body with a given URI and MIME headers (as parsed
;;> with \scheme{mime-headers->list}). Returns an alist of
;;> \scheme{(name . value)} for every query or form parameter.
(define (http-parse-form uri headers . o)
(let* ((in (if (pair? o) (car o) (current-input-port)))
(type (assq-ref headers
'content-type
"application/x-www-form-urlencoded"))
(query0 (or (uri-query (if (string? uri) (string->uri uri) uri)) '()))
(query (if (string? query0) (uri-query->alist query0) query0)))
(cond
((and (>= (string-length type) 10)
(string-ci=? "multipart/" (substring type 0 10)))
(let ((mime (mime-message->sxml in headers)))
(append
(let lp ((ls (cddr mime))
(res '()))
(cond
((null? ls)
res)
((and (pair? (car ls))
(eq? 'mime (caar ls))
(pair? (cdar ls))
(pair? (car (cdar ls)))
(memq (caar (cdar ls)) '(^ @)))
(let* ((disp0 (assq-ref (cdar (cdar ls)) 'content-disposition ""))
(disp (mime-parse-content-type disp0))
(name (assq-ref disp 'name)))
(if name
(lp (cdr ls) (cons (cons name (cadr (cdar ls))) res))
(lp (cdr ls) res))))
(else
(lp (cdr ls) res))))
query)))
(else
query))))