1 ;;; url-auth.el --- Uniform Resource Locator authorization modules
3 ;; Copyright (C) 1996, 1997, 1998, 1999, 2004,
4 ;; 2005, 2006, 2007 Free Software Foundation, Inc.
6 ;; Keywords: comm, data, processes, hypermedia
8 ;; This file is part of GNU Emacs.
10 ;; GNU Emacs is free software; you can redistribute it and/or modify
11 ;; it under the terms of the GNU General Public License as published by
12 ;; the Free Software Foundation; either version 3, or (at your option)
15 ;; GNU Emacs is distributed in the hope that it will be useful,
16 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 ;; GNU General Public License for more details.
20 ;; You should have received a copy of the GNU General Public License
21 ;; along with GNU Emacs; see the file COPYING. If not, write to the
22 ;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
23 ;; Boston, MA 02110-1301, USA.
29 (autoload 'url-warn
"url")
31 (defsubst url-auth-user-prompt
(url realm
)
32 "String to usefully prompt for a username."
33 (concat "Username [for "
34 (or realm
(url-truncate-url-for-viewing
35 (url-recreate-url url
)
36 (- (window-width) 10 20)))
39 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
40 ;;; Basic authorization code
41 ;;; ------------------------
42 ;;; This implements the BASIC authorization type. See the online
44 ;;; http://www.w3.org/hypertext/WWW/AccessAuthorization/Basic.html
45 ;;; for the complete documentation on this type.
47 ;;; This is very insecure, but it works as a proof-of-concept
48 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
49 (defvar url-basic-auth-storage
'url-http-real-basic-auth-storage
50 "Where usernames and passwords are stored.
52 Must be a symbol pointing to another variable that will actually store
53 the information. The value of this variable is an assoc list of assoc
54 lists. The first assoc list is keyed by the server name. The cdr of
55 this is an assoc list based on the 'directory' specified by the url we
58 (defun url-basic-auth (url &optional prompt overwrite realm args
)
59 "Get the username/password for the specified URL.
60 If optional argument PROMPT is non-nil, ask for the username/password
61 to use for the url and its descendants. If optional third argument
62 OVERWRITE is non-nil, overwrite the old username/password pair if it
63 is found in the assoc list. If REALM is specified, use that as the realm
64 instead of the pathname inheritance method."
65 (let* ((href (if (stringp url
)
66 (url-generic-parse-url url
)
68 (server (url-host href
))
69 (port (url-port href
))
70 (path (url-filename href
))
71 (user (url-user href
))
72 (pass (url-password href
))
74 (setq server
(format "%s:%d" server port
)
77 ((string-match "/$" path
) path
)
78 (t (url-basepath path
)))
79 byserv
(cdr-safe (assoc server
80 (symbol-value url-basic-auth-storage
))))
82 ((and prompt
(not byserv
))
83 (setq user
(read-string (url-auth-user-prompt url realm
)
84 (or user
(user-real-login-name)))
85 pass
(read-passwd "Password: " nil
(or pass
"")))
86 (set url-basic-auth-storage
91 (format "%s:%s" user pass
)))))
92 (symbol-value url-basic-auth-storage
))))
94 (setq retval
(cdr-safe (assoc path byserv
)))
96 (string-match "/" path
))
97 (while (and byserv
(not retval
))
98 (setq data
(car (car byserv
)))
99 (if (or (not (string-match "/" data
)) ; It's a realm - take it!
101 (>= (length path
) (length data
))
102 (string= data
(substring path
0 (length data
)))))
103 (setq retval
(cdr (car byserv
))))
104 (setq byserv
(cdr byserv
))))
105 (if (or (and (not retval
) prompt
) overwrite
)
107 (setq user
(read-string (url-auth-user-prompt url realm
)
108 (user-real-login-name))
109 pass
(read-passwd "Password: ")
110 retval
(base64-encode-string (format "%s:%s" user pass
))
111 byserv
(assoc server
(symbol-value url-basic-auth-storage
)))
113 (cons (cons path retval
) (cdr byserv
))))))
114 (t (setq retval nil
)))
115 (if retval
(setq retval
(concat "Basic " retval
)))
118 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
119 ;;; Digest authorization code
120 ;;; ------------------------
121 ;;; This implements the DIGEST authorization type. See the internet draft
122 ;;; ftp://ds.internic.net/internet-drafts/draft-ietf-http-digest-aa-01.txt
123 ;;; for the complete documentation on this type.
125 ;;; This is very secure
126 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
127 (defvar url-digest-auth-storage nil
128 "Where usernames and passwords are stored. Its value is an assoc list of
129 assoc lists. The first assoc list is keyed by the server name. The cdr of
130 this is an assoc list based on the 'directory' specified by the url we are
133 (defun url-digest-auth-create-key (username password realm method uri
)
134 "Create a key for digest authentication method"
135 (let* ((info (if (stringp uri
)
136 (url-generic-parse-url uri
)
138 (a1 (md5 (concat username
":" realm
":" password
)))
139 (a2 (md5 (concat method
":" (url-filename info
)))))
142 (defun url-digest-auth (url &optional prompt overwrite realm args
)
143 "Get the username/password for the specified URL.
144 If optional argument PROMPT is non-nil, ask for the username/password
145 to use for the url and its descendants. If optional third argument
146 OVERWRITE is non-nil, overwrite the old username/password pair if it
147 is found in the assoc list. If REALM is specified, use that as the realm
148 instead of hostname:portnum."
150 (let* ((href (if (stringp url
)
151 (url-generic-parse-url url
)
153 (server (url-host href
))
154 (port (url-port href
))
155 (path (url-filename href
))
156 user pass byserv retval data
)
159 ((string-match "/$" path
) path
)
160 (t (url-basepath path
)))
161 server
(format "%s:%d" server port
)
162 byserv
(cdr-safe (assoc server url-digest-auth-storage
)))
164 ((and prompt
(not byserv
))
165 (setq user
(read-string (url-auth-user-prompt url realm
)
166 (user-real-login-name))
167 pass
(read-passwd "Password: ")
168 url-digest-auth-storage
173 (url-digest-auth-create-key
175 (or url-request-method
"GET")
177 url-digest-auth-storage
)))
179 (setq retval
(cdr-safe (assoc path byserv
)))
180 (if (and (not retval
) ; no exact match, check directories
181 (string-match "/" path
)) ; not looking for a realm
182 (while (and byserv
(not retval
))
183 (setq data
(car (car byserv
)))
184 (if (or (not (string-match "/" data
))
186 (>= (length path
) (length data
))
187 (string= data
(substring path
0 (length data
)))))
188 (setq retval
(cdr (car byserv
))))
189 (setq byserv
(cdr byserv
))))
190 (if (or (and (not retval
) prompt
) overwrite
)
192 (setq user
(read-string (url-auth-user-prompt url realm
)
193 (user-real-login-name))
194 pass
(read-passwd "Password: ")
197 (url-digest-auth-create-key
199 (or url-request-method
"GET")
201 byserv
(assoc server url-digest-auth-storage
))
203 (cons (cons path retval
) (cdr byserv
))))))
204 (t (setq retval nil
)))
206 (let ((nonce (or (cdr-safe (assoc "nonce" args
)) "nonegiven"))
207 (opaque (or (cdr-safe (assoc "opaque" args
)) "nonegiven")))
209 (concat "Digest username=\"%s\", realm=\"%s\","
210 "nonce=\"%s\", uri=\"%s\","
211 "response=\"%s\", opaque=\"%s\"")
212 (nth 0 retval
) realm nonce
(url-filename href
)
213 (md5 (concat (nth 1 retval
) ":" nonce
":"
214 (nth 2 retval
))) opaque
))))))
216 (defvar url-registered-auth-schemes nil
217 "A list of the registered authorization schemes and various and sundry
218 information associated with them.")
221 (defun url-get-authentication (url realm type prompt
&optional args
)
222 "Return an authorization string suitable for use in the WWW-Authenticate
223 header in an HTTP/1.0 request.
225 URL is the url you are requesting authorization to. This can be either a
226 string representing the URL, or the parsed representation returned by
227 `url-generic-parse-url'
228 REALM is the realm at a specific site we are looking for. This should be a
229 string specifying the exact realm, or nil or the symbol 'any' to
230 specify that the filename portion of the URL should be used as the
232 TYPE is the type of authentication to be returned. This is either a string
233 representing the type (basic, digest, etc), or nil or the symbol 'any'
234 to specify that any authentication is acceptable. If requesting 'any'
235 the strongest matching authentication will be returned. If this is
236 wrong, it's no big deal, the error from the server will specify exactly
237 what type of auth to use
238 PROMPT is boolean - specifies whether to ask the user for a username/password
239 if one cannot be found in the cache"
241 (setq realm
(cdr-safe (assoc "realm" args
))))
243 (setq url
(url-generic-parse-url url
)))
244 (if (or (null type
) (eq type
'any
))
246 ;; Go through and get _all_ the authorization strings that could apply
247 ;; to this URL, store them along with the 'rating' we have in the list
248 ;; of schemes, then sort them so that the 'best' is at the front of the
249 ;; list, then get the car, then get the cdr.
250 ;; Zooom zooom zoooooom
257 (if (fboundp (car (cdr scheme
)))
258 (cons (cdr (cdr scheme
))
259 (funcall (car (cdr scheme
)) url nil nil realm
))
261 url-registered-auth-schemes
)
266 ((and (cdr x
) (null (cdr y
))) t
)
267 ((and (cdr x
) (cdr y
))
268 (>= (car x
) (car y
)))
270 (if (symbolp type
) (setq type
(symbol-name type
)))
271 (let* ((scheme (car-safe
272 (cdr-safe (assoc (downcase type
)
273 url-registered-auth-schemes
)))))
274 (if (and scheme
(fboundp scheme
))
275 (funcall scheme url prompt
277 (funcall scheme url nil nil realm args
))
281 (defun url-register-auth-scheme (type &optional function rating
)
282 "Register an HTTP authentication method.
284 TYPE is a string or symbol specifying the name of the method. This
285 should be the same thing you expect to get returned in an Authenticate
286 header in HTTP/1.0 - it will be downcased.
287 FUNCTION is the function to call to get the authorization information. This
288 defaults to `url-?-auth', where ? is TYPE
289 RATING a rating between 1 and 10 of the strength of the authentication.
290 This is used when asking for the best authentication for a specific
291 URL. The item with the highest rating is returned."
293 ((stringp type
) (downcase type
))
294 ((symbolp type
) (downcase (symbol-name type
)))
295 (t (error "Bad call to `url-register-auth-scheme'"))))
296 (function (or function
(intern (concat "url-" type
"-auth"))))
299 ((stringp rating
) (string-to-number rating
))
301 (node (assoc type url-registered-auth-schemes
)))
302 (if (not (fboundp function
))
305 "Tried to register `%s' as an auth scheme"
306 ", but it is not a function!") function
)))
309 (setcdr node
(cons function rating
))
310 (setq url-registered-auth-schemes
311 (cons (cons type
(cons function rating
))
312 url-registered-auth-schemes
)))))
314 (defun url-auth-registered (scheme)
315 "Return non-nil if SCHEME is registered as an auth type."
316 (assoc scheme url-registered-auth-schemes
))
320 ;;; arch-tag: 04058625-616d-44e4-9dbf-4b46b00b2a91
321 ;;; url-auth.el ends here