This solves also Bug#27009.
* lisp/net/tramp.el (tramp-current-domain)
(tramp-current-port): New defvars.
(tramp-file-name): New defstruct.
(tramp-file-name-user-domain, tramp-file-name-host-port)
(tramp-file-name-equal-p): New defuns.
(tramp-file-name-p, tramp-file-name-method)
(tramp-file-name-user, tramp-file-name-host)
(tramp-file-name-localname, tramp-file-name-hop)
(tramp-file-name-real-user, tramp-file-name-domain)
(tramp-file-name-real-host, tramp-file-name-port):
Remove defuns. They are provided by the defstruct, or not
needed anymore.
(tramp-dissect-file-name, tramp-buffer-name)
(tramp-make-tramp-file-name, tramp-get-buffer)
(tramp-set-connection-local-variables)
(tramp-debug-buffer-name, tramp-message)
(tramp-error-with-buffer, with-parsed-tramp-file-name)
(tramp-completion-dissect-file-name1)
(tramp-handle-file-name-as-directory)
(tramp-handle-file-name-directory)
(tramp-handle-file-remote-p, tramp-handle-file-symlink-p)
(tramp-handle-find-backup-file-name)
(tramp-handle-insert-file-contents, tramp-process-actions)
(tramp-check-cached-permissions, tramp-local-host-p)
(tramp-get-remote-tmpdir, tramp-call-process)
(tramp-call-process-region, tramp-read-passwd)
(tramp-clear-passwd):
* lisp/net/tramp-adb.el (tramp-adb-parse-device-names)
(tramp-adb-handle-expand-file-name)
(tramp-adb-handle-file-truename, tramp-adb-handle-copy-file)
(tramp-adb-handle-process-file)
(tramp-adb-maybe-open-connection):
* lisp/net/tramp-cache.el (tramp-get-hash-table)
(tramp-get-file-property, tramp-set-file-property)
(tramp-flush-file-property, tramp-flush-directory-property)
(tramp-get-connection-property)
(tramp-set-connection-property, tramp-connection-property-p)
(tramp-flush-connection-property, tramp-cache-print)
(tramp-list-connections, tramp-dump-connection-properties)
(tramp-parse-connection-properties):
* lisp/net/tramp-cmds.el (tramp-cleanup-connection):
* lisp/net/tramp-ftp.el (tramp-ftp-file-name-handler):
* lisp/net/tramp-gvfs.el (tramp-gvfs-handle-expand-file-name)
(tramp-gvfs-url-file-name, tramp-gvfs-handler-askpassword)
(tramp-gvfs-handler-mounted-unmounted)
(tramp-gvfs-mount-spec, tramp-gvfs-get-remote-uid)
(tramp-gvfs-get-remote-gid)
(tramp-gvfs-maybe-open-connection):
* lisp/net/tramp-sh.el (tramp-sh-handle-file-truename)
(tramp-do-copy-or-rename-file-out-of-band)
(tramp-sh-handle-expand-file-name)
(tramp-sh-handle-start-file-process)
(tramp-sh-handle-process-file, tramp-compute-multi-hops)
(tramp-maybe-open-connection)
(tramp-make-copy-program-file-name, tramp-get-remote-path)
(tramp-get-inline-coding):
* lisp/net/tramp-smb.el (tramp-smb-handle-copy-directory)
(tramp-smb-handle-expand-file-name)
(tramp-smb-handle-file-acl, tramp-smb-handle-process-file)
(tramp-smb-handle-set-file-acl)
(tramp-smb-maybe-open-connection): Adapt according to defstruct.
;; That's why we use `start-process'.
(let ((p (start-process
tramp-adb-program (current-buffer) tramp-adb-program "devices"))
- (v (vector tramp-adb-method tramp-current-user
- tramp-current-host nil nil))
+ (v (tramp-make-tramp-file-name
+ tramp-adb-method tramp-current-user nil
+ tramp-current-host nil nil nil))
result)
(tramp-message v 6 "%s" (mapconcat 'identity (process-command p) " "))
(process-put p 'adjust-window-size-function 'ignore)
;; be problems with UNC shares or Cygwin mounts.
(let ((default-directory (tramp-compat-temporary-file-directory)))
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(tramp-drop-volume-letter
(tramp-run-real-handler
'expand-file-name (list localname))))))))
"%s%s"
(with-parsed-tramp-file-name (expand-file-name filename) nil
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(with-tramp-file-property v localname "file-truename"
(let ((result nil)) ; result steps in reverse order
(tramp-message v 4 "Finding true name for `%s'" filename)
(tramp-compat-file-attribute-type
(file-attributes
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(mapconcat 'identity
(append '("")
(reverse result)
newname (expand-file-name newname))
(if (file-directory-p filename)
- (tramp-file-name-handler 'copy-directory filename newname keep-date t)
+ (copy-directory filename newname keep-date t)
(let ((t1 (tramp-tramp-file-p filename))
(t2 (tramp-tramp-file-p newname)))
(setq input (with-parsed-tramp-file-name infile nil localname))
;; INFILE must be copied to remote host.
(setq input (tramp-make-tramp-temp-file v)
- tmpinput (tramp-make-tramp-file-name method user host input))
+ tmpinput (tramp-make-tramp-file-name
+ method user domain host port input))
(copy-file infile tmpinput t)))
(when input (setq command (format "%s <%s" command input)))
;; file must be deleted after execution.
(setq stderr (tramp-make-tramp-temp-file v)
tmpstderr (tramp-make-tramp-file-name
- method user host stderr))))
+ method user domain host port stderr))))
;; stderr to be discarded.
((null (cadr destination))
(setq stderr "/dev/null"))))
(device (tramp-adb-get-device vec)))
;; Set variables for proper tracing in `tramp-adb-parse-device-names'.
- (setq tramp-current-method (tramp-file-name-method vec)
- tramp-current-user (tramp-file-name-user vec)
+ (setq tramp-current-user (tramp-file-name-user vec)
tramp-current-host (tramp-file-name-host vec))
;; Maybe we know already that "su" is not supported. We cannot
;; An implementation of information caching for remote files.
-;; Each connection, identified by a vector [method user host
-;; localname] or by a process, has a unique cache. We distinguish 3
-;; kind of caches, depending on the key:
+;; Each connection, identified by a `tramp-file-name' structure or by
+;; a process, has a unique cache. We distinguish 3 kind of caches,
+;; depending on the key:
;;
;; - localname is NIL. This are reusable properties. Examples:
;; "remote-shell" identifies the POSIX shell to be called on the
(or (gethash key tramp-cache-data)
(let ((hash
(puthash key (make-hash-table :test 'equal) tramp-cache-data)))
- (when (vectorp key)
+ (when (tramp-file-name-p key)
(dolist (elt tramp-connection-properties)
(when (string-match
(or (nth 0 elt) "")
(tramp-make-tramp-file-name
- (aref key 0) (aref key 1) (aref key 2) nil))
+ (tramp-file-name-method key) (tramp-file-name-user key)
+ (tramp-file-name-domain key) (tramp-file-name-host key)
+ (tramp-file-name-port key) nil))
(tramp-set-connection-property key (nth 1 elt) (nth 2 elt)))))
hash)))
(defun tramp-get-file-property (key file property default)
"Get the PROPERTY of FILE from the cache context of KEY.
Returns DEFAULT if not set."
- ;; Unify localname. Remove hop from vector.
- (setq file (tramp-compat-file-name-unquote file))
- (setq key (copy-sequence key))
- (aset key 3 (tramp-run-real-handler 'directory-file-name (list file)))
- (aset key 4 nil)
+ ;; Unify localname. Remove hop from `tramp-file-name' structure.
+ (setq file (tramp-compat-file-name-unquote file)
+ key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key)
+ (tramp-run-real-handler 'directory-file-name (list file))
+ (tramp-file-name-hop key) nil)
(let* ((hash (tramp-get-hash-table key))
(value (when (hash-table-p hash) (gethash property hash))))
(if
(defun tramp-set-file-property (key file property value)
"Set the PROPERTY of FILE to VALUE, in the cache context of KEY.
Returns VALUE."
- ;; Unify localname. Remove hop from vector.
- (setq file (tramp-compat-file-name-unquote file))
- (setq key (copy-sequence key))
- (aset key 3 (tramp-run-real-handler 'directory-file-name (list file)))
- (aset key 4 nil)
+ ;; Unify localname. Remove hop from `tramp-file-name' structure.
+ (setq file (tramp-compat-file-name-unquote file)
+ key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key)
+ (tramp-run-real-handler 'directory-file-name (list file))
+ (tramp-file-name-hop key) nil)
(let ((hash (tramp-get-hash-table key)))
;; We put the timestamp there.
(puthash property (cons (current-time) value) hash)
(let* ((file (tramp-run-real-handler
'directory-file-name (list file)))
(truename (tramp-get-file-property key file "file-truename" nil)))
- ;; Unify localname. Remove hop from vector.
- (setq file (tramp-compat-file-name-unquote file))
- (setq key (copy-sequence key))
- (aset key 3 file)
- (aset key 4 nil)
+ ;; Unify localname. Remove hop from `tramp-file-name' structure.
+ (setq file (tramp-compat-file-name-unquote file)
+ key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key) file
+ (tramp-file-name-hop key) nil)
(tramp-message key 8 "%s" file)
(remhash key tramp-cache-data)
;; Remove file properties of symlinks.
(tramp-message key 8 "%s" directory)
(maphash
(lambda (key _value)
- (when (and (stringp (tramp-file-name-localname key))
+ (when (and (tramp-file-name-p key)
+ (stringp (tramp-file-name-localname key))
(string-match (regexp-quote directory)
(tramp-file-name-localname key)))
(remhash key tramp-cache-data)))
(defun tramp-get-connection-property (key property default)
"Get the named PROPERTY for the connection.
KEY identifies the connection, it is either a process or a
-vector. A special case is nil, which is used to cache connection
-properties of the local machine. If the value is not set for the
-connection, returns DEFAULT."
- ;; Unify key by removing localname and hop from vector. Work with a
- ;; copy in order to avoid side effects.
- (when (vectorp key)
- (setq key (copy-sequence key))
- (aset key 3 nil)
- (aset key 4 nil))
+`tramp-file-name' structure. A special case is nil, which is
+used to cache connection properties of the local machine. If the
+value is not set for the connection, returns DEFAULT."
+ ;; Unify key by removing localname and hop from `tramp-file-name'
+ ;; structure. Work with a copy in order to avoid side effects.
+ (when (tramp-file-name-p key)
+ (setq key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key) nil
+ (tramp-file-name-hop key) nil))
(let* ((hash (tramp-get-hash-table key))
(value
;; If the key is an auxiliary process object, check whether
(defun tramp-set-connection-property (key property value)
"Set the named PROPERTY of a connection to VALUE.
KEY identifies the connection, it is either a process or a
-vector. A special case is nil, which is used to cache connection
-properties of the local machine. PROPERTY is set persistent when
-KEY is a vector."
- ;; Unify key by removing localname and hop from vector. Work with a
- ;; copy in order to avoid side effects.
- (when (vectorp key)
- (setq key (copy-sequence key))
- (aset key 3 nil)
- (aset key 4 nil))
+`tramp-file-name' structure. A special case is nil, which is
+used to cache connection properties of the local machine.
+PROPERTY is set persistent when KEY is a `tramp-file-name' structure."
+ ;; Unify key by removing localname and hop from `tramp-file-name'
+ ;; structure. Work with a copy in order to avoid side effects.
+ (when (tramp-file-name-p key)
+ (setq key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key) nil
+ (tramp-file-name-hop key) nil))
(let ((hash (tramp-get-hash-table key)))
(puthash property value hash)
(setq tramp-cache-data-changed t)
(defun tramp-connection-property-p (key property)
"Check whether named PROPERTY of a connection is defined.
KEY identifies the connection, it is either a process or a
-vector. A special case is nil, which is used to cache connection
-properties of the local machine."
+`tramp-file-name' structure. A special case is nil, which is
+used to cache connection properties of the local machine."
(not (eq (tramp-get-connection-property key property 'undef) 'undef)))
;;;###tramp-autoload
(defun tramp-flush-connection-property (key)
"Remove all properties identified by KEY.
KEY identifies the connection, it is either a process or a
-vector. A special case is nil, which is used to cache connection
-properties of the local machine."
- ;; Unify key by removing localname and hop from vector. Work with a
- ;; copy in order to avoid side effects.
- (when (vectorp key)
- (setq key (copy-sequence key))
- (aset key 3 nil)
- (aset key 4 nil))
+`tramp-file-name' structure. A special case is nil, which is
+used to cache connection properties of the local machine."
+ ;; Unify key by removing localname and hop from `tramp-file-name'
+ ;; structure. Work with a copy in order to avoid side effects.
+ (when (tramp-file-name-p key)
+ (setq key (copy-tramp-file-name key))
+ (setf (tramp-file-name-localname key) nil
+ (tramp-file-name-hop key) nil))
(tramp-message
key 7 "%s %s" key
(let ((hash (gethash key tramp-cache-data))
(maphash
(lambda (key value)
;; Remove text properties from KEY and VALUE.
- (when (vectorp key)
+ ;; `cl-struct-slot-*' functions exist since Emacs 25 only; we
+ ;; ignore errors.
+ (when (tramp-file-name-p key)
+ ;; (dolist
+ ;; (slot
+ ;; (mapcar 'car (cdr (cl-struct-slot-info 'tramp-file-name))))
+ ;; (when (stringp (cl-struct-slot-value 'tramp-file-name slot key))
+ ;; (setf (cl-struct-slot-value 'tramp-file-name slot key)
+ ;; (substring-no-properties
+ ;; (cl-struct-slot-value 'tramp-file-name slot key))))))
(dotimes (i (length key))
(when (stringp (aref key i))
(aset key i (substring-no-properties (aref key i))))))
;;;###tramp-autoload
(defun tramp-list-connections ()
- "Return a list of all known connection vectors according to `tramp-cache'."
+ "Return all known `tramp-file-name' structs according to `tramp-cache'."
(let (result tramp-verbose)
(maphash
(lambda (key _value)
- (when (and (vectorp key) (null (aref key 3))
+ (when (and (tramp-file-name-p key)
+ (null (tramp-file-name-localname key))
(tramp-connection-property-p key "process-buffer"))
(add-to-list 'result key)))
tramp-cache-data)
;; possibility to use another login name later on.
(maphash
(lambda (key value)
- (if (and (vectorp key)
+ (if (and (tramp-file-name-p key)
(not (tramp-file-name-localname key))
(not (gethash "login-as" value)))
(progn
(let (res)
(maphash
(lambda (key _value)
- (if (and (vectorp key)
+ (if (and (tramp-file-name-p key)
(string-equal method (tramp-file-name-method key))
(not (tramp-file-name-localname key)))
(push (list (tramp-file-name-user key)
element key item)
(while (setq element (pop list))
(setq key (pop element))
- (while (setq item (pop element))
- ;; We set only values which are not contained in
- ;; `tramp-connection-properties'. The cache is
- ;; initialized properly by side effect.
- (unless (tramp-connection-property-p key (car item))
- (tramp-set-connection-property key (pop item) (car item))))))
+ (when (tramp-file-name-p key)
+ (while (setq item (pop element))
+ ;; We set only values which are not contained in
+ ;; `tramp-connection-properties'. The cache is
+ ;; initialized properly by side effect.
+ (unless (tramp-connection-property-p key (car item))
+ (tramp-set-connection-property key (pop item) (car item)))))))
(setq tramp-cache-data-changed nil))
(file-error
;; Most likely because the file doesn't exist yet. No message.
(tramp-make-tramp-file-name
(tramp-file-name-method x)
(tramp-file-name-user x)
+ (tramp-file-name-domain x)
(tramp-file-name-host x)
+ (tramp-file-name-port x)
(tramp-file-name-localname x)))
(tramp-list-connections)))
name)
((memq operation '(file-directory-p file-exists-p))
(if (apply 'ange-ftp-hook-function operation args)
(let ((v (tramp-dissect-file-name (car args) t)))
- (aset v 0 tramp-ftp-method)
+ (setf (tramp-file-name-method v) tramp-ftp-method)
(tramp-set-connection-property v "started" t))
nil))
;; If there is a default location, expand tilde.
(when (string-match "\\`\\(~\\)\\(/\\|\\'\\)" localname)
(save-match-data
- (tramp-gvfs-maybe-open-connection (vector method user host "/" hop)))
+ (tramp-gvfs-maybe-open-connection
+ (tramp-make-tramp-file-name method user domain host port "/" hop)))
(setq localname
(replace-match
(tramp-get-connection-property v "default-location" "~")
;; No tilde characters in file name, do normal
;; `expand-file-name' (this does "/./" and "/../").
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(tramp-run-real-handler
'expand-file-name (list localname))))))
(concat (match-string 2 user) ";" (match-string 1 user))))
(url-parse-make-urlobj
method (and user (url-hexify-string user)) nil
- (tramp-file-name-real-host v) (tramp-file-name-port v)
+ (tramp-file-name-host v) (tramp-file-name-port v)
(and localname (url-hexify-string localname)) nil nil t))
(url-parse-make-urlobj
"file" nil nil nil nil
t ;; password handled.
nil ;; no abort of D-Bus.
password
- (tramp-file-name-real-user l)
+ (tramp-file-name-user l)
domain
nil ;; not anonymous.
0) ;; no password save.
;; No password provided.
- (list nil t "" (tramp-file-name-real-user l) domain nil 0)))
+ (list nil t "" (tramp-file-name-user l) domain nil 0)))
;; When QUIT is raised, we shall return this information to D-Bus.
(quit (list nil t "" "" "" nil 0)))))
(unless (zerop (length port))
(setq host (concat host tramp-prefix-port-format port)))
(with-parsed-tramp-file-name
- (tramp-make-tramp-file-name method user host "") nil
+ (tramp-make-tramp-file-name method user domain host port "") nil
(tramp-message
v 6 "%s %s"
signal-name (tramp-gvfs-stringify-dbus-message mount-info))
(defun tramp-gvfs-mount-spec (vec)
"Return a mount-spec for \"org.gtk.vfs.MountTracker.mountLocation\"."
(let* ((method (tramp-file-name-method vec))
- (user (tramp-file-name-real-user vec))
+ (user (tramp-file-name-user vec))
(domain (tramp-file-name-domain vec))
- (host (tramp-file-name-real-host vec))
+ (host (tramp-file-name-host vec))
(port (tramp-file-name-port vec))
(localname (tramp-file-name-unquote-localname vec))
(share (when (string-match "^/?\\([^/]+\\)" localname)
(with-tramp-connection-property vec (format "uid-%s" id-format)
(let ((method (tramp-file-name-method vec))
(user (tramp-file-name-user vec))
+ (domain (tramp-file-name-domain vec))
(host (tramp-file-name-host vec))
+ (port (tramp-file-name-port vec))
(localname
(tramp-get-connection-property vec "default-location" nil)))
(cond
(localname
(tramp-compat-file-attribute-user-id
(file-attributes
- (tramp-make-tramp-file-name method user host localname) id-format)))
+ (tramp-make-tramp-file-name method user domain host port localname)
+ id-format)))
((equal id-format 'integer) tramp-unknown-id-integer)
((equal id-format 'string) tramp-unknown-id-string)))))
(with-tramp-connection-property vec (format "gid-%s" id-format)
(let ((method (tramp-file-name-method vec))
(user (tramp-file-name-user vec))
+ (domain (tramp-file-name-domain vec))
(host (tramp-file-name-host vec))
+ (port (tramp-file-name-port vec))
(localname
(tramp-get-connection-property vec "default-location" nil)))
(cond
(localname
(tramp-compat-file-attribute-group-id
(file-attributes
- (tramp-make-tramp-file-name method user host localname) id-format)))
+ (tramp-make-tramp-file-name method user domain host port localname)
+ id-format)))
((equal id-format 'integer) tramp-unknown-id-integer)
((equal id-format 'string) tramp-unknown-id-string)))))
(unless (tramp-gvfs-connection-mounted-p vec)
(let* ((method (tramp-file-name-method vec))
(user (tramp-file-name-user vec))
+ (domain (tramp-file-name-domain vec))
(host (tramp-file-name-host vec))
+ (port (tramp-file-name-port vec))
(localname (tramp-file-name-unquote-localname vec))
(object-path
(tramp-gvfs-object-path
- (tramp-make-tramp-file-name method user host ""))))
+ (tramp-make-tramp-file-name method user domain host port ""))))
(when (and (string-equal method "afp")
(string-equal localname "/"))
"%s%s"
(with-parsed-tramp-file-name (expand-file-name filename) nil
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(with-tramp-file-property v localname "file-truename"
(let ((result nil) ; result steps in reverse order
(quoted (tramp-compat-file-name-quoted-p localname))
(tramp-compat-file-attribute-type
(file-attributes
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(mapconcat 'identity
(append '("")
(reverse result)
(let* ((t1 (tramp-tramp-file-p filename))
(t2 (tramp-tramp-file-p newname))
(orig-vec (tramp-dissect-file-name (if t1 filename newname)))
- copy-program copy-args copy-env copy-keep-date port listener spec
+ copy-program copy-args copy-env copy-keep-date listener spec
options source target remote-copy-program remote-copy-args)
(with-parsed-tramp-file-name (if t1 filename newname) nil
tramp-current-user (or (tramp-file-name-user v)
(tramp-get-connection-property
v "login-as" nil))
- tramp-current-host (tramp-file-name-real-host v))
+ tramp-current-host (tramp-file-name-host v))
;; Check which ones of source and target are Tramp files.
(setq source (funcall
(tramp-make-copy-program-file-name v)
(tramp-unquote-shell-quote-argument newname)))
- ;; Check for host and port number.
- (setq host (tramp-file-name-real-host v)
- port (tramp-file-name-port v))
-
;; Check for user. There might be an interactive setting.
(setq user (or (tramp-file-name-user v)
(tramp-get-connection-property v "login-as" nil)))
;; be problems with UNC shares or Cygwin mounts.
(let ((default-directory (tramp-compat-temporary-file-directory)))
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(tramp-drop-volume-letter
(tramp-run-real-handler
'expand-file-name (list localname)))
(tramp-make-tramp-file-name
(tramp-file-name-method v)
(tramp-file-name-user v)
+ (tramp-file-name-domain v)
(tramp-file-name-host v)
+ (tramp-file-name-port v)
(tramp-file-name-localname v))
tramp-initial-end-of-output))
;; We use as environment the difference to toplevel
(setq input (with-parsed-tramp-file-name infile nil localname))
;; INFILE must be copied to remote host.
(setq input (tramp-make-tramp-temp-file v)
- tmpinput (tramp-make-tramp-file-name method user host input))
+ tmpinput
+ (tramp-make-tramp-file-name method user domain host port input))
(copy-file infile tmpinput t)))
(when input (setq command (format "%s <%s" command input)))
;; file must be deleted after execution.
(setq stderr (tramp-make-tramp-temp-file v)
tmpstderr (tramp-make-tramp-file-name
- method user host stderr))))
+ method user domain host port stderr))))
;; stderr to be discarded.
((null (cadr destination))
(setq stderr "/dev/null"))))
;; host name.
(let* ((v (car target-alist))
(method (tramp-file-name-method v))
- (host (tramp-file-name-real-host v)))
+ (host (tramp-file-name-host v)))
(unless
(or
;; There are multi-hops.
;; If Tramp opens the same connection within a short time frame,
;; there is a problem. We shall signal this.
(unless (or (tramp-compat-process-live-p p)
- (not (equal (butlast (append vec nil) 2)
- (car tramp-current-connection)))
+ (not (tramp-file-name-equal-p
+ vec (car tramp-current-connection)))
(> (tramp-time-diff
(current-time) (cdr tramp-current-connection))
(or tramp-connection-min-time-diff 0)))
(set-process-sentinel p 'tramp-process-sentinel)
(process-put p 'adjust-window-size-function 'ignore)
(set-process-query-on-exit-flag p nil)
- (setq tramp-current-connection
- (cons (butlast (append vec nil) 2) (current-time))
+ (setq tramp-current-connection (cons vec (current-time))
tramp-current-host (system-name))
(tramp-message
"Create a file name suitable for `scp', `pscp', or `nc' and workalikes."
(let ((method (tramp-file-name-method vec))
(user (tramp-file-name-user vec))
- (host (tramp-file-name-real-host vec))
+ (host (tramp-file-name-host vec))
(localname
(directory-file-name (tramp-file-name-unquote-localname vec))))
(when (string-match tramp-ipv6-regexp host)
(tramp-make-tramp-file-name
(tramp-file-name-method vec)
(tramp-file-name-user vec)
+ (tramp-file-name-domain vec)
(tramp-file-name-host vec)
+ (tramp-file-name-port vec)
x))
x))
remote-path)))))
(let ((coding-system-for-write 'binary)
(coding-system-for-read 'binary))
(apply
- 'tramp-call-process-region ,vec (point-min) (point-max)
+ 'tramp-call-process-region ',vec (point-min) (point-max)
(car (split-string ,compress)) t t nil
(cdr (split-string ,compress)))))
`(lambda (beg end)
(let ((coding-system-for-write 'binary)
(coding-system-for-read 'binary))
(apply
- 'tramp-call-process-region ,vec beg end
+ 'tramp-call-process-region ',vec beg end
(car (split-string ,compress)) t t nil
(cdr (split-string ,compress))))
(,coding (point-min) (point-max)))))
;; Another guess. We might implement a better check later on.
(tramp-case-insensitive t))))
-;; Add a default for `tramp-default-method-alist'. Rule: If there is
-;; a domain in USER, it must be the SMB method.
-;;;###tramp-autoload
-(add-to-list 'tramp-default-method-alist
- `(nil ,tramp-prefix-domain-regexp ,tramp-smb-method))
-
;; Add a default for `tramp-default-user-alist'. Rule: For the SMB method,
;; the anonymous user is chosen.
;;;###tramp-autoload
(if (not (file-directory-p newname))
(make-directory newname parents))
- (setq tramp-current-method (tramp-file-name-method v)
- tramp-current-user (tramp-file-name-user v)
- tramp-current-host (tramp-file-name-real-host v))
+ (setq tramp-current-method method
+ tramp-current-user user
+ tramp-current-host host)
- (let* ((real-user (tramp-file-name-real-user v))
- (real-host (tramp-file-name-real-host v))
- (domain (tramp-file-name-domain v))
- (port (tramp-file-name-port v))
- (share (tramp-smb-get-share v))
+ (let* ((share (tramp-smb-get-share v))
(localname (file-name-as-directory
(replace-regexp-in-string
"\\\\" "/" (tramp-smb-get-localname v))))
(expand-file-name
tramp-temp-name-prefix
(tramp-compat-temporary-file-directory))))
- (args (list (concat "//" real-host "/" share) "-E")))
+ (args (list (concat "//" host "/" share) "-E")))
- (if (not (zerop (length real-user)))
- (setq args (append args (list "-U" real-user)))
+ (if (not (zerop (length user)))
+ (setq args (append args (list "-U" user)))
(setq args (append args (list "-N"))))
(when domain (setq args (append args (list "-W" domain))))
(setq localname
(replace-match
(if (zerop (length (match-string 1 localname)))
- (tramp-file-name-real-user v)
+ user
(match-string 1 localname))
nil nil localname)))
;; Make the file name absolute.
;; No tilde characters in file name, do normal
;; `expand-file-name' (this does "/./" and "/../").
(tramp-make-tramp-file-name
- method user host
+ method user domain host port
(tramp-run-real-handler 'expand-file-name (list localname))))))
(defun tramp-smb-action-get-acl (proc vec)
(with-tramp-file-property v localname "file-acl"
(when (executable-find tramp-smb-acl-program)
- (setq tramp-current-method (tramp-file-name-method v)
- tramp-current-user (tramp-file-name-user v)
- tramp-current-host (tramp-file-name-real-host v))
+ (setq tramp-current-method method
+ tramp-current-user user
+ tramp-current-host host)
- (let* ((real-user (tramp-file-name-real-user v))
- (real-host (tramp-file-name-real-host v))
- (domain (tramp-file-name-domain v))
- (port (tramp-file-name-port v))
- (share (tramp-smb-get-share v))
+ (let* ((share (tramp-smb-get-share v))
(localname (replace-regexp-in-string
"\\\\" "/" (tramp-smb-get-localname v)))
- (args (list (concat "//" real-host "/" share) "-E")))
+ (args (list (concat "//" host "/" share) "-E")))
- (if (not (zerop (length real-user)))
- (setq args (append args (list "-U" real-user)))
+ (if (not (zerop (length user)))
+ (setq args (append args (list "-U" user)))
(setq args (append args (list "-N"))))
(when domain (setq args (append args (list "-W" domain))))
(setq input (with-parsed-tramp-file-name infile nil localname))
;; INFILE must be copied to remote host.
(setq input (tramp-make-tramp-temp-file v)
- tmpinput (tramp-make-tramp-file-name method user host input))
+ tmpinput
+ (tramp-make-tramp-file-name method user domain host port input))
(copy-file infile tmpinput t))
;; Transform input into a filename powershell does understand.
(setq input (format "//%s%s" host input)))
(ignore-errors
(with-parsed-tramp-file-name filename nil
(when (and (stringp acl-string) (executable-find tramp-smb-acl-program))
- (setq tramp-current-method (tramp-file-name-method v)
- tramp-current-user (tramp-file-name-user v)
- tramp-current-host (tramp-file-name-real-host v))
+ (setq tramp-current-method method
+ tramp-current-user user
+ tramp-current-host host)
(tramp-set-file-property v localname "file-acl" 'undef)
- (let* ((real-user (tramp-file-name-real-user v))
- (real-host (tramp-file-name-real-host v))
- (domain (tramp-file-name-domain v))
- (port (tramp-file-name-port v))
- (share (tramp-smb-get-share v))
+ (let* ((share (tramp-smb-get-share v))
(localname (replace-regexp-in-string
"\\\\" "/" (tramp-smb-get-localname v)))
- (args (list (concat "//" real-host "/" share) "-E" "-S"
+ (args (list (concat "//" host "/" share) "-E" "-S"
(replace-regexp-in-string
"\n" "," acl-string))))
- (if (not (zerop (length real-user)))
- (setq args (append args (list "-U" real-user)))
+ (if (not (zerop (length user)))
+ (setq args (append args (list "-U" user)))
(setq args (append args (list "-N"))))
(when domain (setq args (append args (list "-W" domain))))
(when buf (with-current-buffer buf (erase-buffer)))
(when (and p (processp p)) (delete-process p))
- (let* ((user (tramp-file-name-user vec))
- (host (tramp-file-name-host vec))
- (real-user (tramp-file-name-real-user vec))
- (real-host (tramp-file-name-real-host vec))
- (domain (tramp-file-name-domain vec))
- (port (tramp-file-name-port vec))
+ (let* ((user (tramp-file-name-user vec))
+ (host (tramp-file-name-host vec))
+ (domain (tramp-file-name-domain vec))
+ (port (tramp-file-name-port vec))
args)
(cond
(argument
- (setq args (list (concat "//" real-host))))
+ (setq args (list (concat "//" host))))
(share
- (setq args (list (concat "//" real-host "/" share))))
+ (setq args (list (concat "//" host "/" share))))
(t
- (setq args (list "-g" "-L" real-host ))))
+ (setq args (list "-g" "-L" host ))))
- (if (not (zerop (length real-user)))
- (setq args (append args (list "-U" real-user)))
+ (if (not (zerop (length user)))
+ (setq args (append args (list "-U" user)))
(setq args (append args (list "-N"))))
(when domain (setq args (append args (list "-W" domain))))
(defvar tramp-current-user nil
"Remote login name for this *tramp* buffer.")
+(defvar tramp-current-domain nil
+ "Remote domain name for this *tramp* buffer.")
+
(defvar tramp-current-host nil
"Remote host for this *tramp* buffer.")
+(defvar tramp-current-port nil
+ "Remote port for this *tramp* buffer.")
+
(defvar tramp-current-connection nil
"Last connection timestamp.")
;; internal data structure. Convenience functions for internal
;; data structure.
+;; The basic structure for remote file names. We use a list,
+;; otherwise the test in `tramp-cache-data' fails.
+(cl-defstruct (tramp-file-name (:type list) :named)
+ method user domain host port localname hop)
+
+(defun tramp-file-name-user-domain (vec)
+ "Return user and domain components of VEC."
+ (when (or (tramp-file-name-user vec) (tramp-file-name-domain vec))
+ (concat (tramp-file-name-user vec)
+ (and (tramp-file-name-domain vec)
+ tramp-prefix-domain-format)
+ (tramp-file-name-domain vec))))
+
+(defun tramp-file-name-host-port (vec)
+ "Return host and port components of VEC."
+ (when (or (tramp-file-name-host vec) (tramp-file-name-port vec))
+ (concat (tramp-file-name-host vec)
+ (and (tramp-file-name-port vec)
+ tramp-prefix-port-format)
+ (tramp-file-name-port vec))))
+
+(defun tramp-file-name-equal-p (vec1 vec2)
+ "Check, whether VEC1 and VEC2 denote the same `tramp-file-name'."
+ (and (tramp-file-name-p vec1) (tramp-file-name-p vec2)
+ (string-equal (tramp-file-name-method vec1)
+ (tramp-file-name-method vec2))
+ (string-equal (tramp-file-name-user-domain vec1)
+ (tramp-file-name-user-domain vec2))
+ (string-equal (tramp-file-name-host-port vec1)
+ (tramp-file-name-host-port vec2))))
+
(defun tramp-get-method-parameter (vec param)
"Return the method parameter PARAM.
If VEC is a vector, check first in connection properties.
(assoc param (assoc (tramp-file-name-method vec) tramp-methods))))
(when methods-entry (cadr methods-entry))))))
-(defun tramp-file-name-p (vec)
- "Check, whether VEC is a Tramp object."
- (and (vectorp vec) (= 5 (length vec))))
-
-(defun tramp-file-name-method (vec)
- "Return method component of VEC."
- (and (tramp-file-name-p vec) (aref vec 0)))
-
-(defun tramp-file-name-user (vec)
- "Return user component of VEC."
- (and (tramp-file-name-p vec) (aref vec 1)))
-
-(defun tramp-file-name-host (vec)
- "Return host component of VEC."
- (and (tramp-file-name-p vec) (aref vec 2)))
-
-(defun tramp-file-name-localname (vec)
- "Return localname component of VEC."
- (and (tramp-file-name-p vec) (aref vec 3)))
-
-(defun tramp-file-name-hop (vec)
- "Return hop component of VEC."
- (and (tramp-file-name-p vec) (aref vec 4)))
-
-;; The user part of a Tramp file name vector can be of kind
-;; "user%domain". Sometimes, we must extract these parts.
-(defun tramp-file-name-real-user (vec)
- "Return the user name of VEC without domain."
- (save-match-data
- (let ((user (tramp-file-name-user vec)))
- (if (and (stringp user)
- (string-match tramp-user-with-domain-regexp user))
- (match-string 1 user)
- user))))
-
-(defun tramp-file-name-domain (vec)
- "Return the domain name of VEC."
- (save-match-data
- (let ((user (tramp-file-name-user vec)))
- (and (stringp user)
- (string-match tramp-user-with-domain-regexp user)
- (match-string 2 user)))))
-
-;; The host part of a Tramp file name vector can be of kind
-;; "host#port". Sometimes, we must extract these parts.
-(defun tramp-file-name-real-host (vec)
- "Return the host name of VEC without port."
- (save-match-data
- (let ((host (tramp-file-name-host vec)))
- (if (and (stringp host)
- (string-match tramp-host-with-port-regexp host))
- (match-string 1 host)
- host))))
-
-(defun tramp-file-name-port (vec)
- "Return the port number of VEC."
- (save-match-data
- (let ((host (tramp-file-name-host vec)))
- (or (and (stringp host)
- (string-match tramp-host-with-port-regexp host)
- (string-to-number (match-string 2 host)))
- (tramp-get-method-parameter vec 'tramp-default-port)))))
-
;; The localname can be quoted with "/:". Extract this.
(defun tramp-file-name-unquote-localname (vec)
"Return unquoted localname component of VEC."
(user (match-string (nth 2 (tramp-file-name-structure)) name))
(host (match-string (nth 3 (tramp-file-name-structure)) name))
(localname (match-string (nth 4 (tramp-file-name-structure)) name))
- (hop (match-string (nth 5 (tramp-file-name-structure)) name)))
+ (hop (match-string (nth 5 (tramp-file-name-structure)) name))
+ domain port)
+ (when user
+ (when (string-match tramp-user-with-domain-regexp user)
+ (setq domain (match-string 2 user)
+ user (match-string 1 user))))
+
(when host
+ (when (string-match tramp-host-with-port-regexp host)
+ (setq port (match-string 2 host)
+ host (match-string 1 host)))
(when (string-match (tramp-prefix-ipv6-regexp) host)
(setq host (replace-match "" nil t host)))
(when (string-match (tramp-postfix-ipv6-regexp) host)
(setq host (replace-match "" nil t host))))
- (if nodefault
- (vector method user host localname hop)
- (vector
- (tramp-find-method method user host)
- (tramp-find-user method user host)
- (tramp-find-host method user host)
- localname hop))))))
+
+ (unless nodefault
+ (setq method (tramp-find-method method user host)
+ user (tramp-find-user method user host)
+ host (tramp-find-host method user host)))
+
+ (apply
+ 'make-tramp-file-name
+ (append
+ (unless (zerop (length method)) `(:method ,method))
+ (unless (zerop (length user)) `(:user ,user))
+ (unless (zerop (length domain)) `(:domain ,domain))
+ (unless (zerop (length host)) `(:host ,host))
+ (unless (zerop (length port)) `(:port ,port))
+ `(:localname ,(or localname ""))
+ (unless (zerop (length hop)) `(:hop ,hop))))))))
(defun tramp-buffer-name (vec)
"A name for the connection buffer VEC."
(let ((method (tramp-file-name-method vec))
- (user (tramp-file-name-user vec))
- (host (tramp-file-name-real-host vec)))
- (if (not (zerop (length user)))
- (format "*tramp/%s %s@%s*" method user host)
- (format "*tramp/%s %s*" method host))))
-
-(defun tramp-make-tramp-file-name (method user host localname &optional hop)
+ (user-domain (tramp-file-name-user-domain vec))
+ (host-port (tramp-file-name-host-port vec)))
+ (if (not (zerop (length user-domain)))
+ (format "*tramp/%s %s@%s*" method user-domain host-port)
+ (format "*tramp/%s %s*" method host-port))))
+
+(defun tramp-make-tramp-file-name
+ (method user domain host port localname &optional hop)
"Constructs a Tramp file name from METHOD, USER, HOST and LOCALNAME.
-When not nil, an optional HOP is prepended."
+When not nil, optional DOMAIN, PORT and HOP are used."
(concat (tramp-prefix-format) hop
(unless (or (zerop (length method))
(zerop (length (tramp-postfix-method-format))))
(concat method (tramp-postfix-method-format)))
+ user
+ (unless (zerop (length domain))
+ (concat tramp-prefix-domain-format domain))
(unless (zerop (length user))
- (concat user tramp-postfix-user-format))
+ tramp-postfix-user-format)
(when host
(if (string-match tramp-ipv6-regexp host)
(concat
(tramp-prefix-ipv6-format) host (tramp-postfix-ipv6-format))
host))
+ (unless (zerop (length port))
+ (concat tramp-prefix-port-format port))
(tramp-postfix-host-format)
(when localname localname)))
(tramp-make-tramp-file-name
(tramp-file-name-method vec)
(tramp-file-name-user vec)
+ (tramp-file-name-domain vec)
(tramp-file-name-host vec)
+ (tramp-file-name-port vec)
"/"))
(current-buffer))))
'hack-connection-local-variables-apply
`(:application tramp
:protocol ,(tramp-file-name-method vec)
- :user ,(tramp-file-name-user vec)
- :machine ,(tramp-file-name-host vec)))))
+ :user ,(tramp-file-name-user-domain vec)
+ :machine ,(tramp-file-name-host-port vec)))))
(defun tramp-set-connection-local-variables-for-buffer ()
"Set connection-local variables in the current buffer.
(defun tramp-debug-buffer-name (vec)
"A name for the debug buffer for VEC."
(let ((method (tramp-file-name-method vec))
- (user (tramp-file-name-user vec))
- (host (tramp-file-name-real-host vec)))
- (if (not (zerop (length user)))
- (format "*debug tramp/%s %s@%s*" method user host)
- (format "*debug tramp/%s %s*" method host))))
+ (user-domain (tramp-file-name-user-domain vec))
+ (host-port (tramp-file-name-host-port vec)))
+ (if (not (zerop (length user-domain)))
+ (format "*debug tramp/%s %s@%s*" method user-domain host-port)
+ (format "*debug tramp/%s %s*" method host-port))))
(defconst tramp-debug-outline-regexp
"[0-9]+:[0-9]+:[0-9]+\\.[0-9]+ [a-z0-9-]+ (\\([0-9]+\\)) #"
(setq fmt-string (concat fmt-string "\n%s")
arguments (append arguments (list (buffer-string)))))))
;; Do it.
- (when (vectorp vec-or-proc)
+ (when (tramp-file-name-p vec-or-proc)
(apply 'tramp-debug-message
vec-or-proc
(concat (format "(%d) # " level) fmt-string)
(save-window-excursion
(let* ((buf (or (and (bufferp buf) buf)
(and (processp vec-or-proc) (process-buffer vec-or-proc))
- (and (vectorp vec-or-proc)
+ (and (tramp-file-name-p vec-or-proc)
(tramp-get-connection-buffer vec-or-proc))))
- (vec (or (and (vectorp vec-or-proc) vec-or-proc)
+ (vec (or (and (tramp-file-name-p vec-or-proc) vec-or-proc)
(and buf (with-current-buffer buf
(tramp-dissect-file-name default-directory))))))
(unwind-protect
(discard-input)
(sit-for 30)))
;; Reset timestamp. It would be wrong after waiting for a while.
- (when (equal (butlast (append vec nil) 2)
- (car tramp-current-connection))
+ (when (tramp-file-name-equal-p vec (car tramp-current-connection))
(setcdr tramp-current-connection (current-time)))))))
(defmacro with-parsed-tramp-file-name (filename var &rest body)
`(,(if var (intern (format "%s-%s" var elem)) elem)
(,(intern (format "tramp-file-name-%s" elem))
,(or var 'v))))
- '(method user host localname hop))))
+ '(method user domain host port localname hop))))
`(let* ((,(or var 'v) (tramp-dissect-file-name ,filename))
,@bindings)
;; We don't know which of those vars will be used, so we bind them all,
(save-match-data
(when (string-match (nth 0 structure) name)
- (let ((method (and (nth 1 structure)
- (match-string (nth 1 structure) name)))
- (user (and (nth 2 structure)
- (match-string (nth 2 structure) name)))
- (host (and (nth 3 structure)
- (match-string (nth 3 structure) name)))
- (localname (and (nth 4 structure)
- (match-string (nth 4 structure) name))))
- (vector method user host localname nil)))))
+ (make-tramp-file-name
+ :method (and (nth 1 structure)
+ (match-string (nth 1 structure) name))
+ :user (and (nth 2 structure)
+ (match-string (nth 2 structure) name))
+ :host (and (nth 3 structure)
+ (match-string (nth 3 structure) name))))))
;; This function returns all possible method completions, adding the
;; trailing method delimiter.
(tramp-make-tramp-file-name
(tramp-file-name-method v)
(tramp-file-name-user v)
+ (tramp-file-name-domain v)
(tramp-file-name-host v)
+ (tramp-file-name-port v)
(if (and (zerop (length (tramp-file-name-localname v)))
(not (tramp-connectable-p file)))
""
(tramp-make-tramp-file-name
(tramp-file-name-method v)
(tramp-file-name-user v)
+ (tramp-file-name-domain v)
(tramp-file-name-host v)
+ (tramp-file-name-port v)
(tramp-run-real-handler
'file-name-directory (list (or (tramp-file-name-localname v) "")))
(tramp-file-name-hop v))))
(and (or (not connected) c)
(cond
((eq identification 'method) method)
- ((eq identification 'user) user)
- ((eq identification 'host) host)
+ ;; Domain and port are appended.
+ ((eq identification 'user) (tramp-file-name-user-domain v))
+ ((eq identification 'host) (tramp-file-name-host-port v))
((eq identification 'localname) localname)
((eq identification 'hop) hop)
- (t (tramp-make-tramp-file-name method user host "" hop)))))))))
+ (t (tramp-make-tramp-file-name
+ method user domain host port "" hop)))))))))
(defun tramp-handle-file-symlink-p (filename)
"Like `file-symlink-p' for Tramp files."
(let ((x (tramp-compat-file-attribute-type (file-attributes filename))))
(when (stringp x)
(if (file-name-absolute-p x)
- (tramp-make-tramp-file-name method user host x)
+ (tramp-make-tramp-file-name method user domain host port x)
x)))))
(defun tramp-handle-find-backup-file-name (filename)
(if (and (stringp (cdr x))
(file-name-absolute-p (cdr x))
(not (tramp-file-name-p (cdr x))))
- (tramp-make-tramp-file-name method user host (cdr x))
+ (tramp-make-tramp-file-name
+ method user domain host port (cdr x))
(cdr x))))
tramp-backup-directory-alist)
backup-directory-alist)))
((stringp remote-copy)
(file-local-copy
(tramp-make-tramp-file-name
- method user host remote-copy)))
+ method user domain host port remote-copy)))
((stringp tramp-temp-buffer-file-name)
(copy-file
filename tramp-temp-buffer-file-name 'ok)
(delete-file local-copy))
(when (stringp remote-copy)
(delete-file
- (tramp-make-tramp-file-name method user host remote-copy)))))
+ (tramp-make-tramp-file-name
+ method user domain host port remote-copy)))))
;; Result.
(list (expand-file-name filename)
(tramp-set-connection-property
(tramp-dissect-file-name
(tramp-make-tramp-file-name
- tramp-current-method tramp-current-user tramp-current-host ""))
+ tramp-current-method tramp-current-user tramp-current-domain
+ tramp-current-host tramp-current-port ""))
"first-password-request" t)
(save-restriction
(with-tramp-progress-reporter
(tramp-make-tramp-file-name
(tramp-file-name-method vec)
(tramp-file-name-user vec)
+ (tramp-file-name-domain vec)
(tramp-file-name-host vec)
+ (tramp-file-name-port vec)
(tramp-file-name-localname vec)
(tramp-file-name-hop vec))
(intern suffix))))
;;;###tramp-autoload
(defun tramp-local-host-p (vec)
"Return t if this points to the local host, nil otherwise."
- ;; We cannot use `tramp-file-name-real-host'. A port is an
- ;; indication for an ssh tunnel or alike.
- (let ((host (tramp-file-name-host vec)))
+ (let ((host (tramp-file-name-host vec))
+ (port (tramp-file-name-port vec)))
(and
(stringp host)
(string-match tramp-local-host-regexp host)
+ ;; A port is an indication for an ssh tunnel or alike.
+ (null port)
;; The method shall be applied to one of the shell file name
;; handlers. `tramp-local-host-p' is also called for "smb" and
;; alike, where it must fail.
(tramp-make-tramp-file-name
(tramp-file-name-method vec)
(tramp-file-name-user vec)
- host
+ (tramp-file-name-domain vec)
+ host port
(tramp-compat-temporary-file-directory)))
;; On some systems, chown runs only for root.
(or (zerop (user-uid))
(let ((dir (tramp-make-tramp-file-name
(tramp-file-name-method vec)
(tramp-file-name-user vec)
+ (tramp-file-name-domain vec)
(tramp-file-name-host vec)
+ (tramp-file-name-port vec)
(or (tramp-get-method-parameter vec 'tramp-tmpdir) "/tmp")
(tramp-file-name-hop vec))))
(or (and (file-directory-p dir) (file-writable-p dir)
are written with verbosity of 6."
(let ((default-directory (tramp-compat-temporary-file-directory))
(v (or vec
- (vector tramp-current-method tramp-current-user
- tramp-current-host nil nil)))
+ (tramp-make-tramp-file-name
+ tramp-current-method tramp-current-user tramp-current-domain
+ tramp-current-host tramp-current-port nil nil)))
(destination (if (eq destination t) (current-buffer) destination))
output error result)
(tramp-message
are written with verbosity of 6."
(let ((default-directory (tramp-compat-temporary-file-directory))
(v (or vec
- (vector tramp-current-method tramp-current-user
- tramp-current-host nil nil)))
+ (tramp-make-tramp-file-name
+ tramp-current-method tramp-current-user tramp-current-domain
+ tramp-current-host tramp-current-port nil nil)))
(buffer (if (eq buffer t) (current-buffer) buffer))
result)
(tramp-message
Invokes `password-read' if available, `read-passwd' else."
(let* ((case-fold-search t)
(key (tramp-make-tramp-file-name
- tramp-current-method tramp-current-user
- tramp-current-host ""))
+ tramp-current-method tramp-current-user tramp-current-domain
+ tramp-current-host tramp-current-port ""))
(pw-prompt
(or prompt
(with-current-buffer (process-buffer proc)
"Clear password cache for connection related to VEC."
(let ((method (tramp-file-name-method vec))
(user (tramp-file-name-user vec))
+ (domain (tramp-file-name-domain vec))
(host (tramp-file-name-host vec))
+ (port (tramp-file-name-port vec))
(hop (tramp-file-name-hop vec)))
(when hop
;; Clear also the passwords of the hops.
`(:max 1 ,(and user :user) ,user :host ,host :port ,method))
(tramp-compat-funcall
'auth-source-forget-user-or-password "password" host method))
- (password-cache-remove (tramp-make-tramp-file-name method user host ""))))
+ (password-cache-remove
+ (tramp-make-tramp-file-name method user domain host port ""))))
;; Snarfed code from time-date.el.
;; <http://www.mail-archive.com/tramp-devel@nongnu.org/msg01041.html>.
;; (Bug#6850)
;;
-;; * Use also port to distinguish connections. This is needed for
-;; different hosts sitting behind a single router (distinguished by
-;; different port numbers). (Tzvi Edelman)
-;; Also needed for different systems serve SSH on different ports of
-;; the same IP address. (Bug#27009)
-;;
;; * Refactor code from different handlers. Start with
;; *-process-file. One idea is to generalize `tramp-send-command'
;; and friends, for most of the handlers this is the major