1 ;;; url-expand.el --- expand-file-name for URLs
2 ;; Keywords: comm, data, processes
4 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
5 ;;; Copyright (c) 1999 Free Software Foundation, Inc.
7 ;;; This file is part of GNU Emacs.
9 ;;; GNU Emacs is free software; you can redistribute it and/or modify
10 ;;; it under the terms of the GNU General Public License as published by
11 ;;; the Free Software Foundation; either version 2, or (at your option)
12 ;;; any later version.
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.
19 ;;; You should have received a copy of the GNU General Public License
20 ;;; along with GNU Emacs; see the file COPYING. If not, write to the
21 ;;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
22 ;;; Boston, MA 02111-1307, USA.
23 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
25 (require 'url-methods
)
29 (defun url-expander-remove-relative-links (name)
30 ;; Strip . and .. from pathnames
31 (let ((new (if (not (string-match "^/" name
))
35 ;; If it ends with a '/.' or '/..', tack on a trailing '/' sot hat
36 ;; the tests that follow are not too complicated in terms of
37 ;; looking for '..' or '../', etc.
38 (if (string-match "/\\.+$" new
)
39 (setq new
(concat new
"/")))
42 (while (string-match "/\\(\\./\\)" new
)
43 (setq new
(concat (substring new
0 (match-beginning 1))
44 (substring new
(match-end 1)))))
47 (while (string-match "/\\([^/]*/\\.\\./\\)" new
)
48 (setq new
(concat (substring new
0 (match-beginning 1))
49 (substring new
(match-end 1)))))
51 ;; Remove cruft at the beginning of the string, so people that put
52 ;; in extraneous '..' because they are morons won't lose.
53 (while (string-match "^/\\.\\.\\(/\\)" new
)
54 (setq new
(substring new
(match-beginning 1) nil
)))
57 (defun url-expand-file-name (url &optional default
)
58 "Convert URL to a fully specified URL, and canonicalize it.
59 Second arg DEFAULT is a URL to start with if URL is relative.
60 If DEFAULT is nil or missing, the current buffer's URL is used.
61 Path components that are `.' are removed, and
62 path components followed by `..' are removed, along with the `..' itself."
63 (if (and url
(not (string-match "^#" url
)))
64 ;; Need to nuke newlines and spaces in the URL, or we open
65 ;; ourselves up to potential security holes.
66 (setq url
(mapconcat (function (lambda (x)
67 (if (memq x
'(? ?
\n ?
\r))
72 ;; Need to figure out how/where to expand the fragment relative to
75 ;; Default URL has already been parsed
78 ;; They gave us a default URL in non-parsed format
79 (url-generic-parse-url default
))
81 ;; We are in a URL-based buffer, use the pre-parsed object
83 ((string-match url-nonrelative-link url
)
84 ;; The URL they gave us is absolute, go for it.
87 ;; Hmmm - this shouldn't ever happen.
88 (error "url-expand-file-name confused - no default?"))))
91 ((= (length url
) 0) ; nil or empty string
92 (url-recreate-url default
))
93 ((string-match "^#" url
) ; Offset link, use it raw
95 ((string-match url-nonrelative-link url
) ; Fully-qualified URL, return it immediately
98 (let* ((urlobj (url-generic-parse-url url
))
99 (inhibit-file-name-handlers t
)
100 (expander (url-scheme-get-property (url-type default
) 'expand-file-name
)))
101 (if (string-match "^//" url
)
102 (setq urlobj
(url-generic-parse-url (concat (url-type default
) ":"
104 (funcall expander urlobj default
)
105 (url-recreate-url urlobj
)))))
107 (defun url-identity-expander (urlobj defobj
)
108 (url-set-type urlobj
(or (url-type urlobj
) (url-type defobj
))))
110 (defun url-default-expander (urlobj defobj
)
111 ;; The default expansion routine - urlobj is modified by side effect!
112 (if (url-type urlobj
)
113 ;; Well, they told us the scheme, let's just go with it.
115 (url-set-type urlobj
(or (url-type urlobj
) (url-type defobj
)))
116 (url-set-port urlobj
(or (url-port urlobj
)
117 (and (string= (url-type urlobj
)
120 (if (not (string= "file" (url-type urlobj
)))
121 (url-set-host urlobj
(or (url-host urlobj
) (url-host defobj
))))
122 (if (string= "ftp" (url-type urlobj
))
123 (url-set-user urlobj
(or (url-user urlobj
) (url-user defobj
))))
124 (if (string= (url-filename urlobj
) "")
125 (url-set-filename urlobj
"/"))
126 (if (string-match "^/" (url-filename urlobj
))
131 (if (string-match "[?#]" (url-filename urlobj
))
132 (setq query
(substring (url-filename urlobj
) (match-end 0))
133 file
(substring (url-filename urlobj
) 0 (match-beginning 0))
134 sepchar
(substring (url-filename urlobj
) (match-beginning 0) (match-end 0)))
135 (setq file
(url-filename urlobj
)))
136 (setq file
(url-expander-remove-relative-links
137 (concat (url-basepath (url-filename defobj
)) file
)))
138 (url-set-filename urlobj
(if query
(concat file sepchar query
) file
))))))
140 (provide 'url-expand
)
142 ;;; arch-tag: 7b5f744b-b721-49da-be47-484631680a5a