Support passive mode in FTP.

This commit is contained in:
sperber 2003-01-16 13:32:43 +00:00
parent ec42abd6db
commit 54ece65f35
2 changed files with 84 additions and 63 deletions

View File

@ -256,32 +256,33 @@
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;: connection [ x string ] -> status ;;: connection [ x string ] -> status
(define (ftp-ls connection . maybe-dir) (define (ftp-ls connection . maybe-dir)
(let* ((sock (ftp-open-data-connection connection))) (with-data-connection
(ftp-send-command connection connection
(apply build-command "NLST" maybe-dir) (lambda ()
(code-with-prefix "1")) (ftp-send-command connection
(receive (newsock newsockaddr) (apply build-command "NLST" maybe-dir)
(accept-connection sock) (code-with-prefix "1")))
(let ((lines (port->lines (socket:inport newsock)))) (lambda (data-socket)
(close-socket newsock) (port->lines (socket:inport data-socket)))))
(close-socket sock)
(ftp-read-reply connection) (define (find-port-arg string)
lines)))) (cond
((regexp-search (rx (: (+ digit) (= 5 (: #\, (+ digit))))) string)
=> (lambda (match)
(match:substring match 0)))))
;;: connection [ x string ] -> status ;;: connection [ x string ] -> status
(define (ftp-dir connection . maybe-dir) (define (ftp-dir connection . maybe-dir)
(let* ((sock (ftp-open-data-connection connection))) (with-data-connection
(ftp-send-command connection connection
(apply build-command "LIST" maybe-dir) (lambda ()
(code-with-prefix "1")) (ftp-send-command connection
(receive (newsock newsockaddr) (apply build-command "LIST" maybe-dir)
(accept-connection sock) (code-with-prefix "1")))
(let ((lines (port->lines (socket:inport newsock)))) (lambda (data-socket)
(close-socket newsock) (port->lines (socket:inport data-socket)))))
(close-socket sock)
(ftp-read-reply connection)
lines))))
(define (port->lines port) (define (port->lines port)
(let loop ((reverse-lines '())) (let loop ((reverse-lines '()))
@ -291,16 +292,14 @@
(loop (cons line reverse-lines)))))) (loop (cons line reverse-lines))))))
(define (ftp-get connection remote-file act) (define (ftp-get connection remote-file act)
(let* ((sock (ftp-open-data-connection connection))) (with-data-connection
(ftp-send-command connection connection
(build-command "RETR" remote-file) (lambda ()
(exactly-code "150")) (ftp-send-command connection
(receive (newsock newsockaddr) (build-command "RETR" remote-file)
(accept-connection sock) (exactly-code "150")))
(act (socket:inport newsock)) (lambda (data-socket)
(close-socket newsock) (act (socket:inport data-socket)))))
(close-socket sock)
(ftp-read-reply connection))))
;; FIXME: should have an optional argument :rename which defaults to ;; FIXME: should have an optional argument :rename which defaults to
;; false, which would make us upload to a temporary name and rename at ;; false, which would make us upload to a temporary name and rename at
@ -309,24 +308,22 @@
;; space on device". ;; space on device".
(define (ftp-put connection remote-file act) (define (ftp-put connection remote-file act)
(let ((sock (ftp-open-data-connection connection))) (with-data-connection
(ftp-send-command connection (build-command "STOR" remote-file) connection
(exactly-code "150")) (lambda ()
(receive (newsock newsockaddr) (ftp-send-command connection (build-command "STOR" remote-file)
(accept-connection sock) (exactly-code "150")))
(act (socket:outport newsock)) (lambda (data-socket)
(close-socket newsock) (act (socket:outport data-socket)))))
(close-socket sock))))
(define (ftp-append connection remote-file act) (define (ftp-append connection remote-file act)
(let ((sock (ftp-open-data-connection connection))) (with-data-connection
(ftp-send-command connection (build-command "APPE" remote-file) connection
(exactly-code "150")) (lambda ()
(receive (newsock newsockaddr) (ftp-send-command connection (build-command "APPE" remote-file)
(accept-connection sock) (exactly-code "150"))
(act (socket:outport newsock)) (lambda (data-socket)
(close-socket newsock) (act (socket:outport data-socket))))))
(close-socket sock))))
;; send a command verbatim to the remote server and wait for a ;; send a command verbatim to the remote server and wait for a
;; reply. ;; reply.
@ -338,21 +335,44 @@
;; ------------------------------------------------------------------------ ;; ------------------------------------------------------------------------
;; no exported procedures below ;; no exported procedures below
(define (ftp-open-data-connection connection) (define (with-data-connection connection command-thunk proc)
(let* ((sock (create-socket protocol-family/internet (if (ftp-connection-passive-mode? connection)
socket-type/stream)) (let* ((pasv-reply (ftp-send-command connection "PASV" (exactly-code "227")))
(sockaddr (internet-address->socket-address (port-arg (find-port-arg pasv-reply)))
internet-address/any (call-with-values
0))) ; 0 to accept any port (lambda () (parse-port-arg port-arg))
(set-socket-option sock level/socket socket/reuse-address #t) (lambda (address port)
(set-socket-option sock level/socket socket/linger 120) (let ((data-socket (create-socket protocol-family/internet
(bind-socket sock sockaddr) socket-type/stream)))
(listen-socket sock 0) (set-socket-option data-socket level/socket socket/reuse-address #t)
(ftp-send-command connection ; send PORT command (connect-socket data-socket
(ftp-build-PORT-string (socket-local-address sock))) (internet-address->socket-address
sock)) address port))
(command-thunk)
(let ((retval (proc data-socket)))
(close-socket data-socket)
(ftp-read-reply connection)
retval)))))
(let* ((sock (create-socket protocol-family/internet
socket-type/stream))
(sockaddr (internet-address->socket-address
internet-address/any
0))) ; 0 to accept any port
(set-socket-option sock level/socket socket/reuse-address #t)
(set-socket-option sock level/socket socket/linger 120)
(bind-socket sock sockaddr)
(listen-socket sock 0)
(ftp-send-command connection ; send PORT command
(ftp-build-PORT-string (socket-local-address sock)))
(command-thunk)
(receive (data-socket data-socket-address)
(accept-connection sock)
(let ((retval (proc data-socket)))
(close-socket data-socket)
(close-socket sock)
(ftp-read-reply connection)
retval)))))
;; TODO: Unix-specific commands ;; TODO: Unix-specific commands
;; SITE UMASK 002 ;; SITE UMASK 002

View File

@ -427,7 +427,8 @@
let-opt let-opt
sunet-utilities sunet-utilities
format-net format-net
crlf-io) crlf-io
ftp-library)
(files (lib ftp))) (files (lib ftp)))
(define-structure netrc netrc-interface (define-structure netrc netrc-interface