]> code.delx.au - gnu-emacs/blobdiff - lisp/net/tramp-cmds.el
Merge changes from emacs-23 branch
[gnu-emacs] / lisp / net / tramp-cmds.el
index 72e57799dc407f18a2aee6c7f520cb2818d794f5..5937a737b961f4a2080ec7c29d65ffa4564935b0 100644 (file)
@@ -1,16 +1,17 @@
 ;;; tramp-cmds.el --- Interactive commands for Tramp
 
-;; Copyright (C) 2007 Free Software Foundation, Inc.
+;; Copyright (C) 2007, 2008, 2009, 2010 Free Software Foundation, Inc.
 
 ;; Author: Michael Albinus <michael.albinus@gmx.de>
 ;; Keywords: comm, processes
+;; Package: tramp
 
 ;; This file is part of GNU Emacs.
 
-;; GNU Emacs is free software; you can redistribute it and/or modify
+;; GNU Emacs is free software: you can redistribute it and/or modify
 ;; it under the terms of the GNU General Public License as published by
-;; the Free Software Foundation; either version 3, or (at your option)
-;; any later version.
+;; the Free Software Foundation, either version 3 of the License, or
+;; (at your option) any later version.
 
 ;; GNU Emacs is distributed in the hope that it will be useful,
 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
@@ -18,8 +19,7 @@
 ;; GNU General Public License for more details.
 
 ;; You should have received a copy of the GNU General Public License
-;; along with GNU Emacs; see the file COPYING.  If not, see
-;; <http://www.gnu.org/licenses/>.
+;; along with GNU Emacs.  If not, see <http://www.gnu.org/licenses/>.
 
 ;;; Commentary:
 
@@ -50,6 +50,7 @@
          x)))
     (buffer-list))))
 
