summaryrefslogtreecommitdiff
path: root/lisp/eshell/em-basic.el
blob: 8f68a750bd7bce33593a4c60cb76a114743d9fbb (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
;;; em-basic.el --- basic shell builtin commands  -*- lexical-binding:t -*-

;; Copyright (C) 1999-2024 Free Software Foundation, Inc.

;; Author: John Wiegley <johnw@gnu.org>

;; This file is part of GNU Emacs.

;; 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 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
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
;; GNU General Public License for more details.

;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs.  If not, see <https://www.gnu.org/licenses/>.

;;; Commentary:

;; There are very few basic Eshell commands -- so-called builtins.
;; They are: echo, umask, and version.
;;
;;;_* `echo'
;;
;; The `echo' command repeats its arguments to the screen.  It is
;; optional whether this is done in a Lisp-friendly fashion (so that
;; the value of echo is useful to a Lisp command using the result of
;; echo as an argument), or whether it should try to act like a normal
;; shell echo, and always result in a flat string being returned.

;; An example of the difference is the following:
;;
;;   echo Hello world
;;
;; If `eshell-plain-echo-behavior' is non-nil, this will yield the
;; string "Hello world".  If Lisp behavior is enabled, however, it
;; will yield a list whose two elements are the strings "Hello" and
;; "world".  The way to write an equivalent expression for both would
;; be:
;;
;;   echo "Hello world"
;;
;; This always returns a single string.
;;
;;;_* `umask'
;;
;; The umask command changes the default file permissions for newly
;; created files.  It uses the same syntax as bash.

;;; Code:

(require 'esh-cmd)
(require 'esh-io)
(require 'esh-opt)
(require 'esh-util)

;;;###autoload
(progn
(defgroup eshell-basic nil
  "The \"basic\" code provides a set of convenience functions which
are traditionally considered shell builtins.  Since all of the
functionality provided by them is accessible through Lisp, they are
not really builtins at all, but offer a command-oriented way to do the
same thing."
  :tag "Basic shell commands"
  :group 'eshell-module))

(defcustom eshell-plain-echo-behavior nil
  "If non-nil, `echo' tries to behave like an ordinary shell echo.
This comes at some detriment to Lisp functionality.  However, the Lisp
equivalent of `echo' can always be achieved by using `identity'."
  :type 'boolean
  :group 'eshell-basic)

;;; Functions:

(defun eshell-echo (args &optional output-newline)
  "Implementation code for a Lisp version of `echo'.
It returns a formatted value that should be passed to `eshell-print'
or `eshell-printn' for display."
  (if eshell-plain-echo-behavior
      (progn
        ;; If the output does not end in a newline, do not emit one.
        (setq eshell-ensure-newline-p nil)
        (concat (apply #'eshell-flatten-and-stringify args)
                (when output-newline "\n")))
    (let ((value
	   (cond
	    ((= (length args) 0) "")
	    ((= (length args) 1)
	     (car args))
	    (t
	     (mapcar
              (lambda (arg)
                (if (stringp arg)
                    (set-text-properties 0 (length arg) nil arg))
                arg)
	      args)))))
      (if output-newline
	  (cond
	   ((stringp value)
	    (concat value "\n"))
	   ((listp value)
	    (append value (list "\n")))
	   (t
	    (concat (eshell-stringify value) "\n")))
	value))))

(defun eshell/echo (&rest args)
  "Implementation of `echo'.  See `eshell-plain-echo-behavior'."
  (eshell-eval-using-options
   "echo" args
   '((?n nil (nil) output-newline
         "do not output the trailing newline")
     (?N nil (t)   output-newline
         "terminate with a newline")
     (?E nil nil   _disable-escapes
         "don't interpret backslash escapes (default)")
     (?h "help" nil nil
         "output this help screen")
     :preserve-args
     :usage "[OPTION]... [OBJECT]...")
   (if eshell-plain-echo-behavior
       (eshell-echo args (if output-newline (car output-newline) t))
     ;; In Emacs 28.1 and earlier, "-n" was used to add a newline to
     ;; non-plain echo in Eshell.  This caused confusion due to "-n"
     ;; generally having the opposite meaning for echo.  Retain this
     ;; compatibility for the time being.  For more info, see
     ;; bug#27361.
     (when (equal output-newline '(nil))
       (display-warning
        '(eshell echo)
        "To terminate with a newline, you should use -N instead."))
     (eshell-echo args output-newline))))

(defun eshell/printnl (&rest args)
  "Print out each of the arguments as strings, separated by newlines."
  (let ((elems (flatten-tree args)))
    (dolist (elem elems)
      (eshell-printn (eshell-stringify elem)))))

(defun eshell/listify (&rest args)
  "Return the argument(s) as a single list."
  (if (> (length args) 1)
      args
    (if (listp (car args))
	(car args)
      (list (car args)))))

(defun eshell/umask (&rest args)
  "Shell-like implementation of `umask'."
  (eshell-eval-using-options
   "umask" args
   '((?S "symbolic" nil symbolic-p "display umask symbolically")
     (?h "help" nil nil  "display this usage message")
     :preserve-args
     :usage "[-S] [mode]")
   (cond
    (symbolic-p
     (let ((mode (default-file-modes)))
       (eshell-printn
        (format "u=%s,g=%s,o=%s"
                (concat (and (= (logand mode 64) 64) "r")
                        (and (= (logand mode 128) 128) "w")
                        (and (= (logand mode 256) 256) "x"))
                (concat (and (= (logand mode 8) 8) "r")
                        (and (= (logand mode 16) 16) "w")
                        (and (= (logand mode 32) 32) "x"))
                (concat (and (= (logand mode 1) 1) "r")
                        (and (= (logand mode 2) 2) "w")
                        (and (= (logand mode 4) 4) "x"))))))
    ((not args)
     (eshell-printn (format "%03o" (logand (lognot (default-file-modes))
                                           #o777))))
    (t
     (when (stringp (car args))
       (if (string-match "^[0-7]+$" (car args))
           (setcar args (string-to-number (car args) 8))
         (error "Setting umask symbolically is not yet implemented")))
     (set-default-file-modes (- #o777 (car args)))
     (eshell-print
      "Warning: umask changed for all new files created by Emacs.\n")))
   nil))

(put 'eshell/umask 'eshell-no-numeric-conversions t)

(defun eshell/eshell-debug (&rest args)
  "A command for toggling certain debug variables."
  (eshell-eval-using-options
   "eshell-debug" args
   '((?h "help" nil nil "display this usage message")
     :usage "[KIND]...
This command is used to aid in debugging problems related to Eshell
itself.  It is not useful for anything else.  The recognized `kinds'
are:

   error       stops Eshell from trapping errors
   form        shows command form manipulation in `*eshell last cmd*'
   process     shows process events in `*eshell last cmd*'")
   (if args
       (dolist (kind args)
         (if (equal kind "error")
             (setq eshell-handle-errors (not eshell-handle-errors))
           (let ((kind-sym (intern kind)))
             (if (memq kind-sym eshell-debug-command)
                 (setq eshell-debug-command
                       (delq kind-sym eshell-debug-command))
               (push kind-sym eshell-debug-command)))))
     ;; Output the currently-enabled debug kinds.
     (unless eshell-handle-errors
       (eshell-print "errors\n"))
     (dolist (kind eshell-debug-command)
       (eshell-printn (symbol-name kind))))))

(defun pcomplete/eshell-mode/eshell-debug ()
  "Completion for the `debug' command."
  (while (pcomplete-here '("error" "form" "process"))))

(provide 'em-basic)

;; Local Variables:
;; generated-autoload-file: "esh-groups.el"
;; End:

;;; em-basic.el ends here