Homepage: https://www.gnu.org/software/emacs
Author: Andy Norman
Transparent FTP support for GNU Emacs
This package attempts to make accessing files and directories using FTP from within GNU Emacs as simple and transparent as possible. A subset of the common file-handling routines are extended to interact with FTP. Usage: Some of the common GNU Emacs file-handling operations have been made FTP-smart. If one of these routines is given a filename that matches '/user@host:name' then it will spawn an FTP process connecting to machine 'host' as account 'user' and perform its operation on the file 'name'. For example: if find-file is given a filename of: /ange@anorman:/tmp/notes then ange-ftp spawns an FTP process, connects to the host 'anorman' as user 'ange', gets the file '/tmp/notes' and pops up a buffer containing the contents of that file as if it were on the local filesystem. If ange-ftp needs a password to connect then it reads one in the echo area. Extended filename syntax: The default extended filename syntax is '/user@host:name', where the 'user@' part may be omitted. This syntax can be customized to a certain extent by changing ange-ftp-name-format. There are limitations. The `host' part has an optional suffix `#port' which may be used to specify a non-default port number for the connection. If the user part is omitted then ange-ftp generates a default user instead whose value depends on the variable ange-ftp-default-user. Passwords: A password is required for each host/user pair. Ange-ftp reads passwords as needed. You can also specify a password with ange-ftp-set-passwd, or in a *valid* ~/.netrc file. Passwords for user "anonymous": Passwords for the user "anonymous" (or "ftp") are handled specially. The variable `ange-ftp-generate-anonymous-password' controls what happens: if the value of this variable is a string, then this is used as the password; if non-nil (the default), then the value of `user-mail-address' is used; if nil then the user is prompted for a password as normal. "Dumb" UNIX hosts: The FTP servers on some UNIX machines have problems if the 'ls' command is used. The routine ange-ftp-add-dumb-unix-host can be called to tell ange-ftp to limit itself to the DIR command and not 'ls' for a given UNIX host. Note that this change will take effect for the current GNU Emacs session only. See below for a discussion of non-UNIX hosts. If a large number of machines with similar hostnames have this problem then it is easier to set the value of ange-ftp-dumb-unix-host-regexp in your init file. ange-ftp is unable to automatically recognize dumb unix hosts. File name completion: Full file-name completion is supported on UNIX, VMS, CMS, and MTS hosts. To do filename completion, ange-ftp needs a listing from the remote host. Therefore, for very slow connections, it might not save any time. FTP processes: When ange-ftp starts up an FTP process, it leaves it running for speed purposes. Some FTP servers will close the connection after a period of time, but ange-ftp should be able to quietly reconnect the next time that the process is needed. Killing the "*ftp user@host*" buffer also kills the ftp process. This should not cause ange-ftp any grief. Binary file transfers: By default ange-ftp transfers files in ASCII mode. If a file being transferred matches the value of ange-ftp-binary-file-name-regexp then binary mode is used for that transfer. Account passwords: Some FTP servers require an additional password which is sent by the ACCOUNT command. ange-ftp partially supports this by allowing the user to specify an account password by either calling ange-ftp-set-account, or by specifying an account token in the .netrc file. If the account password is set by either of these methods then ange-ftp will issue an ACCOUNT command upon starting the FTP process. Preloading: ange-ftp can be preloaded, but must be put in the site-init.el file and not the site-load.el file in order for the documentation strings for the functions being overloaded to be available. Status reports: Most ange-ftp commands that talk to the FTP process output a status message on what they are doing. In addition, ange-ftp can take advantage of the FTP client's HASH command to display the status of transferring files and listing directories. See the documentation for the variables ange-ftp-{ascii,binary}-hash-mark-size, ange-ftp-send-hash and ange-ftp-process-verbose for more details. Gateways: Sometimes it is necessary for the FTP process to be run on a different machine than the machine running GNU Emacs. This can happen when the local machine has restrictions on what hosts it can access. ange-ftp has support for running the ftp process on a different (gateway) machine. The way it works is as follows: 1) Set the variable 'ange-ftp-gateway-host' to the name of a machine that doesn't have the access restrictions. 2) Set the variable 'ange-ftp-local-host-regexp' to a regular expression that matches hosts that can be contacted from running a local ftp process, but fails to match hosts that can't be accessed locally. For example: "\\.hp\\.com$\\|^[^.]*$" will match all hosts that are in the .hp.com domain, or don't have an explicit domain in their name, but will fail to match hosts with explicit domains or that are specified by their ip address. 3) Using NFS and symlinks, make sure that there is a shared directory with the *same* name between the local machine and the gateway machine. This directory is necessary for temporary files created by ange-ftp. 4) Set the variable 'ange-ftp-gateway-tmp-name-template' to the name of this directory plus an identifying filename prefix. For example: "/nfs/hplose/ange/ange-ftp" where /nfs/hplose/ange is a directory that is shared between the gateway machine and the local machine. The simplest way of getting a ftp process running on the gateway machine is if you can spawn a remote shell using either 'rsh' or 'remsh'. If you can't do this for some reason such as security then points 7 onwards will discuss an alternative approach. 5) Set the variable ange-ftp-gateway-program to the name of the remote shell process such as 'remsh' or 'rsh' if the default isn't correct. 6) Set the variable ange-ftp-gateway-program-interactive to nil if it isn't already. This tells ange-ftp that you are using a remote shell rather than logging in using telnet or rlogin. That should be all you need to allow ange-ftp to spawn a ftp process on the gateway machine. If you have to use telnet or rlogin to get to the gateway machine then follow the instructions below. 7) Set the variable ange-ftp-gateway-program to the name of the program that lets you log onto the gateway machine. This may be something like telnet or rlogin. 8) Set the variable ange-ftp-gateway-prompt-pattern to a regular expression that matches the prompt you get when you login to the gateway machine. Be very specific here; this regexp must not match *anything* in your login banner except this prompt. shell-prompt-pattern is far too general as it appears to match some login banners from Sun machines. For example: "^$*$ *" 9) Set the variable ange-ftp-gateway-program-interactive to t to let ange-ftp know that it has to "hand-hold" the login to the gateway machine. 10) Set the variable ange-ftp-gateway-setup-term-command to a UNIX command that will put the pty connected to the gateway machine into a no-echoing mode, and will strip off carriage-returns from output from the gateway machine. For example: "stty -onlcr -echo" will work on HP-UX machines, whereas: "stty -echo nl" appears to work for some Sun machines. That's all there is to it. Smart gateways: If you have a "smart" ftp program that allows you to issue commands like "USER foo@bar" which do nice proxy things, then look at the variables ange-ftp-smart-gateway and ange-ftp-smart-gateway-port. Otherwise, if there is an alternate ftp program that implements proxy in a transparent way (i.e. without specifying the proxy host), that will connect you directly to the desired destination host: Set ange-ftp-gateway-ftp-program-name to that program's name. Set ange-ftp-local-host-regexp to a value as stated earlier on. Leave ange-ftp-gateway-host set to nil. Set ange-ftp-smart-gateway to t. Tips for using ange-ftp: 1. For Dired to work on a host which marks symlinks with a trailing @ in an ls -alF listing, you need to (setq dired-ls-F-marks-symlinks t). Most UNIX systems do not do this, but ULTRIX does. If you think that there is a chance you might connect to an ULTRIX machine (such as prep.ai.mit.edu), then set this variable accordingly. This will have the side effect that Dired will have problems with symlinks whose names end in an @. If you get yourself into this situation then editing Dired's ls-switches to remove "F", will temporarily fix things. 2. If you know that you are connecting to a certain non-UNIX machine frequently, and ange-ftp seems to be unable to guess its host-type, then setting the appropriate host-type regexp (ange-ftp-vms-host-regexp, ange-ftp-mts-host-regexp, or ange-ftp-cms-host-regexp) accordingly should help. Also, please report ange-ftp's inability to recognize the host-type as a bug. 3. For slow connections, you might get "listing unreadable" error messages, or get an empty buffer for a file that you know has something in it. The solution is to increase the value of ange-ftp-retry-time. Its default value is 5 which is plenty for reasonable connections. However, for some transatlantic connections I set this to 20. 4. Beware of compressing files on non-UNIX hosts. Ange-ftp will do it by copying the file to the local machine, compressing it there, and then sending it back. Binary file transfers between machines of different architectures can be a risky business. Test things out first on some test files. See "Bugs" below. Also, note that ange-ftp copies files by moving them through the local machine. Again, be careful when doing this with binary files on non-Unix machines. 5. Beware that Dired over ftp will use your setting of dired-no-confirm (list of Dired commands for which confirmation is not asked). You might want to reconsider your setting of this variable, because you might want confirmation for more commands on remote direds than on local direds. For example, I strongly recommend that you not include compress and uncompress in this list. If there is enough demand it might be a good idea to have an alist ange-ftp-dired-no-confirm of pairs ( TYPE . LIST ), where TYPE is an operating system type and LIST is a list of commands for which confirmation would be suppressed. Then remote Dired listings would take their (buffer-local) value of dired-no-confirm from this alist. Who votes for this? --------------------------------------------------------------------- Non-UNIX support: --------------------------------------------------------------------- VMS support: Ange-ftp has full support for VMS hosts. It should be able to automatically recognize any VMS machine. However, if it fails to do this, you can use the command ange-ftp-add-vms-host. Also, you can set the variable ange-ftp-vms-host-regexp in your init file. We would be grateful if you would report any failures to automatically recognize a VMS host as a bug. Filename Syntax: For ease of *implementation*, the user enters the VMS filename syntax in a UNIX-y way. For example: PUB$:[ANONYMOUS.SDSCPUB.NEXT]README.TXT;1 would be entered as: /PUB$$:/ANONYMOUS/SDSCPUB/NEXT/README.TXT;1 i.e. to log in as anonymous on ymir.claremont.edu and grab the file: [.CSV.POLICY]RULES.MEM you would type: C-x C-f /anonymous@ymir.claremont.edu:CSV/POLICY/RULES.MEM A valid VMS filename is of the form: FILE.TYPE;## where FILE can be up to 39 characters TYPE can be up to 39 characters ## is a version number (an integer between 1 and 32,767) Valid characters in FILE and TYPE are A-Z 0-9 _ - $ $ cannot begin a filename, and - cannot be used as the first or last character. Tips: 1. Although VMS is not case sensitive, EMACS running under UNIX is. Therefore, to access a VMS file, you must enter the filename with upper case letters. 2. To access the latest version of file under VMS, you use the filename without the ";" and version number. You should always edit the latest version of a file. If you want to edit an earlier version, copy it to a new file first. This has nothing to do with ange-ftp, but is simply good VMS operating practice. Therefore, to edit FILE.TXT;3 (say 3 is latest version), do C-x C-f /ymir.claremont.edu:FILE.TXT. If you inadvertently do C-x C-f /ymir.claremont.edu:FILE.TXT;3, you will find that VMS will not allow you to save the file because it will refuse to overwrite FILE.TXT;3, but instead will want to create FILE.TXT;4, and attach the buffer to this file. To get out of this situation, M-x write-file /ymir.claremont.edu:FILE.TXT will attach the buffer to latest version of the file. For this reason, in Dired "f" (dired-find-file), always loads the file sans version, whereas "v", (dired-view-file), always loads the explicit version number. The reasoning being that it reasonable to view old versions of a file, but not to edit them. 3. EMACS has a feature in which it does environment variable substitution in filenames. Therefore, to enter a $ in a filename, you must quote it by typing $$. MTS support: Ange-ftp has full support for hosts running the Michigan terminal system. It should be able to automatically recognize any MTS machine. However, if it fails to do this, you can use the command ange-ftp-add-mts-host. As well, you can set the variable ange-ftp-mts-host-regexp in your init file. We would be grateful if you would report any failures to automatically recognize a MTS host as a bug. Filename syntax: MTS filenames are entered in a UNIX-y way. For example, if your account was YYYY, the file FILE in the account XXXX: on mtsg.ubc.ca would be entered as /YYYY@mtsg.ubc.ca:/XXXX:/FILE In other words, MTS accounts are treated as UNIX directories. Of course, to access a file in another account, you must have access permission for it. If FILE were in your own account, then you could enter it in a relative name fashion as /YYYY@mtsg.ubc.ca:FILE MTS filenames can be up to 12 characters. Like UNIX, the structure of the filename does not contain a TYPE (i.e. it can have as many "."'s as you like.) MTS filenames are always in upper case, and hence be sure to enter them as such! MTS is not case sensitive, but an EMACS running under UNIX is. CMS support: Ange-ftp has full support for hosts running CMS. It should be able to automatically recognize any CMS machine. However, if it fails to do this, you can use the command ange-ftp-add-cms-host. As well, you can set the variable ange-ftp-cms-host-regexp in your init file. We would be grateful if you would report any failures to automatically recognize a CMS host as a bug. Filename syntax: CMS filenames are entered in a UNIX-y way. In other words, minidisks are treated as UNIX directories. For example to access the file READ.ME in minidisk *.311 on cuvmb.cc.columbia.edu, you would enter /anonymous@cuvmb.cc.columbia.edu:/*.311/READ.ME If *.301 is the default minidisk for this account, you could access FOO.BAR on this minidisk as /anonymous@cuvmb.cc.columbia.edu:FOO.BAR CMS filenames are of the form FILE.TYPE, where both FILE and TYPE can be up to 8 characters. Again, beware that CMS filenames are always upper case, and hence must be entered as such. Tips: 1. CMS machines, with the exception of anonymous accounts, nearly always need an account password. To have ange-ftp send an account password, you can either include it in your .netrc file, or use ange-ftp-set-account. 2. Ange-ftp cannot send "write passwords" for a minidisk. Hopefully, we can fix this. BS2000 support: Ange-ftp has full support for BS2000 hosts. It should be able to automatically recognize any BS2000 machine. However, if it fails to do this, you can use the command ange-ftp-add-bs2000-host. As well, you can set the variable ange-ftp-bs2000-host-regexp in your .emacs file. We would be grateful if you would report any failures to auto- matically recognize a BS2000 host as a bug. If you want to access the POSIX subsystem on BS2000 you MUST use command ange-ftp-add-bs2000-posix-host for that particular hostname. ange-ftp can't decide if you want to access the native filesystem or the POSIX filesystem, so it accesses the native filesystem by default. And if you have an ASCII filesystem in your BS2000 POSIX subsystem you must use ange-ftp-binary-file-name-regexp to access its files. Filename Syntax: For ease of *implementation*, the user enters the BS2000 filename syntax in a UNIX-y way. For example: :PUB:$PUBLIC.ANONYMOUS.SDSCPUB.NEXT.README.TXT would be entered as: /:PUB:/$$PUBLIC/ANONYMOUS.SDSCPUB.NEXT.README.TXT You don't have to type pubset and account, if they have default values, i.e. to log in as anonymous on bs2000.anywhere.com and grab the file IMPORTANT.TEXT.ON.BS2000 on the default pubset X on userid PUBLIC (there are only 8 characters in a valid username), you could type: C-x C-f /public@bs2000.anywhere.com:/IMPORTANT.TEXT.ON.BS2000 or C-x C-f /anonym@bs2000.anywhere.com:/:X:/$$PUBLIC/IMPORTANT.TEXT.ON.BS2000 If X is not your default pubset, you could add it as 'subdirectory' (BS2000 has a flat architecture) with the command (setq ange-ftp-bs2000-additional-pubsets '(":X:")) and then you could type: C-x C-f /anonym@bs2000.anywhere.com:/:X:/IMPORTANT.TEXT.ON.BS2000 Valid characters in an BS2000 filename are A-Z 0-9 $ # @ . - If the first character in a filename is # or @, this is replaced with ange-ftp-bs2000-special-prefix because names starting with # or @ are reserved for temporary files. This is especially important for auto-save files. Valid file generations are ending with ([+|-|*]0-9...) . File generations are not supported yet! A filename must at least contain one character (A-Z) and cannot be longer than 41 characters. Tips: 1. Although BS2000 is not case sensitive, EMACS running under UNIX is. Therefore, to access a BS2000 file, you must enter the filename with upper case letters. 2. EMACS has a feature in which it does environment variable substitution in filenames. Therefore, to enter a $ in a filename, you must quote it by typing $$. 3. BS2000 machines, with the exception of anonymous accounts, nearly always need an account password. To have ange-ftp send an account password, you can either include it in your .netrc file, or use ange-ftp-set-account. ------------------------------------------------------------------ Bugs: ------------------------------------------------------------------ 1. Umask problems: Be warned that files created by using ange-ftp will take account of the umask of the ftp daemon process rather than the umask of the creating user. This is particularly important when logging in as the root user. The way that I tighten up the ftp daemon's umask under HP-UX is to make sure that the umask is changed to 027 before I spawn /etc/inetd. I suspect that there is something similar on other systems. 2. Some combinations of FTP clients and servers break and get out of sync when asked to list a non-existent directory. Some of the ai.mit.edu machines cause this problem for some FTP clients. Using ange-ftp-kill-ftp-process can restart the ftp process, which should get things back in sync. 3. Ange-ftp does not check to make sure that when creating a new file, you provide a valid filename for the remote operating system. If you do not, then the remote FTP server will most likely translate your filename in some way. This may cause ange-ftp to get confused about what exactly is the name of the file. The most common causes of this are using lower case filenames on systems which support only upper case, and using filenames which are too long. 4. Null (blank) passwords confuse both ange-ftp and some FTP daemons. 5. Ange-ftp likes to use pty's to talk to its FTP processes. If GNU Emacs for some reason creates a FTP process that only talks via pipes then ange-ftp won't be getting the information it requires at the time that it wants it since pipes flush at different times to pty's. One disgusting way around this problem is to talk to the FTP process via rlogin which does the 'right' things with pty's. 6. For CMS support, we send too many cd's. Since cd's are cheap, I haven't worried about this too much. Eventually, we should have some caching of the current minidisk. 7. Some CMS machines do not assign a default minidisk when you ftp them as anonymous. It is then necessary to guess a valid minidisk name, and cd to it. This is (understandably) beyond ange-ftp. 8. Remote to remote copying of files on non-Unix machines can be risky. Depending on the variable ange-ftp-binary-file-name-regexp, ange-ftp will use binary mode for the copy. Between systems of different architecture, this still may not be enough to guarantee the integrity of binary files. Binary file transfers from VMS machines are particularly problematical. Should ange-ftp-binary-file-name-regexp be an alist of OS type, regexp pairs? 9. The code to do compression of files over ftp is not as careful as it should be. It deletes the old remote version of the file, before actually checking if the local to remote transfer of the compressed file succeeds. Of course to delete the original version of the file after transferring the compressed version back is also dangerous, because some OS's have severe restrictions on the length of filenames, and when the compressed version is copied back the "-Z" or ".Z" may be truncated. Then, ange-ftp would delete the only remaining version of the file. Maybe ange-ftp should make backups when it compresses files (of course, the backup "~" could also be truncated off, sigh...). Suggestions? 10. If a dir listing is attempted for an empty directory on (at least some) VMS hosts, an ftp error is given. This is really an ftp bug, and I don't know how to get ange-ftp work to around it. 11. Bombs on filenames that start with a space. Deals well with filenames containing spaces, but beware that the remote ftpd may not like them much. 12. The Dired support for non-Unix-like systems does not currently work. It needs to be reimplemented by modifying the parse-...-listing functions to convert the directory listing to ls -l format. 13. The famous @ bug. As mentioned above in TIPS, ULTRIX marks symlinks with a trailing @ in a ls -alF listing. In order to account for this ange-ftp looks to chop trailing @'s off of symlink names when it is parsing a listing with the F switch. This will cause ange-ftp to incorrectly get the name of a symlink on a non-ULTRIX host if its name ends in an @. ange-ftp will correct itself if you take F out of the Dired ls switches (C-u s will allow you to edit the switches). The Dired buffer will be automatically reverted, which will allow ange-ftp to fix its files hashtable. A cookie to anyone who can think of a fast, sure-fire way to recognize ULTRIX over ftp. If you find any bugs or problems with this package, PLEASE report a bug to the Emacs maintainers via M-x report-emacs-bug. ----------------------------------------------------------- Technical information on this package: ----------------------------------------------------------- ange-ftp works by putting a handler on file-name-handler-alist which is called by many primitives, and a few non-primitives, whenever they see a file name of the appropriate sort. Checklist for adding non-UNIX support for TYPE The following functions may need TYPE versions: (not all functions will be needed for every OS) ange-ftp-fix-name-for-TYPE ange-ftp-fix-dir-name-for-TYPE ange-ftp-TYPE-host ange-ftp-TYPE-add-host ange-ftp-parse-TYPE-listing ange-ftp-TYPE-delete-file-entry ange-ftp-TYPE-add-file-entry ange-ftp-TYPE-file-name-as-directory ange-ftp-TYPE-make-compressed-filename ange-ftp-TYPE-file-name-sans-versions Variables: ange-ftp-TYPE-host-regexp May need to add TYPE to ange-ftp-dumb-host-types Check the following functions for OS dependent coding: ange-ftp-host-type ange-ftp-guess-host-type ange-ftp-allow-child-lookup Host type conventions: The function ange-ftp-host-type and the variable ange-ftp-dired-host-type (mostly) follow the following conventions for remote host types. At least, I think that future code should try to follow these conventions, and the current code should eventually be made compliant. nil = local host type, whatever that is (probably unix). Think nil as in "not a remote host". This value is used by ange-ftp-dired-host-type for local buffers. t = a remote host of unknown type. Think t as in true, it's remote. Currently, `unix' is used as the default remote host type. Maybe we should use t. TYPE = a remote host of TYPE type. TYPE:LIST = a remote host of TYPE type, using a specialized ftp listing program called list. This is currently only used for Unix dl (descriptive listings), when ange-ftp-dired-host-type is set to `unix:dl'. Bug report codes: Because of their naive faith in this code, there are certain situations which the writers of this program believe could never happen. However, being realists they have put calls to `error' in the program at these points. These errors provide a code, which is an integer, greater than 1. To aid debugging. the error codes, and the functions in which they reside are listed below. 1: See ange-ftp-ls ----------------------------------------------------------- Hall of fame: ----------------------------------------------------------- Thanks to Roland McGrath for improving the filename syntax handling, for suggesting many enhancements and for numerous cleanups to the code. Thanks to Jamie Zawinski for bugfixes and for ideas such as gateways. Thanks to Ken Laprade for improved .netrc parsing, password reading, and Dired / shell auto-loading. Thanks to Sebastian Kremer for Dired support and for many ideas and bugfixes. Thanks to Joe Wells for bugfixes, the original non-UNIX system support, VOS support, and hostname completion. Thanks to Nakagawa Takayuki for many good ideas, filename-completion, help with file-name expansion, efficiency worries, stylistic concerns and many bugfixes. Thanks to Sandy Rutherford who re-wrote most of ange-ftp to support VMS, MTS, CMS and UNIX-dls. Sandy also added dired-support for non-UNIX OS and auto-recognition of the host type. Thanks to Dave Smith who wrote the info file for ange-ftp. Finally, thanks to Keith Waclena, Mark D. Baushke, Terence Kelleher, Ping Zhou, Edward Vielmetti, Jack Repenning, Mike Balenger, Todd Kaufmann, Kjetil Svarstad, Tom Wurgler, Linus Tolke, Niko Makila, Carl Edman, Bill Trost, Dave Brennan, Dan Jacobson, Andy Scott, Steve Anderson, Sanjay Mathur, the folks on the ange-ftp-lovers mailing list and many others whose names I've forgotten who have helped to debug and fix problems with ange-ftp.el.