1 ;;; ob-eval.el --- org-babel functions for external code evaluation
3 ;; Copyright (C) 2009, 2010 Free Software Foundation, Inc.
5 ;; Author: Eric Schulte
6 ;; Keywords: literate programming, reproducible research, comint
7 ;; Homepage: http://orgmode.org
10 ;; This file is part of GNU Emacs.
12 ;; GNU Emacs is free software: you can redistribute it and/or modify
13 ;; it under the terms of the GNU General Public License as published by
14 ;; the Free Software Foundation, either version 3 of the License, or
15 ;; (at your option) any later version.
17 ;; GNU Emacs is distributed in the hope that it will be useful,
18 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
19 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 ;; GNU General Public License for more details.
22 ;; You should have received a copy of the GNU General Public License
23 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
27 ;; These functions build existing Emacs support for executing external
32 (eval-when-compile (require 'cl
))
34 (defun org-babel-eval-error-notify (exit-code stderr
)
35 "Open a buffer to display STDERR and a message with the value of EXIT-CODE."
36 (let ((buf (get-buffer-create "*Org-Babel Error Output*")))
37 (with-current-buffer buf
38 (goto-char (point-max))
39 (save-excursion (insert stderr
)))
41 (message "Babel evaluation exited with code %S" exit-code
))
43 (defun org-babel-eval (cmd body
)
45 If CMD succeeds then return its results, otherwise display
46 STDERR with `org-babel-eval-error-notify'."
47 (let ((err-buff (get-buffer-create "*Org-Babel Error*")) exit-code
)
48 (with-current-buffer err-buff
(erase-buffer))
52 (org-babel-shell-command-on-region
53 (point-min) (point-max) cmd t
'replace err-buff
))
54 (if (or (not (numberp exit-code
)) (> exit-code
0))
56 (with-current-buffer err-buff
57 (org-babel-eval-error-notify exit-code
(buffer-string)))
61 (defun org-babel-eval-read-file (file)
62 "Return the contents of FILE as a string."
63 (with-temp-buffer (insert-file-contents file
)
66 (defun org-babel-shell-command-on-region (start end command
67 &optional output-buffer replace
68 error-buffer display-error-buffer
)
69 "Execute COMMAND in an inferior shell with region as input.
71 Fixes bugs in the emacs 23.1.1 version of `shell-command-on-region'
73 Normally display output (if any) in temp buffer `*Shell Command Output*';
74 Prefix arg means replace the region with it. Return the exit code of
77 To specify a coding system for converting non-ASCII characters in
78 the input and output to the shell command, use
79 \\[universal-coding-system-argument] before this command. By
80 default, the input (from the current buffer) is encoded in the
81 same coding system that will be used to save the file,
82 `buffer-file-coding-system'. If the output is going to replace
83 the region, then it is decoded from that same coding system.
85 The noninteractive arguments are START, END, COMMAND,
86 OUTPUT-BUFFER, REPLACE, ERROR-BUFFER, and DISPLAY-ERROR-BUFFER.
87 Noninteractive callers can specify coding systems by binding
88 `coding-system-for-read' and `coding-system-for-write'.
90 If the command generates output, the output may be displayed
91 in the echo area or in a buffer.
92 If the output is short enough to display in the echo area
93 \(determined by the variable `max-mini-window-height' if
94 `resize-mini-windows' is non-nil), it is shown there. Otherwise
95 it is displayed in the buffer `*Shell Command Output*'. The output
96 is available in that buffer in both cases.
98 If there is output and an error, a message about the error
99 appears at the end of the output.
101 If there is no output, or if output is inserted in the current buffer,
102 then `*Shell Command Output*' is deleted.
104 If the optional fourth argument OUTPUT-BUFFER is non-nil,
105 that says to put the output in some other buffer.
106 If OUTPUT-BUFFER is a buffer or buffer name, put the output there.
107 If OUTPUT-BUFFER is not a buffer and not nil,
108 insert output in the current buffer.
109 In either case, the output is inserted after point (leaving mark after it).
111 If REPLACE, the optional fifth argument, is non-nil, that means insert
112 the output in place of text from START to END, putting point and mark
115 If optional sixth argument ERROR-BUFFER is non-nil, it is a buffer
116 or buffer name to which to direct the command's standard error output.
117 If it is nil, error output is mingled with regular output.
118 If DISPLAY-ERROR-BUFFER is non-nil, display the error buffer if there
119 were any errors. (This is always t, interactively.)
120 In an interactive call, the variable `shell-command-default-error-buffer'
121 specifies the value of ERROR-BUFFER."
122 (interactive (let (string)
124 (error "The mark is not set now, so there is no region"))
125 ;; Do this before calling region-beginning
126 ;; and region-end, in case subprocess output
127 ;; relocates them while we are in the minibuffer.
128 (setq string
(read-shell-command "Shell command on region: "))
129 ;; call-interactively recognizes region-beginning and
130 ;; region-end specially, leaving them in the history.
131 (list (region-beginning) (region-end)
135 shell-command-default-error-buffer
140 (expand-file-name "scor"
141 (or (unless (featurep 'xemacs
)
142 small-temporary-file-directory
)
143 temporary-file-directory
)))
148 (not (or (bufferp output-buffer
) (stringp output-buffer
)))))
149 ;; Replace specified region with output from command.
150 (let ((swap (and replace
(< start end
))))
151 ;; Don't muck with mark unless REPLACE says we should.
153 (and replace
(push-mark (point) 'nomsg
))
155 (call-process-region start end shell-file-name t
157 (list output-buffer error-file
)
159 nil shell-command-switch command
))
160 ;; It is rude to delete a buffer which the command is not using.
161 ;; (let ((shell-buffer (get-buffer "*Shell Command Output*")))
162 ;; (and shell-buffer (not (eq shell-buffer (current-buffer)))
163 ;; (kill-buffer shell-buffer)))
164 ;; Don't muck with mark unless REPLACE says we should.
165 (and replace swap
(exchange-point-and-mark)))
166 ;; No prefix argument: put the output in a temp buffer,
167 ;; replacing its entire contents.
168 (let ((buffer (get-buffer-create
169 (or output-buffer
"*Shell Command Output*"))))
171 (if (eq buffer
(current-buffer))
172 ;; If the input is the same buffer as the output,
173 ;; delete everything but the specified region,
174 ;; then replace that region with the output.
175 (progn (setq buffer-read-only nil
)
176 (delete-region (max start end
) (point-max))
177 (delete-region (point-min) (min start end
))
179 (call-process-region (point-min) (point-max)
184 nil shell-command-switch
186 ;; Clear the output buffer, then run the command with
188 (let ((directory default-directory
))
189 (with-current-buffer buffer
190 (setq buffer-read-only nil
)
191 (if (not output-buffer
)
192 (setq default-directory directory
))
195 (call-process-region start end shell-file-name nil
197 (list buffer error-file
)
199 nil shell-command-switch command
)))
200 ;; Report the output.
201 (with-current-buffer buffer
202 (setq mode-line-process
203 (cond ((null exit-status
)
205 ((stringp exit-status
)
206 (format " - Signal [%s]" exit-status
))
207 ((not (equal 0 exit-status
))
208 (format " - Exit [%d]" exit-status
)))))
209 (if (with-current-buffer buffer
(> (point-max) (point-min)))
210 ;; There's some output, display it
211 (display-message-or-buffer buffer
)
215 (< 0 (nth 7 (file-attributes error-file
))))
218 (cond ((null exit-status
)
219 (message "(Shell command failed with error)"))
220 ((equal 0 exit-status
)
221 (message "(Shell command succeeded with %s)"
223 ((stringp exit-status
)
224 (message "(Shell command killed by signal %s)"
227 (message "(Shell command failed with code %d and %s)"
228 exit-status output
))))
229 ;; Don't kill: there might be useful info in the undo-log.
230 ;; (kill-buffer buffer)
233 (when (and error-file
(file-exists-p error-file
))
234 (if (< 0 (nth 7 (file-attributes error-file
)))
235 (with-current-buffer (get-buffer-create error-buffer
)
236 (let ((pos-from-end (- (point-max) (point))))
239 ;; Do no formatting while reading error file,
240 ;; because that can run a shell command, and we
241 ;; don't want that to cause an infinite recursion.
242 (format-insert-file error-file nil
)
243 ;; Put point after the inserted errors.
244 (goto-char (- (point-max) pos-from-end
)))
245 (and display-error-buffer
246 (display-buffer (current-buffer)))))
247 (delete-file error-file
))
252 ;; arch-tag: 5328b17f-957d-42d9-94da-a2952682d04d
254 ;;; ob-eval.el ends here