url-auth fix for password-protected urls in url-handler-mode.
[bpt/emacs.git] / lisp / url / url-auth.el
CommitLineData
8c8b8430 1;;; url-auth.el --- Uniform Resource Locator authorization modules
00eef4de 2
acaf905b 3;; Copyright (C) 1996-1999, 2004-2012 Free Software Foundation, Inc.
00eef4de 4
8c8b8430
SM
5;; Keywords: comm, data, processes, hypermedia
6
00eef4de
LH
7;; This file is part of GNU Emacs.
8
4936186e 9;; GNU Emacs is free software: you can redistribute it and/or modify
00eef4de 10;; it under the terms of the GNU General Public License as published by
4936186e
GM
11;; the Free Software Foundation, either version 3 of the License, or
12;; (at your option) any later version.
00eef4de
LH
13
14;; GNU Emacs is distributed in the hope that it will be useful,
15;; but WITHOUT ANY WARRANTY; without even the implied warranty of
16;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17;; GNU General Public License for more details.
18
19;; You should have received a copy of the GNU General Public License
4936186e 20;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
00eef4de
LH
21
22;;; Code:
8c8b8430
SM
23
24(require 'url-vars)
25(require 'url-parse)
26(autoload 'url-warn "url")
563790b6 27(autoload 'auth-source-search "auth-source")
97d1c236 28
8c8b8430
SM
29(defsubst url-auth-user-prompt (url realm)
30 "String to usefully prompt for a username."
31 (concat "Username [for "
32 (or realm (url-truncate-url-for-viewing
33 (url-recreate-url url)
34 (- (window-width) 10 20)))
35 "]: "))
36
37;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
38;;; Basic authorization code
39;;; ------------------------
40;;; This implements the BASIC authorization type. See the online
41;;; documentation at
42;;; http://www.w3.org/hypertext/WWW/AccessAuthorization/Basic.html
43;;; for the complete documentation on this type.
44;;;
45;;; This is very insecure, but it works as a proof-of-concept
46;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
47(defvar url-basic-auth-storage 'url-http-real-basic-auth-storage
48 "Where usernames and passwords are stored.
49
50Must be a symbol pointing to another variable that will actually store
51the information. The value of this variable is an assoc list of assoc
52lists. The first assoc list is keyed by the server name. The cdr of
d1ce47b0 53this is an assoc list based on the 'directory' specified by the URL we
8c8b8430
SM
54are looking up.")
55
56(defun url-basic-auth (url &optional prompt overwrite realm args)
57 "Get the username/password for the specified URL.
58If optional argument PROMPT is non-nil, ask for the username/password
59to use for the url and its descendants. If optional third argument
60OVERWRITE is non-nil, overwrite the old username/password pair if it
61is found in the assoc list. If REALM is specified, use that as the realm
ff51e86d 62instead of the filename inheritance method."
8c8b8430
SM
63 (let* ((href (if (stringp url)
64 (url-generic-parse-url url)
65 url))
66 (server (url-host href))
97d1c236 67 (type (url-type href))
8c8b8430 68 (port (url-port href))
ff51e86d 69 (file (url-filename href))
2540a3ab
VJL
70 (user (url-user href))
71 (pass (url-password href))
d8d469ef 72 (enable-recursive-minibuffers t) ; for url-handler-mode (bug#10298)
2540a3ab 73 byserv retval data)
8c8b8430 74 (setq server (format "%s:%d" server port)
ff51e86d 75 file (cond
8c8b8430 76 (realm realm)
ff51e86d
RS
77 ((string= "" file) "/")
78 ((string-match "/$" file) file)
79 (t (url-file-directory file)))
8c8b8430
SM
80 byserv (cdr-safe (assoc server
81 (symbol-value url-basic-auth-storage))))
82 (cond
83 ((and prompt (not byserv))
8705576e 84 (setq user (or
563790b6 85 (url-do-auth-source-search server type :user)
97d1c236
TZ
86 (read-string (url-auth-user-prompt url realm)
87 (or user (user-real-login-name))))
8705576e 88 pass (or
563790b6 89 (url-do-auth-source-search server type :secret)
97d1c236 90 (read-passwd "Password: " nil (or pass ""))))
8c8b8430
SM
91 (set url-basic-auth-storage
92 (cons (list server
ff51e86d 93 (cons file
8c8b8430
SM
94 (setq retval
95 (base64-encode-string
ab8dad36
CY
96 (format "%s:%s" user
97 (encode-coding-string pass 'utf-8))))))
8c8b8430
SM
98 (symbol-value url-basic-auth-storage))))
99 (byserv
ff51e86d 100 (setq retval (cdr-safe (assoc file byserv)))
8c8b8430 101 (if (and (not retval)
ff51e86d 102 (string-match "/" file))
8c8b8430
SM
103 (while (and byserv (not retval))
104 (setq data (car (car byserv)))
d7c2974d 105 (if (or (not (string-match "/" data)) ; It's a realm - take it!
8c8b8430 106 (and
ff51e86d
RS
107 (>= (length file) (length data))
108 (string= data (substring file 0 (length data)))))
8c8b8430
SM
109 (setq retval (cdr (car byserv))))
110 (setq byserv (cdr byserv))))
111 (if (or (and (not retval) prompt) overwrite)
112 (progn
8705576e 113 (setq user (or
563790b6 114 (url-do-auth-source-search server type :user)
97d1c236
TZ
115 (read-string (url-auth-user-prompt url realm)
116 (user-real-login-name)))
8705576e 117 pass (or
563790b6 118 (url-do-auth-source-search server type :secret)
97d1c236 119 (read-passwd "Password: "))
8c8b8430
SM
120 retval (base64-encode-string (format "%s:%s" user pass))
121 byserv (assoc server (symbol-value url-basic-auth-storage)))
122 (setcdr byserv
ff51e86d 123 (cons (cons file retval) (cdr byserv))))))
8c8b8430
SM
124 (t (setq retval nil)))
125 (if retval (setq retval (concat "Basic " retval)))
126 retval))
127
128;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
129;;; Digest authorization code
130;;; ------------------------
131;;; This implements the DIGEST authorization type. See the internet draft
132;;; ftp://ds.internic.net/internet-drafts/draft-ietf-http-digest-aa-01.txt
133;;; for the complete documentation on this type.
134;;;
135;;; This is very secure
136;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
137(defvar url-digest-auth-storage nil
d1ce47b0
JB
138 "Where usernames and passwords are stored.
139Its value is an assoc list of assoc lists. The first assoc list is
140keyed by the server name. The cdr of this is an assoc list based
141on the 'directory' specified by the url we are looking up.")
8c8b8430
SM
142
143(defun url-digest-auth-create-key (username password realm method uri)
144 "Create a key for digest authentication method"
145 (let* ((info (if (stringp uri)
146 (url-generic-parse-url uri)
147 uri))
148 (a1 (md5 (concat username ":" realm ":" password)))
149 (a2 (md5 (concat method ":" (url-filename info)))))
150 (list a1 a2)))
151
152(defun url-digest-auth (url &optional prompt overwrite realm args)
153 "Get the username/password for the specified URL.
154If optional argument PROMPT is non-nil, ask for the username/password
d1ce47b0 155to use for the URL and its descendants. If optional third argument
8c8b8430
SM
156OVERWRITE is non-nil, overwrite the old username/password pair if it
157is found in the assoc list. If REALM is specified, use that as the realm
158instead of hostname:portnum."
159 (if args
160 (let* ((href (if (stringp url)
161 (url-generic-parse-url url)
162 url))
163 (server (url-host href))
97d1c236 164 (type (url-type href))
8c8b8430 165 (port (url-port href))
ff51e86d 166 (file (url-filename href))
8c8b8430 167 user pass byserv retval data)
ff51e86d 168 (setq file (cond
8c8b8430 169 (realm realm)
ff51e86d
RS
170 ((string-match "/$" file) file)
171 (t (url-file-directory file)))
8c8b8430
SM
172 server (format "%s:%d" server port)
173 byserv (cdr-safe (assoc server url-digest-auth-storage)))
174 (cond
175 ((and prompt (not byserv))
97d1c236 176 (setq user (or
563790b6 177 (url-do-auth-source-search server type :user)
97d1c236
TZ
178 (read-string (url-auth-user-prompt url realm)
179 (user-real-login-name)))
180 pass (or
563790b6 181 (url-do-auth-source-search server type :secret)
97d1c236 182 (read-passwd "Password: "))
8c8b8430
SM
183 url-digest-auth-storage
184 (cons (list server
ff51e86d 185 (cons file
8c8b8430
SM
186 (setq retval
187 (cons user
188 (url-digest-auth-create-key
189 user pass realm
190 (or url-request-method "GET")
191 url)))))
192 url-digest-auth-storage)))
193 (byserv
ff51e86d 194 (setq retval (cdr-safe (assoc file byserv)))
8c8b8430 195 (if (and (not retval) ; no exact match, check directories
ff51e86d 196 (string-match "/" file)) ; not looking for a realm
8c8b8430
SM
197 (while (and byserv (not retval))
198 (setq data (car (car byserv)))
199 (if (or (not (string-match "/" data))
200 (and
ff51e86d
RS
201 (>= (length file) (length data))
202 (string= data (substring file 0 (length data)))))
8c8b8430
SM
203 (setq retval (cdr (car byserv))))
204 (setq byserv (cdr byserv))))
e652840b
JW
205 (if overwrite
206 (if (and (not retval) prompt)
8705576e 207 (setq user (or
563790b6 208 (url-do-auth-source-search server type :user)
97d1c236
TZ
209 (read-string (url-auth-user-prompt url realm)
210 (user-real-login-name)))
8705576e 211 pass (or
563790b6 212 (url-do-auth-source-search server type :secret)
97d1c236 213 (read-passwd "Password: "))
e652840b
JW
214 retval (setq retval
215 (cons user
216 (url-digest-auth-create-key
217 user pass realm
218 (or url-request-method "GET")
219 url)))
220 byserv (assoc server url-digest-auth-storage))
8c8b8430 221 (setcdr byserv
ff51e86d 222 (cons (cons file retval) (cdr byserv))))))
8c8b8430
SM
223 (t (setq retval nil)))
224 (if retval
e652840b
JW
225 (if (cdr-safe (assoc "opaque" args))
226 (let ((nonce (or (cdr-safe (assoc "nonce" args)) "nonegiven"))
227 (opaque (cdr-safe (assoc "opaque" args))))
228 (format
229 (concat "Digest username=\"%s\", realm=\"%s\","
230 "nonce=\"%s\", uri=\"%s\","
231 "response=\"%s\", opaque=\"%s\"")
232 (nth 0 retval) realm nonce (url-filename href)
233 (md5 (concat (nth 1 retval) ":" nonce ":"
234 (nth 2 retval))) opaque))
235 (let ((nonce (or (cdr-safe (assoc "nonce" args)) "nonegiven")))
236 (format
237 (concat "Digest username=\"%s\", realm=\"%s\","
238 "nonce=\"%s\", uri=\"%s\","
239 "response=\"%s\"")
240 (nth 0 retval) realm nonce (url-filename href)
241 (md5 (concat (nth 1 retval) ":" nonce ":"
242 (nth 2 retval))))))))))
8c8b8430
SM
243
244(defvar url-registered-auth-schemes nil
245 "A list of the registered authorization schemes and various and sundry
246information associated with them.")
247
563790b6
TZ
248(defun url-do-auth-source-search (server type parameter)
249 (let* ((auth-info (auth-source-search :max 1 :host server :port type))
250 (auth-info (nth 0 auth-info))
251 (token (plist-get auth-info parameter))
252 (token (if (functionp token) (funcall token) token)))
253 token))
254
8c8b8430
SM
255;;;###autoload
256(defun url-get-authentication (url realm type prompt &optional args)
257 "Return an authorization string suitable for use in the WWW-Authenticate
258header in an HTTP/1.0 request.
259
260URL is the url you are requesting authorization to. This can be either a
261 string representing the URL, or the parsed representation returned by
262 `url-generic-parse-url'
263REALM is the realm at a specific site we are looking for. This should be a
264 string specifying the exact realm, or nil or the symbol 'any' to
265 specify that the filename portion of the URL should be used as the
266 realm
267TYPE is the type of authentication to be returned. This is either a string
268 representing the type (basic, digest, etc), or nil or the symbol 'any'
269 to specify that any authentication is acceptable. If requesting 'any'
270 the strongest matching authentication will be returned. If this is
d7c2974d 271 wrong, it's no big deal, the error from the server will specify exactly
8c8b8430
SM
272 what type of auth to use
273PROMPT is boolean - specifies whether to ask the user for a username/password
274 if one cannot be found in the cache"
275 (if (not realm)
276 (setq realm (cdr-safe (assoc "realm" args))))
277 (if (stringp url)
278 (setq url (url-generic-parse-url url)))
279 (if (or (null type) (eq type 'any))
280 ;; Whooo doogies!
281 ;; Go through and get _all_ the authorization strings that could apply
282 ;; to this URL, store them along with the 'rating' we have in the list
283 ;; of schemes, then sort them so that the 'best' is at the front of the
284 ;; list, then get the car, then get the cdr.
285 ;; Zooom zooom zoooooom
286 (cdr-safe
287 (car-safe
288 (sort
289 (mapcar
290 (function
291 (lambda (scheme)
292 (if (fboundp (car (cdr scheme)))
293 (cons (cdr (cdr scheme))
294 (funcall (car (cdr scheme)) url nil nil realm))
295 (cons 0 nil))))
296 url-registered-auth-schemes)
297 (function
298 (lambda (x y)
299 (cond
300 ((null (cdr x)) nil)
301 ((and (cdr x) (null (cdr y))) t)
302 ((and (cdr x) (cdr y))
303 (>= (car x) (car y)))
304 (t nil)))))))
305 (if (symbolp type) (setq type (symbol-name type)))
306 (let* ((scheme (car-safe
307 (cdr-safe (assoc (downcase type)
308 url-registered-auth-schemes)))))
309 (if (and scheme (fboundp scheme))
310 (funcall scheme url prompt
311 (and prompt
312 (funcall scheme url nil nil realm args))
313 realm args)))))
314
315;;;###autoload
316(defun url-register-auth-scheme (type &optional function rating)
317 "Register an HTTP authentication method.
318
d1ce47b0
JB
319TYPE is a string or symbol specifying the name of the method.
320 This should be the same thing you expect to get returned in
321 an Authenticate header in HTTP/1.0 - it will be downcased.
322FUNCTION is the function to call to get the authorization information.
323 This defaults to `url-?-auth', where ? is TYPE.
8c8b8430
SM
324RATING a rating between 1 and 10 of the strength of the authentication.
325 This is used when asking for the best authentication for a specific
326 URL. The item with the highest rating is returned."
327 (let* ((type (cond
328 ((stringp type) (downcase type))
329 ((symbolp type) (downcase (symbol-name type)))
330 (t (error "Bad call to `url-register-auth-scheme'"))))
331 (function (or function (intern (concat "url-" type "-auth"))))
332 (rating (cond
333 ((null rating) 2)
216d3806 334 ((stringp rating) (string-to-number rating))
8c8b8430
SM
335 (t rating)))
336 (node (assoc type url-registered-auth-schemes)))
337 (if (not (fboundp function))
338 (url-warn 'security
caae2fd8
SM
339 (format (concat
340 "Tried to register `%s' as an auth scheme"
341 ", but it is not a function!") function)))
8c8b8430
SM
342
343 (if node
344 (setcdr node (cons function rating))
345 (setq url-registered-auth-schemes
346 (cons (cons type (cons function rating))
347 url-registered-auth-schemes)))))
348
349(defun url-auth-registered (scheme)
3ecd3a56 350 "Return non-nil if SCHEME is registered as an auth type."
8c8b8430
SM
351 (assoc scheme url-registered-auth-schemes))
352
353(provide 'url-auth)
e5566bd5 354
00eef4de 355;;; url-auth.el ends here