* in netrc.scm:
- changes to make Eric's code fit our desires: . functional concept (netrc:parse returns a record that has to be carried with) . usual errors (no such file, no such machine) are caught and handled adequately (nevertheless the user can get the errors, if he wants to) . netrc:parse works when no ~/.netrc file exists . default-values (file-name, login, password) can be set while calling netrc:parse . netrc:lookup, netrc:lookup-password, netrc:lookup-login return default-values by default (can be switched off) - removed bug in netrc:lookup * in modules.scm: new export in netrc-interface: netrc:try-parse (returns error if file does not exist) * in ecm-utilities: rewrote nslookup-fqdn: uses host-info now
This commit is contained in:
parent
75633864c3
commit
89bb20c63e
|
@ -1,6 +1,6 @@
|
|||
;; ecm-utilities.scm -- Utility procedures for ecm-net code
|
||||
;;
|
||||
;; $Id: ecm-utilities.scm,v 1.2 2001/11/15 11:12:23 mainzelm Exp $
|
||||
;; $Id: ecm-utilities.scm,v 1.3 2001/12/18 18:08:08 interp Exp $
|
||||
;;
|
||||
;; Please send suggestions and bug reports to <emarsden@mail.dotcom.fr>
|
||||
|
||||
|
@ -12,12 +12,19 @@
|
|||
sysname
|
||||
(nslookup-fqdn))))
|
||||
|
||||
;; This doesn't work on my system. Probably it is not configured well.
|
||||
;; Nevertheless, the alternative seems better to me
|
||||
;(define (nslookup-fqdn)
|
||||
; (let* ((cmd (format #f "nslookup ~a" (system-name)))
|
||||
; (raw (string-join (run/strings (nslookup ,(system-name)))))
|
||||
; (match (string-match "Name: +([-a-zA-Z0-9.]+)" raw)))
|
||||
; (display raw)
|
||||
; (match:substring match 1)))
|
||||
|
||||
(define (nslookup-fqdn)
|
||||
(let* ((cmd (format #f "nslookup ~a" (system-name)))
|
||||
(raw (string-join (run/strings (nslookup ,(system-name)))))
|
||||
(match (string-match "Name: +([-a-zA-Z0-9.]+)" raw)))
|
||||
(display raw)
|
||||
(match:substring match 1)))
|
||||
(host-info:name (host-info (system-name))))
|
||||
; another easy alternative:
|
||||
; (car (run/strings (hostname "--long"))))
|
||||
|
||||
|
||||
;; prefer this to :optional
|
||||
|
|
|
@ -529,19 +529,21 @@
|
|||
|
||||
(define-interface netrc-interface
|
||||
(export user-mail-address
|
||||
netrc:default-login
|
||||
netrc:default-password
|
||||
netrc:lookup
|
||||
netrc:lookup-password
|
||||
netrc:lookup-login
|
||||
netrc:parse))
|
||||
netrc:parse
|
||||
netrc:try-parse))
|
||||
|
||||
(define-structure netrc netrc-interface
|
||||
(open defrec-package
|
||||
records
|
||||
scsh
|
||||
error-package
|
||||
ecm-utilities
|
||||
string-lib
|
||||
conditions signals
|
||||
let-opt
|
||||
scheme)
|
||||
(files netrc))
|
||||
|
||||
|
|
264
netrc.scm
264
netrc.scm
|
@ -1,6 +1,6 @@
|
|||
;;; netrc.scm -- parse authentication information contained in ~/.netrc
|
||||
;;
|
||||
;; $Id: netrc.scm,v 1.2 2001/11/15 11:12:24 mainzelm Exp $
|
||||
;; $Id: netrc.scm,v 1.3 2001/12/18 18:08:08 interp Exp $
|
||||
;;
|
||||
;; Please send suggestions and bug reports to <emarsden@mail.dotcom.fr>
|
||||
|
||||
|
@ -18,7 +18,17 @@
|
|||
;;
|
||||
;; The ~/.netrc file should be protected by appropriate permissions,
|
||||
;; and (like /usr/bin/ftp) this library will refuse to read the file if
|
||||
;; it is badly protected.
|
||||
;; it is badly protected. (unlike /usr/bin/ftp this library will always
|
||||
;; refuse to read the file -- /usr/bin/ftp refuses it only if the password
|
||||
;; is given for a non-default account). Appropriate permissions are set
|
||||
;; if only the user has permissions on the file.
|
||||
;;
|
||||
;; Note following restrictions / differences:
|
||||
;; * The macdef statement (defining macros) is not supported.
|
||||
;; * The settings for one machine must be on a single line.
|
||||
;; * The is no error proof while reading the file.
|
||||
;; * default must not be the last line of the netrc-file
|
||||
|
||||
|
||||
|
||||
;;; Entry points =======================================================
|
||||
|
@ -28,15 +38,21 @@
|
|||
;; the same name. Will take into account the environment variable
|
||||
;; REPLYTO, if set.
|
||||
;;
|
||||
;; (netrc:default-login) -> string | #f
|
||||
;; Return the default login specified by the ~/.netrc file, or #f.
|
||||
;; (netrc:default-login) -> string
|
||||
;; Return the default login specified by the ~/.netrc file or "anonymous"
|
||||
;;
|
||||
;; (netrc:default-password) -> string | #f
|
||||
;; Return the default password specified by the ~/.netrc file, or #f.
|
||||
;; (netrc:default-password) -> string
|
||||
;; Return the default password specified by the ~/.netrc file or
|
||||
;; to the mail-addres (result of (user-mail-address))
|
||||
;;
|
||||
;; (netrc:lookup machine) -> string x string x string
|
||||
;; Return the login,password,account information for MACHINE
|
||||
;; specified by the ~/.netrc file.
|
||||
;; If there is no such machine specified, the condition 'netrc:no-such-entry
|
||||
;; is signalled, that can be caught with with-handler.
|
||||
;; If you catch this signal, you probably want to use netrc:default-login
|
||||
;; and netrc:default-password, that are always set to usable values.
|
||||
|
||||
|
||||
|
||||
;;; Related work ========================================================
|
||||
|
@ -51,119 +67,221 @@
|
|||
;;; Portability ==================================================
|
||||
;;
|
||||
;; getenv, scsh file primitives, regexp code, format
|
||||
;; define-record
|
||||
;; define-record, ecm-utilities
|
||||
|
||||
|
||||
;;; Desirable things =============================================
|
||||
;;
|
||||
;; * Remove restrictions (as stated in 'Overview') and behave like
|
||||
;; /usr/bin/ftp behaves
|
||||
;; * perhaps: adding case-insensitivity
|
||||
;; * perhaps: better record-disclosers for netrc-entry- and netrc-records
|
||||
|
||||
(define (user-mail-address)
|
||||
(or (getenv "REPLYTO")
|
||||
(string-append (user-login-name) "@" (system-fqdn))))
|
||||
|
||||
(define (netrc:default-login) *netrc:default-login*)
|
||||
(define (netrc:default-password) *netrc:default-password*)
|
||||
|
||||
;;: string -> string x string x string
|
||||
(define (netrc:lookup machine)
|
||||
(let ((record
|
||||
(find-suchthat (lambda (rec)
|
||||
(and (equal? (netrc:machine rec) machine)
|
||||
(list (netrc:login rec)
|
||||
(netrc:password rec)
|
||||
(netrc:account rec))))
|
||||
*netrc*)))
|
||||
(values (netrc:login record)
|
||||
(netrc:password record)
|
||||
(netrc:account record))))
|
||||
; looks up the desired machine in a netrc-record
|
||||
; if the machine is found in the entries-section
|
||||
; following three values are returned: login, password and account
|
||||
; if the machine is not found in the entries-section
|
||||
; the behavior depends on lookup-default? which defaults to #t:
|
||||
; if lookup-default? is #t
|
||||
; following three values are returned: default-login default-password #f
|
||||
; otherwise #f #f #f is returned.
|
||||
(define (netrc:lookup netrc-record machine . lookup-default?)
|
||||
(let-optionals lookup-default?
|
||||
((lookup-default? #t))
|
||||
(let ((record (find-record netrc-record machine)))
|
||||
(if record
|
||||
(values (netrc-entry:login record)
|
||||
(netrc-entry:password record)
|
||||
(netrc-entry:account record))
|
||||
(and lookup-default?
|
||||
(values (netrc:default-login netrc-record)
|
||||
(netrc:default-password netrc-record)
|
||||
#f)
|
||||
(values #f #f #f))))))
|
||||
|
||||
(define (netrc:lookup-password machine)
|
||||
(receive (login password account)
|
||||
(netrc:lookup machine)
|
||||
password))
|
||||
; does the same as netrc:lookup, but returns only the password (or #f)
|
||||
(define (netrc:lookup-password netrc-record machine . lookup-default?)
|
||||
(let-optionals lookup-default?
|
||||
((lookup-default? #t))
|
||||
(let ((record (find-record netrc-record machine)))
|
||||
(if record
|
||||
(netrc-entry:password record)
|
||||
(and lookup-default?
|
||||
(netrc:default-password netrc-record))))))
|
||||
|
||||
(define (netrc:lookup-login machine)
|
||||
(receive (login password account)
|
||||
(netrc:lookup machine)
|
||||
login))
|
||||
; does the same as netrc:lookup, but returns only the login (or #f)
|
||||
(define (netrc:lookup-login netrc-record machine . lookup-default?)
|
||||
(let-optionals lookup-default?
|
||||
((lookup-default? #t))
|
||||
(let ((record (find-record netrc-record machine)))
|
||||
(if record
|
||||
(netrc-entry:login record)
|
||||
(and lookup-default?
|
||||
(netrc:default-login netrc-record))))))
|
||||
|
||||
; does the work for netrc:parse
|
||||
; file-name has to be resolved
|
||||
(define (netrc:try-parse file-name default-password default-login)
|
||||
(netrc:check-permissions file-name)
|
||||
(let ((fd (open-input-file file-name))
|
||||
(netrc-record (make-netrc '() default-password default-login file-name)))
|
||||
(for-each-line (parse-line netrc-record) fd)))
|
||||
|
||||
; parses the netrc-file
|
||||
; expected arguments: filename default-password default-login
|
||||
; filename: filename of the .netrc-file (defaults to ~/.netrc)
|
||||
; default-password: default password for any not specified machine
|
||||
; defaults to (user-mail-address)
|
||||
; default password in netrc-file overwrites this setting
|
||||
; default-login: default login name for any not specified machine
|
||||
; defaults to "anonymous"
|
||||
; default login in netrc-file overwrites this setting
|
||||
; (default-login is expected after default-password as users usually want
|
||||
; to change the default-password (to something else than their mail-address)
|
||||
; rather than the login-name)(define (netrc:parse . args)
|
||||
; if the given file does not exist, than a default netrc-record is returned
|
||||
; if you don't want this, use netrc:try-parse; note that you have to
|
||||
; resolve the file-name on your own
|
||||
(define (netrc:parse . args)
|
||||
(let-optionals
|
||||
args ((file-name "~/.netrc")
|
||||
(default-password #f) ; both ...
|
||||
(default-login #f)) ; ... are set if netrc-file does not provide default-values
|
||||
(let ((file-name (resolve-file-name file-name)))
|
||||
(call-with-current-continuation
|
||||
(lambda (exit)
|
||||
(with-errno-handler*
|
||||
(lambda (errno packet)
|
||||
(if (= errno errno/noent)
|
||||
(exit (make-netrc '()
|
||||
(or default-login "anonymous")
|
||||
(or default-password (user-mail-address))
|
||||
#f))))
|
||||
(lambda ()
|
||||
(let ((netrc-record (netrc:try-parse file-name default-password default-login)))
|
||||
(if netrc-record
|
||||
netrc-record
|
||||
(begin
|
||||
(set-netrc:default-password netrc-record (or default-password (user-mail-address)))
|
||||
(set-netrc:default-login netrc-record (or default-login "anonymous"))
|
||||
netrc-record))))))))))
|
||||
|
||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||
;; nothing exported below
|
||||
|
||||
(define-record netrc
|
||||
(define-record netrc-entry
|
||||
machine
|
||||
login
|
||||
password
|
||||
account)
|
||||
|
||||
(define *netrc* '())
|
||||
(define *netrc:default-login* "anonymous")
|
||||
(define *netrc:default-password* (user-mail-address))
|
||||
(define *netrc:file* (resolve-file-name "~/.netrc"))
|
||||
(define-record netrc
|
||||
entries ; list of netrc-entrys
|
||||
default-login ; default-values (either library-default or netrc-file-default)
|
||||
default-password
|
||||
file-name) ; debug-purpose
|
||||
|
||||
|
||||
(define (netrc:parse)
|
||||
(netrc:check-permissions)
|
||||
(set! *netrc* '())
|
||||
(let ((fd (open-input-file *netrc:file*)))
|
||||
(for-each-line netrc:parse-line fd)))
|
||||
(define-record-discloser type/netrc-entry
|
||||
(lambda (netrc-entry)
|
||||
(list 'netrc-entry))) ; perhaps something else later on
|
||||
|
||||
(define-record-discloser type/netrc
|
||||
(lambda (netrc)
|
||||
(list 'netrc))) ; perhaps something else later on
|
||||
|
||||
; finds a record in the entries-list of a netrc-record
|
||||
; matching the given machine
|
||||
; returns the netrc-entry-record if found, otherwise #f
|
||||
(define (find-record netrc-record machine)
|
||||
(find-first (lambda (rec)
|
||||
(and (equal? (netrc-entry:machine rec) machine)
|
||||
rec))
|
||||
(netrc:entries netrc-record)))
|
||||
|
||||
|
||||
;; raise error if any permissions are set for group or others.
|
||||
(define (netrc:check-permissions)
|
||||
(let ((perms (- (file-mode *netrc:file*) 32768)))
|
||||
(define (netrc:check-permissions file-name)
|
||||
(let ((perms (- (file-mode file-name) 32768)))
|
||||
(if (positive? (bitwise-and #b000111111 perms))
|
||||
(error "Not parsing ~/.netrc file; dangerous permissions"))))
|
||||
(error "Not parsing ~/.netrc file; dangerous permissions."))))
|
||||
|
||||
(define (netrc:try-match target line)
|
||||
; tries to match target on line and returns the first group,
|
||||
; or #f if there is no match
|
||||
(define (try-match target line)
|
||||
(let ((match (string-match target line)))
|
||||
(and match
|
||||
(match:substring match 1))))
|
||||
|
||||
(define (netrc:parse-default line)
|
||||
(let ((login (netrc:try-match "login[ \t]+([^ \t]+)" line))
|
||||
(password (netrc:try-match "password[ \t]+([^ \t]+)" line)))
|
||||
; parses the default line of the netrc-file
|
||||
(define (parse-default netrc-record line)
|
||||
(let ((login (try-match "login[ \t]+([^ \t]+)" line))
|
||||
(password (try-match "password[ \t]+([^ \t]+)" line)))
|
||||
(if login
|
||||
(set! *netrc:default-login* login))
|
||||
(set-netrc:default-login netrc-record login))
|
||||
(if password
|
||||
(set! *netrc:default-password* password))))
|
||||
(set-netrc:default-password netrc-record password))
|
||||
netrc-record))
|
||||
|
||||
(define (netrc:parse-line line)
|
||||
; parses a line of the netrc-file
|
||||
(define (parse-line netrc-record)
|
||||
(lambda (line)
|
||||
(cond ((string-match "default" line)
|
||||
(netrc:parse-default line))
|
||||
(parse-default netrc-record line))
|
||||
(else
|
||||
(let ((machine (netrc:try-match "machine[ \t]+([^ \t]+)" line))
|
||||
(login (netrc:try-match "login[ \t]+([^ \t]+)" line))
|
||||
(password (netrc:try-match "password[ \t]+([^ \t]+)" line))
|
||||
(account (netrc:try-match "account[ \t]+([^ \t]+)" line)))
|
||||
(let ((machine (try-match "machine[ \t]+([^ \t]+)" line))
|
||||
(login (try-match "login[ \t]+([^ \t]+)" line))
|
||||
(password (try-match "password[ \t]+([^ \t]+)" line))
|
||||
(account (try-match "account[ \t]+([^ \t]+)" line)))
|
||||
(if (or machine login password account)
|
||||
(netrc:add machine login password account))))))
|
||||
(add netrc-record machine login password account)))))))
|
||||
|
||||
(define (netrc:add machine login password account)
|
||||
(set! *netrc* (cons (make-netrc machine login password account) *netrc*)))
|
||||
; adds machine login password account stored in a netrc-entry-record
|
||||
; to the entries-list of a netrc-record
|
||||
(define (add netrc-record machine login password account)
|
||||
(set-netrc:entries netrc-record
|
||||
(cons (make-netrc-entry machine login password account)
|
||||
(netrc:entries netrc-record)))
|
||||
netrc-record)
|
||||
|
||||
;; for testing
|
||||
(define (netrc:dump)
|
||||
(format #t "~%--- Dumping ~~/.netrc contents ---")
|
||||
(define (netrc:dump netrc-record)
|
||||
(format #t "~%--- Dumping ~s contents ---" (netrc:file-name netrc-record))
|
||||
(for-each (lambda (rec)
|
||||
(format #t "~% machine ~a login ~a password ~a account ~a"
|
||||
(netrc:machine rec)
|
||||
(netrc:login rec)
|
||||
(netrc:password rec)
|
||||
(netrc:account rec)))
|
||||
*netrc*)
|
||||
(format #t "~%--- End of ~~/.netrc contents ---~%"))
|
||||
(netrc-entry:machine rec)
|
||||
(netrc-entry:login rec)
|
||||
(netrc-entry:password rec)
|
||||
(netrc-entry:account rec)))
|
||||
(netrc:entries netrc-record))
|
||||
(format #t "~% default login: ~s" (netrc:default-login netrc-record))
|
||||
(format #t "~% default password: ~s" (netrc:default-password netrc-record))
|
||||
(format #t "~%--- End of ~s contents ---~%" (netrc:file-name netrc-record)))
|
||||
|
||||
|
||||
; runs proc for each line of fd (line is argument to proc)
|
||||
; returns either nothing, if the fd had no line
|
||||
; or the value returned by proc called on the last line
|
||||
(define (for-each-line proc fd)
|
||||
(let ((line (read-line fd)))
|
||||
(and (not (eof-object? line))
|
||||
(proc line)
|
||||
(for-each-line proc fd))))
|
||||
(if (not (eof-object? line))
|
||||
(let loop ((last-result (proc line)))
|
||||
(let ((line (read-line fd)))
|
||||
(if (not (eof-object? line))
|
||||
(loop (proc line))
|
||||
last-result))))))
|
||||
|
||||
(define (find-suchthat pred l)
|
||||
; finds first element in l for which pred doesn't return #f
|
||||
; returns either #f (no such element found)
|
||||
; or the result of the last call to pred
|
||||
(define (find-first pred l)
|
||||
(if (null? l) #f
|
||||
(or (pred (car l))
|
||||
(find-suchthat pred (cdr l)))))
|
||||
|
||||
; do we need this here?
|
||||
;(netrc:parse)
|
||||
(find-first pred (cdr l)))))
|
||||
|
||||
;; EOF
|
||||
|
|
Loading…
Reference in New Issue