+;;;###tramp-autoload
 (defun tramp-cleanup-connection (vec)
   "Flush all connection related objects.
 This includes password cache, file cache, connection cache, buffers.
@@ -61,12 +62,12 @@ When called interactively, a Tramp connection has to be selected."
     (let ((connections
           (mapcar
            (lambda (x)
-             (with-current-buffer x (list (file-remote-p default-directory))))
-           ;; We shall not count debug buffers, because their
-           ;; default-directory is random.  It could be even a remote
-           ;; one from another connection.
-           (all-completions
-            "*tramp" (mapcar 'list (tramp-list-tramp-buffers)))))
+             (tramp-make-tramp-file-name
+              (tramp-file-name-method x)
+              (tramp-file-name-user x)
+              (tramp-file-name-host x)
+              (tramp-file-name-localname x)))
+           (tramp-list-connections)))
          name)
 
       (when connections
@@ -85,11 +86,11 @@ When called interactively, a Tramp connection has to be selected."
     (tramp-clear-passwd vec)
 
     ;; Flush file cache.
-    (tramp-flush-directory-property vec "/")
+    (tramp-flush-directory-property vec "")
 
     ;; Flush connection cache.
-    (tramp-flush-connection-property (tramp-get-connection-process vec) nil)
-    (tramp-flush-connection-property vec nil)
+    (tramp-flush-connection-property (tramp-get-connection-process vec))
+    (tramp-flush-connection-property vec)
 
     ;; Remove buffers.
     (dolist
@@ -103,9 +104,11 @@ When called interactively, a Tramp connection has to be selected."
 This includes password cache, file cache, connection cache, buffers."
   (interactive)
 
+  ;; Unlock Tramp.
+  (setq tramp-locked nil)
+
   ;; Flush password cache.
-  (when (functionp 'password-reset)
-    (funcall (symbol-function 'password-reset)))
+  (tramp-compat-funcall 'password-reset)
 
   ;; Flush file and connection cache.
   (clrhash tramp-cache-data)
@@ -125,15 +128,279 @@ This includes password cache, file cache, connection cache, buffers."
   (dolist (name (tramp-list-remote-buffers))
     (when (bufferp (get-buffer name)) (kill-buffer name))))
 
+;; Tramp version is useful in a number of situations.
+
+;;;###tramp-autoload
+(defun tramp-version (arg)
+  "Print version number of tramp.el in minibuffer or current buffer."
+  (interactive "P")
+  (if arg (insert tramp-version) (message tramp-version)))
+
+;; Make the `reporter` functionality available for making bug reports about
+;; the package. A most useful piece of code.
+
+(autoload 'reporter-submit-bug-report "reporter")
+
+(defun tramp-bug ()
+  "Submit a bug report to the Tramp developers."
+  (interactive)
+  (require 'reporter)
+  (catch 'dont-send
+    (let ((reporter-prompt-for-summary-p t))
+      (reporter-submit-bug-report
+       tramp-bug-report-address                ; to-address
+       (format "tramp (%s)" tramp-version) ; package name and version
+       (delq nil
+            `(;; Current state
+              tramp-current-method
+              tramp-current-user
+              tramp-current-host
+
+              ;; System defaults
+              tramp-auto-save-directory        ; vars to dump
+              tramp-default-method
+              tramp-default-method-alist
+              tramp-default-host
+              tramp-default-proxies-alist
+              tramp-default-user
+              tramp-default-user-alist
+              tramp-rsh-end-of-line
+              tramp-default-password-end-of-line
+              tramp-login-prompt-regexp
+              ;; Mask non-7bit characters
+              (tramp-password-prompt-regexp . tramp-reporter-dump-variable)
+              tramp-wrong-passwd-regexp
+              tramp-yesno-prompt-regexp
+              tramp-yn-prompt-regexp
+              tramp-terminal-prompt-regexp
+              tramp-temp-name-prefix
+              tramp-file-name-structure
+              tramp-file-name-regexp
+              tramp-methods
+              tramp-end-of-output
+              tramp-local-coding-commands
+              tramp-remote-coding-commands
+              tramp-actions-before-shell
+              tramp-actions-copy-out-of-band
+              tramp-terminal-type
+              ;; Mask non-7bit characters
+              (tramp-shell-prompt-pattern . tramp-reporter-dump-variable)
+              ,(when (boundp 'tramp-backup-directory-alist)
+                 'tramp-backup-directory-alist)
+              ,(when (boundp 'tramp-bkup-backup-directory-info)
+                 'tramp-bkup-backup-directory-info)
+              ;; Dump cache.
+              (tramp-cache-data . tramp-reporter-dump-variable)
+
+              ;; Non-tramp variables of interest
+              ;; Mask non-7bit characters
+              (shell-prompt-pattern . tramp-reporter-dump-variable)
+              backup-by-copying
+              backup-by-copying-when-linked
+              backup-by-copying-when-mismatch
+              ,(when (boundp 'backup-by-copying-when-privileged-mismatch)
+                 'backup-by-copying-when-privileged-mismatch)
+              ,(when (boundp 'password-cache)
+                 'password-cache)
+              ,(when (boundp 'password-cache-expiry)
+                 'password-cache-expiry)
+              ,(when (boundp 'backup-directory-alist)
+                 'backup-directory-alist)
+              ,(when (boundp 'bkup-backup-directory-info)
+                 'bkup-backup-directory-info)
+              file-name-handler-alist))
+
+       'tramp-load-report-modules      ; pre-hook
+       'tramp-append-tramp-buffers     ; post-hook
+       "\
+Enter your bug report in this message, including as much detail
+as you possibly can about the problem, what you did to cause it
+and what the local and remote machines are.
+
+If you can give a simple set of instructions to make this bug
+happen reliably, please include those.  Thank you for helping
+kill bugs in Tramp.
+
+Before reproducing the bug, you might apply
+
+  M-x tramp-cleanup-all-connections
+
+This allows to investigate from a clean environment.  Another
+useful thing to do is to put
+
+  (setq tramp-verbose 9)
+
+in the ~/.emacs file and to repeat the bug.  Then, include the
+contents of the *tramp/foo* buffer and the *debug tramp/foo*
+buffer in your bug report.
+
+--bug report follows this line--
+"))))
+
+(defun tramp-reporter-dump-variable (varsym mailbuf)
+  "Pretty-print the value of the variable in symbol VARSYM.
+Used for non-7bit chars in strings."
+  (let* ((reporter-eval-buffer (symbol-value 'reporter-eval-buffer))
+        (val (with-current-buffer reporter-eval-buffer
+               (symbol-value varsym))))
+
+    (if (hash-table-p val)
+       ;; Pretty print the cache.
+       (set varsym (read (format "(%s)" (tramp-cache-print val))))
+      ;; There are characters to be masked.
+      (when (and (boundp 'mm-7bit-chars)
+                (string-match
+                 (concat "[^" (symbol-value 'mm-7bit-chars) "]") val))
+       (with-current-buffer reporter-eval-buffer
+         (set varsym (format "(base64-decode-string \"%s\""
+                             (base64-encode-string val))))))
+
+    ;; Dump variable.
+    (tramp-compat-funcall 'reporter-dump-variable varsym mailbuf)
+
+    (unless (hash-table-p val)
+      ;; Remove string quotation.
+      (forward-line -1)
+      (when (looking-at
+            (concat "\\(^.*\\)" "\""                       ;; \1 "
+                    "\\((base64-decode-string \\)" "\\\\"  ;; \2 \
+                    "\\(\".*\\)" "\\\\"                    ;; \3 \
+                    "\\(\")\\)" "\"$"))                    ;; \4 "
+       (replace-match "\\1\\2\\3\\4")
+       (beginning-of-line)
+       (insert " ;; variable encoded due to non-printable characters\n"))
+      (forward-line 1))
+
+    ;; Reset VARSYM to old value.
+    (with-current-buffer reporter-eval-buffer
+      (set varsym val))))
+
+(defun tramp-load-report-modules ()
+  "Load needed modules for reporting."
+
+  ;; We load message.el and mml.el from Gnus.
+  (if (featurep 'xemacs)
+      (progn
+       (load "message" 'noerror)
+       (load "mml" 'noerror))
+    (require 'message nil 'noerror)
+    (require 'mml nil 'noerror))
+  (tramp-compat-funcall 'message-mode)
+  (tramp-compat-funcall 'mml-mode t))
+
+(defun tramp-append-tramp-buffers ()
+  "Append Tramp buffers and buffer local variables into the bug report."
+
+  (goto-char (point-max))
+
+  ;; Dump buffer local variables.
+  (dolist (buffer
+          (delq nil
+                (mapcar
+                 '(lambda (b)
+                    (when (string-match "\\*tramp/" (buffer-name b)) b))
+                 (buffer-list))))
+    (let ((reporter-eval-buffer buffer)
+         (buffer-name (buffer-name buffer))
+         (elbuf (get-buffer-create " *tmp-reporter-buffer*")))
+      (with-current-buffer elbuf
+       (emacs-lisp-mode)
+       (erase-buffer)
+       (insert "\n(setq\n")
+       (lisp-indent-line)
+       (tramp-compat-funcall
+        'reporter-dump-variable 'buffer-name (current-buffer))
+       (dolist (varsym-or-cons-cell (buffer-local-variables buffer))
+         (let ((varsym (or (car-safe varsym-or-cons-cell)
+                           varsym-or-cons-cell)))
+           (when (string-match "tramp" (symbol-name varsym))
+             (tramp-compat-funcall
+              'reporter-dump-variable varsym (current-buffer)))))
+       (lisp-indent-line)
+       (insert ")\n"))
+      (insert-buffer-substring elbuf)))
+
+  ;; Append buffers only when we are in message mode.
+  (when (and
+        (eq major-mode 'message-mode)
+        (boundp 'mml-mode)
+        (symbol-value 'mml-mode))
+
+    (let ((tramp-buf-regexp "\\*\\(debug \\)?tramp/")
+         (buffer-list (tramp-compat-funcall 'tramp-list-tramp-buffers))
+         (curbuf (current-buffer)))
+
+      ;; There is at least one Tramp buffer.
+      (when buffer-list
+       (switch-to-buffer (list-buffers-noselect nil))
+       (delete-other-windows)
+       (setq buffer-read-only nil)
+       (goto-char (point-min))
+       (while (not (eobp))
+         (if (re-search-forward
+              tramp-buf-regexp (tramp-compat-line-end-position) t)
+             (forward-line 1)
+           (forward-line 0)
+           (let ((start (point)))
+             (forward-line 1)
+             (kill-region start (point)))))
+       (insert "
+The buffer(s) above will be appended to this message.  If you
+don't want to append a buffer because it contains sensitive data,
+or because the buffer is too large, you should delete the
+respective buffer.  The buffer(s) will contain user and host
+names.  Passwords will never be included there.")
+
+       (when (>= tramp-verbose 6)
+         (insert "\n\n")
+         (let ((start (point)))
+           (insert "\
+Please note that you have set `tramp-verbose' to a value of at
+least 6.  Therefore, the contents of files might be included in
+the debug buffer(s).")
+           (add-text-properties start (point) (list 'face 'italic))))
+
+       (set-buffer-modified-p nil)
+       (setq buffer-read-only t)
+       (goto-char (point-min))
+
+       (if (y-or-n-p "Do you want to append the buffer(s)? ")
+           ;; OK, let's send.  First we delete the buffer list.
+           (progn
+             (kill-buffer nil)
+             (switch-to-buffer curbuf)
+             (goto-char (point-max))
+             (insert "\n\
+This is a special notion of the `gnus/message' package.  If you
+use another mail agent (by copying the contents of this buffer)
+please ensure that the buffers are attached to your email.\n\n")
+             (dolist (buffer buffer-list)
+               (tramp-compat-funcall
+                'mml-insert-empty-tag 'part 'type "text/plain"
+                'encoding "base64" 'disposition "attachment" 'buffer buffer
+                'description buffer))
+             (set-buffer-modified-p nil))
+
+         ;; Don't send.  Delete the message buffer.
+         (set-buffer curbuf)
+         (set-buffer-modified-p nil)
+         (kill-buffer nil)
+         (throw 'dont-send nil))))))
+
+(defalias 'tramp-submit-bug 'tramp-bug)
+
+(add-hook 'tramp-unload-hook
+         (lambda () (unload-feature 'tramp-cmds 'force)))
+
 (provide 'tramp-cmds)
 
 ;;; TODO:
 
 ;; * Clean up unused *tramp/foo* buffers after a while.  (Pete Forman)
-;; * WIBNI there was an interactive command prompting for tramp
+;; * WIBNI there was an interactive command prompting for Tramp
 ;;   method, hostname, username and filename and translates the user
 ;;   input into the correct filename syntax (depending on the Emacs
-;;   flavor) (Reiner Steib)
+;;   flavor)  (Reiner Steib)
 ;; * Let the user edit the connection properties interactively.
 ;;   Something like `gnus-server-edit-server' in Gnus' *Server* buffer.
 ;; * It's just that when I come to Customize `tramp-default-user-alist'
@@ -142,7 +409,7 @@ This includes password cache, file cache, connection cache, buffers."
 ;;   Option and should not be modified by the code.  add-to-list is
 ;;   called in several places. One way to handle that is to have a new
 ;;   ordinary variable that gets its initial value from
-;;   tramp-default-user-alist and then is added to. (Pete Forman)
+;;   tramp-default-user-alist and then is added to.  (Pete Forman)
 
 ;; arch-tag: 190d4c33-76bb-4e99-8b6f-71741f23d98c
 ;;; tramp-cmds.el ends here