;;; hpath.el --- GNU Hyperbole support routines for handling POSIX and MSWindows paths ;; ;; Author: Bob Weiner ;; ;; Orig-Date: 1-Nov-91 at 00:44:23 ;; ;; Copyright (C) 1991-2019 Free Software Foundation, Inc. ;; See the "HY-COPY" file for license information. ;; ;; This file is part of GNU Hyperbole. ;;; Commentary: ;;; Code: ;;; ************************************************************************ ;;; Other required Elisp libraries ;;; ************************************************************************ (require 'hversion) ;; for (hyperb:window-system) definition (require 'hui-select) ;; for `hui-select-markup-modes' (unless (fboundp 'br-in-browser) ;; Then the OO-Browser is not loaded, so we can never be within the ;; browser. Define this as a dummy function that always returns nil ;; until the OO-Browser is ever loaded. (defun br-in-browser () "Always returns nil since the OO-Browser is not loaded." nil)) ;;; ************************************************************************ ;;; Public Variables ;;; ************************************************************************ (defcustom hpath:find-file-urls-mode nil "This is `t' when a remote file access library is available and use of ftp and http urls in file finding commands has been enabled. Default is `nil' since this can slow down normal file finding." :type 'boolean :initialize #'custom-initialize-default :set (lambda (_symbol _value) (call-interactively #'hpath:find-file-urls-mode)) :group 'hyperbole-buttons) ;;; ************************************************************************ ;;; MS WINDOWS PATH CONVERSIONS ;;; ************************************************************************ ;; This section adds automatic recognition of MSWindows implicit path ;; links and converts disk drive and path separators to whatever ;; format is needed by the underlying OS upon which Emacs is one, ;; notably either for POSIX or MSWindows (with no POSIC layer). ;; Especially useful when running Emacs under Windows Subsystem for ;; Linux (WSL) where the system-type variable is gnu/linux but ;; MSWindows is underneath so the user likely has many Windows ;; formatted links. ;; See "https://docs.microsoft.com/en-us/dotnet/standard/io/file-path-formats" ;; and "https://docs.microsoft.com/en-us/windows/wsl/interop" for ;; Windows path specifications and use under WSL. (defvar hpath:posix-mount-points-regexp "^\\(Filesystem\\|rootfs\\|none\\) " "Regexp of 'mount' command output lines that are not mount points of MSWindows paths.") (defvar hpath:mswindows-mount-prefix (cond ((eq system-type 'cygwin) "/cygdrive/") (hyperb:microsoft-os-p "") (t ;; POSIX "/mnt/")) "Path prefix to add when converting MSWindows drive paths to POSIX-style. Must include a trailing directory separator or be nil.") (defconst hpath:mswindows-drive-regexp (format "\\`\\(%s\\)?[\\/]?\\([a-zA-Z]\\)[:\\/]" hpath:mswindows-mount-prefix) "Regular expression matching an MSWindows drive letter at the beginning of a path string. Grouping 2 is the actual letter of the drive. If the value of 'hpath:mswindows-mount-prefix' changes, then re-initialize this constant.") (defconst hpath:mswindows-path-regexp "\\`.*\\.*[a-zA-Z0-9_.]" "Regular expression matching the start of an MSWindows path that does not start with a drive letter but contains directory separators.") (defvar hpath:mswindows-path-posix-mount-alist nil "Automatically set alist of (window-path-prefix . posix-mount-point) elements.") (defvar hpath:directory-expand-alist nil "Automatically set alist of (posix-mount-point . window-path-prefix) elements. Used to expand windows prefixes to posix mount points during mswindows-to-posix.") ;;;###autoload (defun hpath:mswindows-to-posix (path) "Convert a recognizable MSWindows PATH to a Posix-style path or return the path unchanged. If path begins with an MSWindows drive letter, prefix the converted path with the value of 'hpath:mswindows-mount-prefix'." (interactive "sMSWindows path to convert to POSIX: ") (when (stringp path) (setq path (hpath:mswindows-to-posix-separators path)) (when (string-match hpath:mswindows-drive-regexp path) (when (string-match hpath:mswindows-drive-regexp path) (let* ((drive-prefix (downcase (match-string 2 path))) (rest-of-path (substring path (match-end 0))) (absolute-p (and (not (string-empty-p rest-of-path)) (= (aref rest-of-path 0) ?/)))) ;; Convert MSWindows disk drive paths to POSIX-style with a mount prefix. (setq path (concat hpath:mswindows-mount-prefix drive-prefix (cond (hyperb:microsoft-os-p ":") (absolute-p "") (t "/")) rest-of-path)))))) path) (defun hpath:mswindows-to-posix-separators (path) "Replace all backslashes with forward slashes in PATH and expand the path against `directory-abbrev-alist', if possible. Path must be a string or an error will be triggered." (let ((directory-abbrev-alist hpath:directory-expand-alist)) (replace-regexp-in-string "\\\\" "/" (abbreviate-file-name path) nil t))) ;;;###autoload (defun hpath:posix-to-mswindows (path) "Convert and return a Posix-style PATH to an MSWindows path or return the path unchanged. If path begins with an optional mount prefix, 'hpath:mswindows-mount-prefix', followed by an MSWindows drive letter, remove the mount prefix." (interactive "sPOSIX path to convert to MSWindows: ") (when (stringp path) (setq path (hpath:posix-to-mswindows-separators path)) ;; Remove any POSIX mount prefix preceding an MSWindows path. (if (eq 0 (string-match hpath:mswindows-mount-prefix path)) (setq path (substring path (match-end 0)))) (when (string-match hpath:mswindows-drive-regexp path) (when (string-match hpath:mswindows-drive-regexp path) (let* ((drive-prefix (downcase (match-string 2 path))) (rest-of-path (substring path (match-end 0))) (absolute-p (= (aref path (1- (match-end 0))) ?\\))) ;; Convert formerly Posix-style Windows disk drive paths to MSWindows-style. (setq path (concat drive-prefix ":" (if (or (not absolute-p) (string-match "\\`[~/]" rest-of-path)) "" "\\") rest-of-path)))))) path) (defun hpath:posix-to-mswindows-separators (path) "Replace all forward slashes with backslashes in PATH and abbreviate the path if possible. Path must be a string or an error will be triggered. See 'abbreviate-file-name' for how path abbreviation is handled." (replace-regexp-in-string "/" "\\\\" (abbreviate-file-name path))) (defun hpath:posix-path-p (path) "Return non-nil if PATH looks like a Posix path." (and (stringp path) (string-match "/" path))) ;;;###autoload (defun hpath:substitute-posix-or-mswindows-at-point () "If point is within a recognizable Posix or MSWindows path, change the path to the other type of path." (interactive "*") (barf-if-buffer-read-only) (let* ((opoint (point)) (str-and-positions (hpath:delimited-possible-path t t)) (path (car str-and-positions)) (start (nth 1 str-and-positions)) (end (nth 2 str-and-positions))) (when path (if (hpath:posix-path-p path) (setq path (hpath:posix-to-mswindows path)) (setq path (hpath:mswindows-to-posix path))) (delete-region start end) (insert path) (goto-char (min opoint (point-max)))))) ;;;###autoload (defun hpath:substitute-posix-or-mswindows (path) "Change a recognizable Posix or MSWindows PATH to the other type of path." (when (stringp path) (if (hpath:posix-path-p path) (hpath:posix-to-mswindows path) (hpath:mswindows-to-posix path)))) ;;;###autoload (defun hpath:cache-mswindows-mount-points () "Cache valid MSWindows mount points in 'directory-abbrev-alist' when under a non-MSWindows operating system, e.g. WSL. Call this function manually if mount points change after Hyperbole is loaded." (interactive) (when (not hyperb:microsoft-os-p) (let ((mount-points-to-add ;; Sort alist of (path-mounted . mount-point) elements from shortest ;; to longest path so that the longest path is selected first within ;; 'directory-abbrev-alist' (elements are added in reverse order). (sort ;; Convert plist to alist for sorting. (hypb:map-plist (lambda (path mount-point) (if (string-match "\\`\\([a-zA-Z]\\):\\'" path) ;; Drive letter must be downcased ;; in order to work when converted back to Posix. (setq path (concat "/" (downcase (match-string 1 path))))) ;; Assume all mounted Windows paths are ;; lowercase for now. (cons (downcase path) mount-point)) ;; Return a plist of MSWindows path-mounted mount-point pairs. (split-string (shell-command-to-string (format "df -a -t drvfs 2> /dev/null | sort | uniq | grep -v '%s' | sed -e 's+ .*[-%%] /+ /+g'" hpath:posix-mount-points-regexp)))) (lambda (cons1 cons2) (<= (length (car cons1)) (length (car cons2)))))) path mount-point) (mapcar (lambda (path-and-mount-point) (setq path (car path-and-mount-point) mount-point (cdr path-and-mount-point)) (add-to-list 'directory-abbrev-alist (cons (format "\\`%s" (regexp-quote path)) mount-point))) mount-points-to-add) (setq hpath:directory-expand-alist ;; Save the reverse of each mount-points-to-add so ;; can expand paths when going from posix-to-mswindows. (mapcar (lambda (elt) (cons (cdr elt) (car elt))) mount-points-to-add)) mount-points-to-add))) ;;; ************************************************************************ ;;; FILE VIEWER COMMAND SETTINGS ;;; ************************************************************************ (defcustom hpath:external-open-office-suffixes "doc[mx]?\\|od[st]\\|ppsx?\\|ppt[mx]?\\|xls[mx]?" "*Regexp of Open Office document suffix alternatives to display externally. See http://www.openwith.org/programs/openoffice for a full list of possible suffixes." :type 'string :group 'hyperbole-commands) (defcustom hpath:external-display-alist-macos (list (cons (format "\\.\\(app\\|%s\\|adaptor\\|app\\|bshlf\\|clr\\|concur\\|create\\|diagram\\|dp\\|e?ps\\|frame\\|gif\\|locus\\|Mesa\\|nib\\|pdf\\|project\\|rtf\\|sense\\|tiff\\|tree\\)$" hpath:external-open-office-suffixes) "open")) "*An alist of (FILENAME-REGEXP . DISPLAY-PROGRAM-STRING-OR-LIST) elements for the macOS window system. See the function `hpath:get-external-display-alist' for detailed format documentation." :type 'regexp :group 'hyperbole-commands) (defvar hpath:external-display-alist-mswindows (list '("\\.vba$" . "/c/Windows/System32/cmd.exe //c start \"${@//&/^&}\"") (cons (format "\\.\\(%s\\)$" hpath:external-open-office-suffixes) "openoffice.exe")) "*An alist of (FILENAME-REGEXP . DISPLAY-PROGRAM-STRING-OR-LIST) elements for MS Windows. See the function `hpath:get-external-display-alist' for detailed format documentation.") (defvar hpath:external-display-alist-x (list '("\\.e?ps$" . "ghostview") '("\\.dvi$" . "xdvi") (cons (format "\\.\\(%s\\)$" hpath:external-open-office-suffixes) "openoffice") '("\\.pdf$" . ("xpdf" "acroread")) '("\\.ps\\.g?[zZ]$" . "zcat %s | ghostview -") '("\\.\\(gif\\|tiff?\\|xpm\\|xbm\\|xwd\\|pm\\|pbm\\|jpe?g\\)" . "xv") '("\\.ra?s$" . "snapshot -l")) "*An alist of (FILENAME-REGEXP . DISPLAY-PROGRAM-STRING-OR-LIST) elements for the X Window System. See the function `hpath:get-external-display-alist' for detailed format documentation.") (defvar hpath:info-suffix "\\.info\\(-[0-9]+\\)?\\(\\.gz\\|\\.Z\\|-z\\)?\\'" "Regexp matching to the end of Info manual file names.") (defvar hpath:internal-display-alist (let ((info-suffix "\\.info\\(-[0-9]+\\)?\\(\\.gz\\|\\.Z\\|-z\\)?\\'")) (delq nil (list ;; Support internal sound when available. (if (fboundp 'play-sound-file) '("\\.\\(au\\|mp3\\|ogg\\|wav\\)$" . play-sound-file)) ;; Run the OO-Browser on OOBR or OOBR-FTR Environment files. '("\\(\\`\\|/\\)\\(OOBR\\|oobr\\).*\\(-FTR\\|-ftr\\)?\\'" . br-env-browse) ;; Display the top node from Info online manuals. (cons (concat hpath:info-suffix "\\|/\\(info\\|INFO\\)/[^.]+$\\|/\\(info-local\\|INFO-LOCAL\\)/[^.]+$") (lambda (file) (if (and (string-match hpath:info-suffix file) (match-beginning 1)) ;; Removed numbered trailer to get basic filename. (setq file (concat (substring-no-properties file 0 (match-beginning 1)) (substring-no-properties file (match-end 1))))) (require 'info) ;; Ensure that *info* buffer is displayed in the right place. (hpath:display-buffer (current-buffer)) (condition-case () (Info-find-node file "Top") (error (if (and file (file-exists-p file)) (progn (if (get-buffer "*info*") (kill-buffer "*info*")) (Info-find-node file "*" nil t)) (error "Invalid file")))))) '("\\.rdb\\'" . rdb:initialize) ))) "*Alist of (FILENAME-REGEXP . EDIT-FUNCTION) elements for calling special functions to display particular file types within Emacs. See also the function (hpath:get-external-display-alist) for external display program settings.") (defvar hpath:display-buffer-alist (list (list 'this-window #'switch-to-buffer) (list 'other-window (lambda (b) (if (br-in-browser) (progn (br-to-view-window) (switch-to-buffer b)) (switch-to-buffer-other-window b)))) (list 'one-window (lambda (b) (if (br-in-browser) (br-quit)) (delete-other-windows) (switch-to-buffer b))) (list 'new-frame (lambda (b) (select-frame (make-frame)) (switch-to-buffer b))) (list 'other-frame #'hpath:display-buffer-other-frame) (list 'other-frame-one-window (lambda (b) (hpath:display-buffer-other-frame b) (delete-other-windows)))) "*Alist of (DISPLAY-WHERE-SYMBOL DISPLAY-BUFFER-FUNCTION) elements. This permits fine-grained control of where Hyperbole displays linked to buffers. The default value of DISPLAY-WHERE-SYMBOL is given by `hpath:display-where'. Valid DISPLAY-WHERE-SYMBOLs are: this-window - display in the current window other-window - display in another window in the current frame one-window - display in the current window, deleting other windows new-frame - display in a new frame other-frame - display in another, possibly existing, frame other-frame-one-window - display in another frame, deleting other windows.") (defvar hpath:display-where 'other-window "Symbol specifying the default method to use to display Hyperbole link referents. See documentation of `hpath:display-where-alist' for valid values.") (defvar hpath:display-where-alist (list (list 'this-window #'find-file) (list 'other-window (lambda (f) (if (br-in-browser) (progn (br-to-view-window) (find-file f)) (find-file-other-window f)))) (list 'one-window (lambda (f) (if (br-in-browser) (br-quit)) (delete-other-windows) (find-file f))) (list 'new-frame (lambda (f) (if (fboundp 'find-file-new-frame) (find-file-new-frame f) (hpath:find-other-frame f)))) (list 'other-frame #'hpath:find-other-frame) (list 'other-frame-one-window (lambda (f) (hpath:find-other-frame f) (delete-other-windows)))) "*Alist of (DISPLAY-WHERE-SYMBOL DISPLAY-FILE-FUNCTION) elements. This permits fine-grained control of where Hyperbole displays linked to files. The default value of DISPLAY-WHERE-SYMBOL is given by `hpath:display-where'. Valid DISPLAY-WHERE-SYMBOLs are: this-window - display in the current window other-window - display in another window in the current frame one-window - display in the current window, deleting other windows new-frame - display in a new frame other-frame - display in another, possibly existing, frame other-frame-one-window - display in another frame, deleting other windows.") (defcustom hpath:native-image-suffixes "\\.\\(xpm\\|png\\|gif\\|jpe?g\\)\\'" "Regular expression matching file name suffixes of natively handled image types. Used only if the function `image-mode' is defined." :type 'regexp :group 'hyperbole-commands) ;;; ************************************************************************ ;;; LINK PATH VARIABLE SUBSTITUTION SETTINGS ;;; ************************************************************************ ;; The following variable permits sharing of links over wide areas, where ;; links may contain variable references whose values may differ between ;; link creator and link activator. ;; ;; When a link is created, if its path contains a match for any of the ;; variable values in hpath:variables, then the variable's symbol ;; surrounded by ${ } delimiters is substituted for the literal value. ;; Hyperbole then replaces the variable with a matching value when the ;; link is later resolved. ;; (defcustom hpath:variables '(hyperb:dir Info-directory Info-directory-list sm-directory load-path exec-path) "*List of Emacs Lisp variable symbols to substitute within matching link paths. Each variable value, if bound, must be either a pathname or a list of pathnames. When embedded within a path, the format is ${variable}." :type '(repeat variable) :group 'hyperbole-commands) ;;; ************************************************************************ ;;; Other public variables ;;; ************************************************************************ (defvar hpath:rfc "/ftp:anonymous@ftp.ietf.org:rfc/rfc%s.txt" "*String to be used in the call: (hpath:rfc rfc-num) to create a path to the RFC document for `rfc-num'.") (defcustom hpath:suffixes '(".gz" ".Z") "*List of filename suffixes to add or remove within `hpath:exists-p' and `hpath:substitute-dir' calls." :type '(repeat string) :group 'hyperbole-commands) (defvar hpath:tmp-prefix "/tmp/remote-" "*Pathname prefix to attach to remote files copied locally for use with external viewers.") ;; WWW URL format: [URL[:=]]:/[@][:][/] ;; or [URL[:=]]://[@][:][] ;; or URL[:=][@][:][] (no protocol specified) ;; Avoid [a-z]:/path patterns since these may be disk paths on OS/2, DOS or ;; Windows. (defvar hpath:url-regexp "]+[^\]\[@ \t\n\r\"`'|(){}<>.,?#!*]\\)*\\)?\\)>?" "Regular expression which matches a Url in a string or buffer. Its match groupings and their names are: 1 = hpath:url-keyword-grpn = optional `URL:' or `URL=' literal 2 = hpath:url-grpn = the whole URL 3 = hpath:protocol-grpn = access protocol 4 = hpath:username-grpn = optional username 5 = hpath:sitename-grpn = URL site to connect to 6 = unused = for compatibility with hpath:url-regexp2 7 = hpath:portnumber-grpn = optional port number to use 8 = hpath:pathname-grpn = optional pathname to access.") (defvar hpath:url-hostnames-regexp "\\(www\\|ftp\\|telnet\\|news\\|nntp\\)" "Grouped regexp alternatives of hostnames that automatically determine the Url access protocol to use.") (defvar hpath:url-regexp2 (concat "]+[^\]\[@ \t\n\r\"`'|(){}<>.,?#!*]\\)*\\)?\\)>?") "Regular expression which matches a Url in a string or buffer. Its match groupings and their names are: 1 = hpath:url-keyword-grpn = optional `URL:' or `URL=' literal 2 = hpath:url-grpn = the whole URL 3 = unused = for compatibility with hpath:url-regexp 4 = unused = for compatibility with hpath:url-regexp 5 = hpath:sitename-grpn = URL site to connect to 6 = hpath:hostname-grpn = hostname used to determine the access protocol, e.g. ftp.domain.com 7 = hpath:portnumber-grpn = optional port number to use 8 = hpath:pathname-grpn = optional pathname to access.") (defvar hpath:url-regexp3 (concat "]+[^\]\[@ \t\n\r\"`'|(){}<>.,?#!*]\\)*\\)?\\)>?") "Regular expression which matches a Url in a string or buffer. Its match groupings and their names are: 1 = hpath:url-keyword-grpn = required `URL:' or `URL=' literal 2 = hpath:url-grpn = the whole URL 3 = unused = for compatibility with hpath:url-regexp 4 = unused = for compatibility with hpath:url-regexp 5 = hpath:sitename-grpn = URL site to connect to 6 = hpath:hostname-grpn = hostname used to determine the access protocol, e.g. ftp.domain.com 7 = hpath:portnumber-grpn = optional port number to use 8 = hpath:pathname-grpn = optional pathname to access.") (defconst hpath:url-keyword-grpn 1 "Optional `URL:' or `URL=' literal. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:url-grpn 2 "The whole URL. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:protocol-grpn 3 "Access protocol. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:username-grpn 4 "Optional username. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:sitename-grpn 5 "URL site to connect to. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:hostname-grpn 6 "Hostname used to determine the access protocol, e.g. ftp.domain.com. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:portnumber-grpn 7 "Optional port number to use. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defconst hpath:pathname-grpn 8 "Optional pathname to access. See doc for `hpath:url-regexp' and `hpath:url-regexp[2,3]'.") (defvar hpath:string-url-regexp (concat "\\`" hpath:url-regexp "\\'") "Regular expression that matches to a string that contains a possibly delimited Url and nothing else. See the documentation for `hpath:url-regexp' for match groupings to use with string-match.") (defvar hpath:string-url-regexp2 (concat "\\`" hpath:url-regexp2 "\\'") "Regular expression that matches to a string that contains a possibly delimited terse Url and nothing else. See the documentation for `hpath:url-regexp' for match groupings to use with string-match.") (defvar hpath:string-url-regexp3 (concat "\\`" hpath:url-regexp3 "\\'") "Regular expression that matches to a string that contains a possibly delimited terse Url and nothing else. See the documentation for `hpath:url-regexp' for match groupings to use with string-match.") ;;; ************************************************************************ ;;; Private variables ;;; ************************************************************************ (defconst hpath:html-anchor-id-pattern "\\(id\\|name\\)=['\"]%s['\"]?" "Regexp matching an html anchor id definition and containing a %s for replacement of a specific anchor id.") (defconst hpath:markdown-anchor-id-pattern "^[ ]*%s: " "Regexp matching a Markdown anchor id definition and containing a %s for replacement of a specific anchor id.") (defconst hpath:markdown-section-pattern "^#+[ \t]+%s\\([ \t[:punct:]]*\\)$" "Regexp matching a Markdown section header and containing a %s for replacement of a specific section name.") (defconst hpath:markdown-suffix-regexp "\\.[mM][dD]" "Regexp that matches to a Markdown file suffix.") (defconst hpath:markup-link-anchor-regexp (concat "\\`\\(#?[^#]+\\)\\(#\\)\\([^\]\[#^{}<>\"`'\\\n\t\f\r]*\\)") "Regexp that matches a markup filename followed by a hash (#) and an optional in-file anchor name.") (defconst hpath:outline-section-pattern "^\*+[ \t]+%s\\([ \t[:punct:]]*\\)$" "Regexp matching an Emacs outline section header and containing a %s for replacement of a specific section name.") (defvar hpath:prefix-regexp "\\`[-!&][ ]*" "Regexp matching command characters which may precede a pathname. These are used to indicate how to display or execute the pathname. - means evaluate it as Emacs Lisp; ! means execute it as a shell script & means run it under the current window system.") (defvar hpath:remote-regexp "\\`/[^/:]+:\\|\\`ftp[:.]\\|\\`www\\.\\|\\`https?:" "Regexp matching remote pathnames and urls which invoke remote file handlers.") (defconst hpath:texinfo-section-pattern "^@node+[ \t]+%s[ \t]*\\(,\\|$\\)" "Regexp matching a Texinfo section header and containing a %s for replacement of a specific section name.") ;;; ************************************************************************ ;;; Public functions ;;; ************************************************************************ (defun hpath:absolute-to (path &optional default-dirs) "Returns PATH as an absolute path relative to one directory from optional DEFAULT-DIRS or `default-directory'. Returns PATH unchanged when it is not a valid path or when DEFAULT-DIRS is invalid. DEFAULT-DIRS when non-nil may be a single directory or a list of directories. The first one in which PATH is found is used." (if (not (and (stringp path) (hpath:is-p path nil t))) path (if (not (cond ((null default-dirs) (setq default-dirs (cons default-directory nil))) ((stringp default-dirs) (setq default-dirs (cons default-dirs nil))) ((listp default-dirs)) (t nil))) path (let ((rtn) dir) (while (and default-dirs (null rtn)) (setq dir (expand-file-name (file-name-as-directory (car default-dirs))) rtn (expand-file-name path dir) default-dirs (cdr default-dirs)) (or (file-exists-p rtn) (setq rtn nil))) (or rtn path))))) (defun hpath:tramp-file-name-regexp () "Returns a modified tramp-file-name-regexp for matching to the beginning of a remote file name. Removes bol anchor and removes match to empty string if present." (let ((tramp-regexp (car (if (fboundp 'tramp-file-name-structure) (tramp-file-name-structure) tramp-file-name-structure)))) (substring-no-properties (replace-regexp-in-string "\\\\'" "" tramp-regexp) 1))) (defun hpath:remote-at-p () "Returns a remote pathname that point is within or nil. See the `(emacs)Remote Files' info documentation for pathname format details. Always returns nil if (hpath:remote-available-p) returns nil." (let ((remote-package (hpath:remote-available-p)) (user (hpath:remote-default-user)) (path)) (when remote-package (setq path (save-excursion (skip-chars-backward "^[ \t\n\r\f\"`'|\(\{<") (cond ((and (eq remote-package 'tramp) (looking-at (hpath:tramp-file-name-regexp))) (match-string-no-properties 0)) ((looking-at hpath:url-regexp) (if (string-equal (match-string-no-properties hpath:protocol-grpn) "ftp") (concat "/ftp:" ;; user (if (match-beginning hpath:username-grpn) (match-string-no-properties hpath:username-grpn) (concat user "@")) ;; sitename (hpath:delete-trailer (match-string-no-properties hpath:sitename-grpn)) ":" ;; path (if (match-beginning hpath:pathname-grpn) (match-string-no-properties hpath:pathname-grpn))) ;; else ignore this other type of WWW path )) ((or (looking-at hpath:url-regexp2) (looking-at hpath:url-regexp3)) (if (string-equal (match-string-no-properties hpath:hostname-grpn) "ftp") (concat "/ftp:" user "@" ;; site (hpath:delete-trailer (match-string-no-properties hpath:sitename-grpn)) ":" ;; path (if (match-beginning hpath:pathname-grpn) (match-string-no-properties hpath:pathname-grpn))) ;; else ignore this other type of WWW path )) ;; user, site and path ((looking-at "/?[^/:@ \t\n\r\"`'|]+@[^/:@ \t\n\r\"`'|]+:[^]@ \t\n\r\"`'|\)\}]+") (match-string-no-properties 0)) ;; @site and path ((looking-at "@[^/:@ \t\n\r\"`'|]+:[^]@:, \t\n\r\"`'|\)\}]+") (concat "/" user (match-string-no-properties 0))) ;; site and path ((and (looking-at "/?\\(\\([^/:@ \t\n\r\"`'|]+\\):[^]@:, \t\n\r\"`'|\)\}]+\\)[] \t\n\r,.\"`'|\)\}]") (setq path (match-string-no-properties 1)) (string-match "[^.]\\.[^.]" (match-string-no-properties 2))) (concat "/" user "@" path)) ;; host and path ((and (looking-at "/\\([^/:@ \t\n\r\"`'|]+:[^]@:, \t\n\r\"`'|\)\}]+\\)") (setq path (match-string-no-properties 1))) (concat "/" user "@" path)) ))) (hpath:delete-trailer path)))) (defun hpath:remote-p (path) "Returns a normalized form of PATH if it is a remote, non-local, pathname, else nil. See the `(Tramp)Top' Emacs lisp package manual for remote pathname format details. Always returns nil if (hpath:remote-available-p) returns nil." (and (stringp path) (let ((remote-package (hpath:remote-available-p)) (user (hpath:remote-default-user)) result) (setq result (cond ((null remote-package) nil) ((eq remote-package 'tramp) (if (tramp-tramp-file-p path) path)) ((string-match hpath:string-url-regexp path) (if (string-equal "ftp" (match-string-no-properties hpath:protocol-grpn path)) (concat "/" ;; user (if (match-beginning hpath:username-grpn) (match-string-no-properties hpath:username-grpn path) (concat user "@")) ;; site (hpath:delete-trailer (match-string-no-properties hpath:sitename-grpn path)) ":" ;; path (if (match-beginning hpath:pathname-grpn) (match-string-no-properties hpath:pathname-grpn path))) ;; else ignore this other type of WWW path )) ((or (string-match hpath:string-url-regexp2 path) (string-match hpath:string-url-regexp3 path)) (if (string-equal "ftp" (match-string-no-properties hpath:hostname-grpn path)) (concat "/" user "@" ;; site (hpath:delete-trailer (match-string-no-properties hpath:sitename-grpn path)) ":" ;; path (if (match-beginning hpath:pathname-grpn) (match-string-no-properties hpath:pathname-grpn path))) ;; else ignore this other type of WWW path )) ;; user, site and path ((string-match "/?[^/:@ \t\n\r\"`'|]+@[^/:@ \t\n\r\"`'|]+:[^]@ \t\n\r\"`'|\)\}]*" path) (match-string-no-properties 0 path)) ;; @site and path ((string-match "@[^/:@ \t\n\r\"`'|]+:[^]@ \t\n\r\"`'|\)\}]*" path) (concat "/" user (match-string-no-properties 0 path))) ;; site and path ((and (string-match "/?\\(\\([^/:@ \t\n\r\"`'|]+\\):[^]@:, \t\n\r\"`'|\)\}]*\\)" path) (setq result (match-string-no-properties 1 path)) (string-match "[^.]\\.[^.]" (match-string-no-properties 2 path))) (concat "/" user "@" result)) ;; host and path ((and (string-match "/\\([^/:@ \t\n\r\"`'|]+:[^]@:, \t\n\r\"`'|\)\}]*\\)" path) (setq result (match-string-no-properties 1 path))) (concat "/" user "@" result)) )) (hpath:delete-trailer result)))) (defun hpath:at-p (&optional type non-exist) "Returns delimited path or non-delimited remote path at point, if any. World-Wide Web urls are ignored and therefore dealt with by other code. Delimiters may be: double quotes, open and close single quote, whitespace, or Texinfo file references. If optional TYPE is the symbol 'file or 'directory, then only that path type is accepted as a match. Only locally reachable paths are checked for existence. With optional NON-EXIST, nonexistent local paths are allowed. Absolute pathnames must begin with a `/' or `~'." (cond ;; Local file URLs ;; ((hpath:is-p (hargs:delimited "file://" "[ \t\n\r\"\'\}]" nil t))) ((hpath:remote-at-p)) ((hpath:www-at-p) nil) ((hpath:is-p (hpath:delimited-possible-path non-exist) type non-exist)))) (defun hpath:delimited-possible-path (&optional non-exist include-positions) "Returns delimited possible path or non-delimited remote path at point, if any. No validity checking is done on the possible path. Delimiters may be: double quotes, open and close single quote, whitespace, or Texinfo file references. With optional NON-EXIST, nonexistent local paths are allowed. Absolute pathnames must begin with a `/' or `~'. With optional INCLUDE-POSITIONS, returns a triplet list of (path start-pos end-pos) or nil." ;; Prevents MSWindows to Posix path substitution (let ((hyperb:microsoft-os-p t)) (or (hargs:delimited "\"" "\"" nil nil include-positions "[`'’]") ;; Filenames in Info docs or Python files (hargs:delimited "[`'‘]" "[`'’]" t t include-positions "\"") ;; Filenames in TexInfo docs (hargs:delimited "@file{" "}" nil nil include-positions) ;; Any existing whitespace delimited filename at point. ;; If match consists of only punctuation, like ;; . or .., don't treat it as a pathname. Only look for ;; whitespace delimited filenames if non-exist is nil. (unless non-exist (let* ((triplet (hargs:delimited "^\\|\\(\\s-\\|[\]\[(){}<>\;&,@]\\)*" "\\([\]\[(){}<>\;&,@]\\|:*\\s-\\)+\\|$" t t t)) (p (car triplet)) (punc (char-syntax ?.))) (if (delq nil (mapcar (lambda (c) (/= punc (char-syntax c))) p)) (if include-positions triplet p))))))) ;;;###autoload (defun hpath:display-buffer (buffer &optional display-where) "Displays and selects BUFFER at optional DISPLAY-WHERE location or at `hpath:display-where'. BUFFER must be a buffer or a buffer name. See the documentation of `hpath:display-buffer-alist' for valid values of DISPLAY-WHERE. Returns the window in which the buffer is displayed or nil if not displayed because BUFFER is invalid." (interactive "bDisplay buffer: ") (if (stringp buffer) (setq buffer (get-buffer buffer))) (when buffer ;; BW 4/30/2016 - Commented out in case interferes with Smart Key ;; selection and yanking of the region via drags. ;; (hpath:push-tag-mark) (unless display-where (setq display-where hpath:display-where)) (funcall (car (cdr (or (assq display-where hpath:display-buffer-alist) (assq 'other-window hpath:display-buffer-alist)))) buffer) (selected-window))) (defun hpath:display-buffer-other-frame (buffer) "Displays and selects BUFFER, in another frame. BUFFER must be a buffer or a buffer name. May create a new frame, or reuse an existing one. See the documentation of `hpath:display-buffer' for details. Returns the window in which the buffer is displayed." (interactive "bDisplay buffer in other frame: ") ;; BW 4/30/2016 - Commented out in case interferes with Smart Key ;; selection and yanking or the region via drags. ;; (hpath:push-tag-mark) (if (= (length (frame-list)) 1) (select-frame (make-frame)) (other-frame 1)) (if (br-in-browser) (br-to-view-window)) (switch-to-buffer buffer) (selected-window)) (defun hpath:find (filename &optional display-where) "Edits file FILENAME using user customizable settings of display program and location. FILENAME may start with a special prefix character which is handled as follows: !filename - execute as a non-windowed program within a shell; &filename - execute as a windowed program; -filename - load as an Emacs Lisp program. Otherwise, if FILENAME matches a regular expression in the alist returned by \(hpath:get-external-display-alist), the associated external display program is invoked. If not, `hpath:internal-display-alist' is consulted for a specialized internal display function to use. If no matches are found there, `hpath:display-where-alist' is consulted using the optional argument, DISPLAY-WHERE (a symbol) or if that is nil, the value of `hpath:display-where', and the matching display function is used. Allows for hash-style link references to HTML, Markdown or Emacs outline headings of the form, #. Returns non-nil iff file is displayed within a buffer (not with an external program)." (interactive "FFind file: ") (let ((case-fold-search t) (default-directory default-directory) modifier loc anchor hash path) (if (string-match hpath:prefix-regexp filename) (setq modifier (aref filename 0) filename (substring filename (match-end 0)))) (setq path (hpath:substitute-value (if (string-match hpath:markup-link-anchor-regexp filename) (progn (setq hash t anchor (match-string 3 filename)) (substring filename 0 (match-end 1))) filename)) loc (hattr:get 'hbut:current 'loc) default-directory (file-name-directory ;; Loc may be a buffer without a file (if (stringp loc) loc default-directory)) filename (hpath:absolute-to path default-directory)) (let ((remote-filename (hpath:remote-p path))) (or modifier remote-filename (file-exists-p filename) (error "(hpath:find): \"%s\" does not exist" filename)) (or modifier remote-filename (file-readable-p filename) (error "(hpath:find): \"%s\" is not readable" filename)) ;; If filename is a remote file (not a directory, we have to copy it to ;; a temporary local file and then display that. (when (and remote-filename (not (file-directory-p remote-filename))) (copy-file remote-filename (setq path (concat hpath:tmp-prefix (file-name-nondirectory remote-filename))) t t) (setq filename (cond (anchor (concat remote-filename "#" anchor)) (hash (concat remote-filename "#")) (t path))))) (cond (modifier (cond ((eq modifier ?!) (hact 'exec-shell-cmd filename)) ((eq modifier ?&) (hact 'exec-window-cmd filename)) ((eq modifier ?-) (load filename))) nil) (t (let ((display-executables (hpath:find-program path)) executable) (cond ((stringp display-executables) (hact 'exec-window-cmd (hpath:command-string display-executables filename)) nil) ((functionp display-executables) (funcall display-executables filename) t) ((and (listp display-executables) display-executables) (setq executable (hpath:find-executable display-executables)) (if executable (hact 'exec-window-cmd (hpath:command-string executable filename)) (error "(hpath:find): No available executable from: %s" display-executables))) (t (setq path (hpath:validate path)) (if (null display-where) (setq display-where hpath:display-where)) (funcall (car (cdr (or (assq display-where hpath:display-where-alist) (assq 'other-window hpath:display-where-alist)))) path) (if (or hash anchor) (hpath:to-markup-anchor hash anchor)) t))))))) (defun hpath:to-markup-anchor (hash anchor) "Move point to the definition of ANCHOR if found or if only HASH is non-nil, move to the beginning of the buffer." (cond ((and (stringp anchor) (not (equal anchor ""))) (cond ((memq major-mode hui-select-markup-modes) ;; In HTML-like mode where link ids are case-sensitive. (let ((opoint (point)) (case-fold-search)) (goto-char (point-min)) (if (re-search-forward (format hpath:html-anchor-id-pattern (regexp-quote anchor)) nil t) (progn (forward-line 0) (recenter 0)) (goto-char opoint) (error "(hpath:to-markup-anchor): Anchor `%s' not found in the visible buffer portion" anchor)))) (t (let ((opoint (point)) ;; Markdown or outline link ids are case ;; insensitive and - characters are converted to ;; spaces at the point of definition. (case-fold-search t) (anchor-name (subst-char-in-string ?- ?\ anchor))) (goto-char (point-min)) (if (re-search-forward (format (cond ((string-match hpath:markdown-suffix-regexp buffer-file-name) hpath:markdown-section-pattern) ((eq major-mode 'texinfo-mode) hpath:texinfo-section-pattern) (t hpath:outline-section-pattern)) (regexp-quote anchor-name)) nil t) (progn (forward-line 0) (recenter 0)) (goto-char opoint) (error "(hpath:to-markup-anchor): Section `%s' not found in the visible buffer portion" anchor-name)))))) (hash (goto-char (point-min))))) (defun hpath:find-executable (executable-list) "Return the first executable string from EXECUTABLE-LIST found within `exec-path' or nil." (catch 'found (mapc (lambda (executable) (if (stringp executable) ;; Match only to files with execute permission. (if (locate-file executable exec-path nil #'file-executable-p) (throw 'found executable)) (error "(hpath:find-executable): Non-string entry, %s" executable-list))) executable-list) nil)) (defun hpath:find-line (filename line-num &optional display-where) "Edits file FILENAME with point placed at LINE-NUM. `hpath:display-where-alist' is consulted using the optional argument, DISPLAY-WHERE (a symbol) or if that is nil, the value of `hpath:display-where', and the matching display function is used to determine where to display the file, e.g. in another frame. Always returns t." (interactive "FFind file: ") ;; Just delete any special Hyperbole command characters preceding the filename, ignoring them. (if (string-match hpath:prefix-regexp filename) (setq filename (substring filename (match-end 0)))) (setq filename (hpath:substitute-value filename) filename (hpath:validate filename)) (if (null display-where) (setq display-where hpath:display-where)) (funcall (car (cdr (or (assq display-where hpath:display-where-alist) (assq 'other-window hpath:display-where-alist)))) filename) (if (integerp line-num) (progn (widen) (goto-char (point-min)) (forward-line (1- line-num)))) t) (defun hpath:find-other-frame (filename) "Edits file FILENAME, in another frame. May create a new frame, or reuse an existing one. See documentation of `hpath:find' for details. Returns the buffer of displayed file." (interactive "FFind file in other frame: ") (if (= (length (frame-list)) 1) (if (fboundp 'id-create-frame) (id-create-frame) (select-frame (make-frame))) (other-frame 1)) (if (br-in-browser) (br-to-view-window)) (find-file filename)) (defun hpath:find-other-window (filename) "Edits file FILENAME, in another window or using an external program. May create a new window, or reuse an existing one; see the function display-buffer. See documentation of `hpath:find' for details. Returns non-nil iff file is displayed within a buffer." (interactive "FFind file in other window: ") (hpath:find filename 'other-window)) ;; (hyperb:window-system) function from "hversion.el" must be defined ;; prior to use of this function. (defun hpath:get-external-display-alist () "Returns an alist of (FILENAME-REGEXP . DISPLAY-PROGRAM-STRING-OR-LIST) elements for the current window system. These are used to display matching filenames with external window system programs, such as a pdf reader. The cdr of each element may be: a string which must represent either an executable name or a shell command with an embedded %s for substitution of the current context filename; a list of executable names \(the first valid one is used); or a function of one filename argument. See also `hpath:internal-display-alist' for internal, window-system independent display settings." (cond ((memq window-system '(dps ns)) hpath:external-display-alist-macos) (hyperb:microsoft-os-p hpath:external-display-alist-mswindows) (t (cdr (assoc (hyperb:window-system) (list (cons "emacs" hpath:external-display-alist-x) ; GNU Emacs under X (cons "next" hpath:external-display-alist-macos))))))) (defun hpath:is-p (path &optional type non-exist) "Returns PATH if PATH is a Posix or MSWindows path, else nil. If optional TYPE is the symbol 'file or 'directory, then only that path type is accepted as a match. The existence of the path is checked only for locally reachable paths (Info paths are not checked). Single spaces are permitted in the middle of existing pathnames, but not at the start or end. Tabs and newlines are converted to space before the pathname is checked, this normalized path form is what is returned for PATH. With optional NON-EXIST, nonexistent local paths are allowed." (when (stringp path) (setq path (hpath:mswindows-to-posix path)) (let ((rtn-path path) (suffix)) ;; Path may be a link reference with components other than a ;; pathname. These components always follow a comma or # symbol, so ;; strip them, if any, before checking path. (and (if (string-match "\\`[^#][^#,]*\\([ \t\n\r]*[#,]\\)" path) (setq rtn-path (concat (substring path 0 (match-beginning 1)) "%s" (substring path (match-beginning 1))) path (substring path 0 (match-beginning 1))) (setq rtn-path (concat rtn-path "%s"))) ;; If path is just a local reference that begins with #, ;; prepend the file name to it. (cond ((and buffer-file-name ;; ignore HTML color strings (not (string-match "\\`#[0-9A-Fa-f][0-9A-Fa-f][0-9A-Fa-f][0-9A-Fa-f][0-9A-Fa-f][0-9A-Fa-f]\\'" path)) ;; match to in-file HTML references (string-match "\\`#[^\'\"<>#]+\\'" path)) (setq rtn-path (concat "file://" buffer-file-name rtn-path) path buffer-file-name)) ((string-match "\\`[^#]+\\(#[^#]*\\)\\'" path) ;; file and # reference (setq path (substring path 0 (match-beginning 1))) (if (memq (aref path 0) '(?/ ?~)) ;; absolute (setq rtn-path (concat "file://" rtn-path)) (setq path (concat default-directory path) rtn-path (concat "file://" default-directory rtn-path)))) (t)) (if (string-match hpath:prefix-regexp path) (setq path (substring path (match-end 0))) t) (not (or (string-equal path "") (string-match "\\`\\s \\|\\s \\'" path))) ;; Convert tabs and newlines to space. (setq path (hbut:key-to-label (hbut:label-to-key path))) (or (not (string-match "[()]" path)) (string-match "\\`([^ \t\n\r\)]+)[ *A-Za-z0-9]" path)) (if (string-match "\\$\{[^\}]+}" path) (setq path (hpath:substitute-value path)) t) (not (string-match "[\t\n\r\"`'|{}\\]" path)) (or (not (hpath:www-p path)) (string-match "\\`ftp[:.]" path)) (let ((remote-path (string-match "\\(@.+:\\|^/.+:\\|..+:/\\).*[^:0-9/]" path))) (if (cond (remote-path (cond ((eq type 'file) (not (string-equal "/" (substring path -1)))) ((eq type 'directory) (string-equal "/" (substring path -1))) (t))) ((or (and non-exist (or ;; Info or remote path, so don't check for. (string-match "[()]" path) (hpath:remote-p path) (setq suffix (hpath:exists-p path t)) ;; Don't allow spaces in non-existent ;; pathnames. (not (string-match " " path)))) (setq suffix (hpath:exists-p path t))) (cond ((eq type 'file) (not (file-directory-p path))) ((eq type 'directory) (file-directory-p path)) (t)))) (progn ;; Might be an encoded URL with % characters, so ;; decode it before calling format below. (when (string-match "%" rtn-path) (let (decoded-path) (while (not (equal rtn-path (setq decoded-path (hypb:decode-url rtn-path)))) (setq rtn-path decoded-path)))) ;; Quote any % except for one %s at the end of the ;; path part of rtn-path (immediately preceding a # ;; or , character or the end of string). (setq rtn-path (hypb:replace-match-string "%" rtn-path "%%") rtn-path (hypb:replace-match-string "%%s\\([#,]\\|\\'\\)" rtn-path "%s\\1")) ;; Return path if non-nil return value. (if (stringp suffix) ;; suffix could = t, which we ignore (if (string-match (concat (regexp-quote suffix) "%s") rtn-path) ;; remove suffix (concat (substring rtn-path 0 (match-beginning 0)) (substring rtn-path (match-end 0))) ;; add suffix (format rtn-path suffix)) (format rtn-path ""))))))))) (defun hpath:push-tag-mark () "Add a tag return marker at point if within a programming language file buffer. Is a no-op if the function `push-tag-mark' is not available." (and buffer-file-name comment-start (not (memq last-command '(xref-find-definitions find-tag find-tag-other-window tags-loop-continue))) (or (and (fboundp 'xref-push-marker-stack) ;; push old position (xref-push-marker-stack)) (and (fboundp 'push-tag-mark) ;; push old position (push-tag-mark))))) (defun hpath:relative-to (path &optional default-dir) "Returns PATH relative to optional DEFAULT-DIR or `default-directory'. Returns PATH unchanged when it is not a valid path." (if (not (and (stringp path) (hpath:is-p path))) path (setq default-dir (expand-file-name (file-name-as-directory (or default-dir default-directory))) path (expand-file-name path)) (and path default-dir (if (string-equal "/" default-dir) path (let ((end-dir (min (length path) (length default-dir)))) (cond ((string-equal (substring path 0 end-dir) default-dir) (concat "./" (substring path end-dir))) ((progn (setq default-dir (file-name-directory (directory-file-name default-dir)) end-dir (min (length path) (length default-dir))) (string-equal (substring path 0 end-dir) default-dir)) (concat "../" (substring path end-dir))) ((progn (setq default-dir (file-name-directory (directory-file-name default-dir)) end-dir (min (length path) (length default-dir))) (string-equal (substring path 0 end-dir) default-dir)) (concat "../../" (substring path end-dir))) (t path))))))) (defun hpath:rfc (rfc-num) "Return pathname to textual rfc document indexed by RFC-NUM. See the documentation of the `hpath:rfc' variable." (format hpath:rfc rfc-num)) (defun hpath:substitute-value (path) "Substitutes matching value for Emacs Lisp variables and environment variables in PATH and returns PATH." ;; Uses free variables `match' and `start' from `hypb:replace-match-string'. (substitute-in-file-name (hypb:replace-match-string "\\$\{[^\}]+}" path (lambda (str) (let* ((var-group (substring path match start)) (var-name (substring path (+ match 2) (1- start))) (rest-of-path (substring path start)) (sym (intern-soft var-name))) (when (file-name-absolute-p rest-of-path) (setq rest-of-path (substring rest-of-path 1))) (if (or (and sym (boundp sym)) (getenv var-name)) (directory-file-name (hpath:substitute-dir var-name rest-of-path)) var-group))) t))) (defun hpath:substitute-var (path) "Replaces up to one match in PATH with the first variable from `hpath:variables' whose value contains a string match to PATH. After any match, the resulting path will contain a varible reference like ${variable}." (if (not (and (stringp path) (string-match "/" path) (hpath:is-p path))) path (setq path (hpath:symlink-referent path)) (let ((new-path) (vars hpath:variables) result var val) (while (and vars (null new-path)) (setq var (car vars) vars (cdr vars)) (if (boundp var) (progn (setq val (symbol-value var)) (cond ((stringp val) (if (setq result (hpath:substitute-var-name var val path)) (setq new-path result))) ((null val)) ((listp val) (while (and val (null new-path)) (if (setq result (hpath:substitute-var-name var (car val) path)) (setq new-path result)) (setq val (cdr val)))) (t (error "(hpath:substitute-var): `%s' has invalid value for hpath:variables" var)))))) (or new-path path) ))) ;; ;; The following function recursively resolves all POSIX links to their ;; final referents. ;; Works with variable-based and other strange links like: ;; /usr/local -> $(SERVER_LOCAL)/usr/local, /usr/bin -> ;; ../$(SYSTYPE)/usr/bin and /tmp -> `node_data/tmp. It also handles ;; relative links properly as in /usr/local/emacs -> gnu/emacs which must ;; be resolved relative to the `/usr/local' directory. ;; (defun hpath:symlink-referent (linkname) "Returns expanded file or directory referent of LINKNAME. LINKNAME should not end with a directory delimiter. Returns nil if LINKNAME is not a string. Returns LINKNAME unchanged if it is not a symbolic link but is a pathname." (if (stringp linkname) (or (file-symlink-p linkname) linkname))) (defun hpath:symlink-expand (referent dirname) "Returns expanded file or directory REFERENT relative to DIRNAME." (let ((var-link) (dir dirname)) (while (string-match "\\$(\\([^\)]*\\))" referent) (setq var-link (getenv (substring referent (match-beginning 1) (match-end 1))) referent (concat (substring referent 0 (match-beginning 0)) var-link (substring referent (match-end 0))))) ;; If referent is not an absolute path (let ((nd-abbrev (string-match "`node_data" referent))) (if (and nd-abbrev (= nd-abbrev 0)) (setq referent (concat ;; Prepend node name given in dirname, if any (and (string-match "^//[^/]+" dirname) (substring dirname 0 (match-end 0))) "/sys/" (substring referent 1))))) (while (string-match "\\(^\\|/\\)\\.\\.\\(/\\|$\\)" referent) ;; Match to "//.." or "/.." at the start of link referent (while (string-match "^\\(//\\.\\.\\|/\\.\\.\\)\\(/\\|$\\)" referent) (setq referent (substring referent (match-end 1)))) ;; Match to "../" or ".." at the start of link referent (while (string-match "^\\.\\.\\(/\\|$\\)" referent) (setq dir (file-name-directory (directory-file-name dir)) referent (concat dir (substring referent (match-end 0))))) ;; Match to rest of "../" in link referent (while (string-match "[^/]+/\\.\\./" referent) (setq referent (concat (substring referent 0 (match-beginning 0)) (substring referent (match-end 0)))))) (and (not (eq (aref referent 0) ?~)) (not (eq (aref referent 0) ?/)) (setq referent (expand-file-name referent dirname)))) referent) (defun hpath:validate (path) "Returns PATH if PATH is a valid, readable path, else signals error. Info and remote pathnames are considered readable without any validation checks. Default-directory should be equal to the current Hyperbole button source directory when called, so that PATH is expanded relative to it." (unless (stringp path) (error "(hpath:validate): \"%s\" is not a pathname." path)) (setq path (hpath:mswindows-to-posix path)) (cond ((or (string-match "[()]" path) (hpath:remote-p path)) ;; info or remote path, so don't validate path) ((if (not (hpath:www-p path)) ;; Otherwise, must not be a WWW link ref and must be a readable ;; path. (let ((return-path (hpath:exists-p path))) (and return-path (file-readable-p return-path) return-path)))) (t (error "(hpath:validate): \"%s\" is not readable." path)))) ;;; URL Handling (defun hpath:find-file-urls-p () (and (boundp 'file-name-handler-alist) (hpath:remote-available-p) t)) ;; Partial setup which makes file finding commands recognize full and ;; abbreviated ftp and www URLs when a remote file access library is ;; available. (defun hpath:handle-urls () (let ((remote-fs-package (hpath:remote-available-p))) ;; www-url functions are defined in "hsys-www.el". (put 'expand-file-name remote-fs-package 'www-url-expand-file-name) (put 'find-file-noselect remote-fs-package 'www-url-find-file-noselect) ;; Necessary since Dired overrides other file-name-handler-alist ;; settings. (put 'expand-file-name 'dired 'www-url-expand-file-name) (put 'find-file-noselect 'dired 'www-url-find-file-noselect) (unless (fboundp 'hyperb:substitute-in-file-name) (progn ;; Overload `substitute-in-file-name' to eliminate truncation of URL prefixes ;; such as http://. (unless (fboundp 'hyperb:substitute-in-file-name) (defalias 'hyperb:substitute-in-file-name (symbol-function 'substitute-in-file-name))) (defun substitute-in-file-name (filename) "Substitute environment variables referred to in FILENAME (skip Urls). `$FOO' where FOO is an environment variable name means to substitute the value of that variable. The variable name should be terminated with a character not a letter, digit or underscore; otherwise, enclose the entire variable name in braces. If `/~' appears, all of FILENAME through that `/' is discarded." (if (string-match "\\(/\\|[^a-zA-Z0-9]\\)?\\(https?\\|ftp\\|telnet\\|news\\|nntp\\):[/~]" filename) (substring filename (match-beginning 2)) (hyperb:substitute-in-file-name filename))) )))) (defun hpath:enable-find-file-urls () "Enable the use of ftp and www Urls as arguments to `find-file' commands." (interactive) (cond ((hpath:find-file-urls-p) ;; Setup for remote pathname support whenever an ftp or www ;; URL is given to a find-file command. (and (boundp 'allow-remote-paths) (setq allow-remote-paths t)) (require 'tramp) (add-hook 'file-name-handler-alist (cons hpath:remote-regexp 'tramp-file-name-handler)) (hpath:handle-urls) (setq hpath:find-file-urls-mode t) (or (assq 'hpath:find-file-urls-mode minor-mode-alist) (if (fboundp 'add-minor-mode) (add-minor-mode 'hpath:find-file-urls-mode " URLs" nil 'abbrev-mode) (setq minor-mode-alist (cons '(hpath:find-file-urls-mode " URLs") minor-mode-alist)))) (force-mode-line-update) (if (called-interactively-p 'interactive) (message "(hpath:enable-find-file-urls): Find file commands will accept URLs."))) ((called-interactively-p 'interactive) (message "(hpath:enable-find-file-urls): A remote file library access like Tramp is needed to utilize find-file URLs.") (beep)))) (defun hpath:disable-find-file-urls () "Disable the use of ftp and www Urls as arguments to `find-file' commands." (interactive) (when (hpath:find-file-urls-p) (remove-hook 'file-name-handler-alist (cons hpath:remote-regexp 'tramp-file-name-handler)) ;; www-url functions are defined in "hsys-www.el". (put 'expand-file-name 'tramp nil) (put 'find-file-noselect 'tramp nil) ;; Necessary since Dired overrides other file-name-handler-alist ;; settings. (put 'expand-file-name 'dired nil) (put 'find-file-noselect 'dired nil) ;; Remove overloaded functions. (if (fboundp 'hyperb:substitute-in-file-name) (progn (defalias 'substitute-in-file-name (symbol-function 'hyperb:substitute-in-file-name)) (fmakunbound 'hyperb:substitute-in-file-name))) (setq hpath:find-file-urls-mode nil) (force-mode-line-update) (if (called-interactively-p 'interactive) (message "(hpath:disable-find-file-urls): Find file commands will not accept URLs.")))) ;;;###autoload (defun hpath:find-file-urls-mode (&optional arg) "Toggle enabling/disabling the use of ftp and www Urls as arguments to `find-file' commands. With optional prefix ARG, enable this feature if ARG is positive or turn it off otherwise." (interactive "P") (funcall (if (called-interactively-p 'interactive) #'call-interactively #'funcall) (if (if arg (> (prefix-numeric-value arg) 0) (not hpath:find-file-urls-mode)) #'hpath:enable-find-file-urls #'hpath:disable-find-file-urls))) (defun hpath:url-at-p () "Return world-wide-web universal resource locator (url) that point immediately precedes or nil. See the documentation for `hpath:url-regexp' for match-string groupings." (if (or (looking-at hpath:url-regexp) (looking-at hpath:url-regexp2) (looking-at hpath:url-regexp3)) (save-excursion (goto-char (match-end hpath:url-grpn)) (skip-chars-backward ".,?#!*()") (buffer-substring-no-properties (match-beginning hpath:url-grpn) (point))))) (defun hpath:url-p (obj) "Return t if OBJ is a world-wide-web universal resource locator (url) string, else nil. See the documentation for `hpath:url-regexp' for match groupings to use with string-match." (and (stringp obj) (or (string-match hpath:string-url-regexp obj) (string-match hpath:string-url-regexp2 obj) (string-match hpath:string-url-regexp3 obj)) t)) (defun hpath:www-at-p (&optional include-start-and-end-p) "Returns a world-wide-web link reference that point is within or nil. With optional INCLUDE-START-AND-END-P non-nil, returns list of: (link-string begin-position end-position)." (save-excursion (skip-chars-backward "^\[ \t\n\r\f\"`'|\(\{<") (cond ((not include-start-and-end-p) (hpath:url-at-p)) ((or (looking-at hpath:url-regexp) (looking-at hpath:url-regexp2) (looking-at hpath:url-regexp3)) (goto-char (match-end hpath:url-grpn)) (skip-chars-backward ".,?#!*()") (list (buffer-substring-no-properties (match-beginning hpath:url-grpn) (point)) (match-beginning hpath:url-grpn) (point)))))) (defun hpath:www-p (path) "Returns PATH iff PATH is a world-wide-web link reference, else nil." (and (stringp path) (hpath:url-p path) path)) ;;; ************************************************************************ ;;; Private functions ;;; ************************************************************************ (defun hpath:command-string (cmd filename) "Return a single string that runs a shell CMD over FILENAME. CMD may contain a single `%s' indicating where FILENAME is to be integrated, otherwise the filename is appended as an argument." ;; Permit %s substitution of filename within program. (if (string-match "[^%]%s" cmd) (format cmd filename) (format "%s \"%s\"" cmd filename))) (defun hpath:remote-available-p () "Return non-nil if a remote file access package is available, nil otherwise. Return the symbol for the appropriate package to require/use. Either the package must have been loaded already or under versions of Emacs 19 and above, it must be set for autoloading via `file-name-handler-alist', in which case, it is loaded here." (cond ((featurep 'tramp) 'tramp) ((featurep 'efs) 'efs) ((featurep 'ange-ftp) 'ange-ftp) ((boundp 'file-name-handler-alist) ; GNU Emacs (cond ((or (rassq 'tramp-file-name-handler file-name-handler-alist) (rassq 'tramp-autoload-file-name-handler file-name-handler-alist)) (require 'tramp)) ((rassq 'efs-file-handler-function file-name-handler-alist) (require 'efs)) ((rassq 'ange-ftp-hook-function file-name-handler-alist) (require 'ange-ftp)))))) (defun hpath:remote-default-user () "Return default user account for remote file access. Returns \"anonymous\" if no default user is set." (cond ((and (boundp 'tramp-default-user) (stringp tramp-default-user)) tramp-default-user) ((and (boundp 'efs-default-user) (stringp efs-default-user)) efs-default-user) ((and (boundp 'ange-ftp-default-user) (stringp ange-ftp-default-user)) ange-ftp-default-user) (t "anonymous"))) (defun hpath:delete-trailer (string) "Return string minus any trailing .?#!*() or quoting characters." (save-match-data (if (and (stringp string) (> (length string) 0) (string-match "[.?#!*()`'\"]+\\'" string)) (substring string 0 (match-beginning 0)) string))) (defun hpath:exists-p (path &optional suffix-flag) "Return PATH if it exists. (This does not mean you can read it). If PATH exists with or without a suffix from hpath:suffixes, then that pathname is returned. With optional SUFFIX-FLAG and PATH exists, return suffix added or removed from path or t." (let ((return-path) (suffix) suffixes) (if (file-exists-p path) (setq return-path path) (setq suffixes hpath:suffixes) (while suffixes (setq suffix (car suffixes)) (if (string-match (concat (regexp-quote suffix) "\\'") path) ;; Remove suffix (setq return-path (substring path 0 (match-beginning 0))) ;; Add suffix (setq return-path (concat path suffix))) (if (file-exists-p return-path) (setq suffixes nil);; found a match (setq suffix nil suffixes (cdr suffixes) return-path nil)))) (if return-path (if suffix-flag (or suffix t) return-path)))) ;; Next function from: 2006-11-02 Mats Lidell (defun hpath:find-file-mailcap (file-name) "Find command to view FILE-NAME according to the mailcap file." (if (featurep 'mailcap) (progn (mailcap-parse-mailcaps) (let (mime-type method) (if (and (string-match "\\.[^\\.]+$" file-name) (setq mime-type (mailcap-extension-to-mime (match-string-no-properties 0 file-name))) (stringp (setq method (cdr (assoc 'viewer (car (mailcap-mime-info mime-type 'all))))))) (mm-mailcap-command method file-name nil)))))) (defun hpath:find-program (filename) "Return one or a list of shell or Lisp commands to execute to display FILENAME or nil. Return nil if FILENAME is a directory name or an image file that emacs can display. See also documentation for the function (hpath:get-external-display-alist) and the variable `hpath:internal-display-alist'." (cond ((and (fboundp 'image-mode) (string-match hpath:native-image-suffixes filename)) nil) ((let ((case-fold-search nil)) (hpath:match filename hpath:internal-display-alist))) ((let ((case-fold-search t)) (hpath:match filename (hpath:get-external-display-alist)))) ((and (stringp filename) (file-directory-p filename)) nil) ;; 01/21/2019 - RSW commented this next line out since it can ;; trigger external viewers on many file types that Emacs ;; displays natively. ;; (t (hpath:find-file-mailcap filename)) )) (defun hpath:match (filename regexp-alist) "If FILENAME matches the car of any element in REGEXP-ALIST, return its cdr. REGEXP-ALIST elements must be of the form ( . ). may be a string representing an external window-system command to run or it may be a Lisp function to call with FILENAME as its single argument." (let ((cmd) elt) (while (and (not cmd) regexp-alist) (if (string-match (car (setq elt (car regexp-alist))) filename) (setq cmd (cdr elt))) (setq regexp-alist (cdr regexp-alist))) cmd)) (defun hpath:substitute-dir (var-name rest-of-path) "Returns a dir for VAR-NAME using REST-OF-PATH to find match or triggers an error when no match. VAR-NAME's value may be a directory or a list of directories. If it is a list, the first directory prepended to REST-OF-PATH which produces a valid local pathname is returned." (let (sym val) (cond ((not (stringp var-name)) (error "(hpath:substitute-dir): VAR-NAME, `%s', must be a string" var-name)) ((not (or (and (setq sym (intern-soft var-name)) (boundp sym)) (getenv var-name))) (error "(hpath:substitute-dir): VAR-NAME, \"%s\", is not a bound variable nor a set environment variable" var-name)) ((let ((case-fold-search t)) (stringp (setq val (cond ((and (boundp sym) sym) (symbol-value sym)) ((string-match "path" var-name) (split-string (getenv var-name) ":")) (t (getenv var-name)))))) (if (hpath:validate (expand-file-name rest-of-path val)) val)) ((listp val) (let* ((path (locate-file rest-of-path val (cons "" hpath:suffixes))) (dir (if path (file-name-directory path)))) (if dir (hpath:validate (directory-file-name dir)) (error "(hpath:substitute-dir): Can't find match for \"%s\"" (concat "$\{" var-name "\}/" rest-of-path)))) ;; Code prior to utilizing locate-file. ;; (let (dir path) ;; (while (and val (not dir)) ;; (setq dir (car val) val (cdr val)) ;; (or (and (stringp dir) ;; (file-name-absolute-p dir) ;; (setq path (hpath:exists-p (expand-file-name rest-of-path dir))) ;; (file-readable-p path)) ;; (setq dir nil))) ;; (if dir ;; (hpath:validate (directory-file-name dir)) ;; (error "(hpath:substitute-dir): Can't find match for \"%s\"" ;; (concat "$\{" var-name "\}/" rest-of-path)))) ) (t (error "(hpath:substitute-dir): Value of VAR-NAME, \"%s\", must be a string or list" var-name))))) (defun hpath:substitute-var-name (var-symbol var-dir-val path) "Replaces with VAR-SYMBOL any occurrences of VAR-DIR-VAL in PATH. Replacement is done iff VAR-DIR-VAL is an absolute path. If PATH is modified, returns PATH, otherwise returns nil." (if (and (stringp var-dir-val) (file-name-absolute-p var-dir-val)) (let ((new-path (hypb:replace-match-string (regexp-quote (file-name-as-directory (or var-dir-val default-directory))) path (concat "$\{" (symbol-name var-symbol) "\}/") t))) (if (equal new-path path) nil new-path)))) (provide 'hpath) ;;; hpath.el ends here