org-babel-exp.el 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. ;;; org-babel-exp.el --- Exportation of org-babel source blocks
  2. ;; Copyright (C) 2009 Eric Schulte, Dan Davison
  3. ;; Author: Eric Schulte, Dan Davison
  4. ;; Keywords: literate programming, reproducible research
  5. ;; Homepage: http://orgmode.org
  6. ;; Version: 0.01
  7. ;;; License:
  8. ;; This program is free software; you can redistribute it and/or modify
  9. ;; it under the terms of the GNU General Public License as published by
  10. ;; the Free Software Foundation; either version 3, or (at your option)
  11. ;; any later version.
  12. ;;
  13. ;; This program is distributed in the hope that it will be useful,
  14. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. ;; GNU General Public License for more details.
  17. ;;
  18. ;; You should have received a copy of the GNU General Public License
  19. ;; along with GNU Emacs; see the file COPYING. If not, write to the
  20. ;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
  21. ;; Boston, MA 02110-1301, USA.
  22. ;;; Commentary:
  23. ;; for more information see the comments in org-babel.el
  24. ;;; Code:
  25. (require 'org-babel)
  26. (require 'org-exp-blocks)
  27. (org-export-blocks-add-block '(src org-babel-exp-src-blocks nil))
  28. (add-to-list 'org-export-interblocks '(src org-babel-exp-inline-src-blocks))
  29. (add-to-list 'org-export-interblocks '(lob org-babel-exp-lob-one-liners))
  30. (add-hook 'org-export-blocks-postblock-hook 'org-exp-res/src-name-cleanup)
  31. (defvar org-babel-function-def-export-keyword "function"
  32. "When exporting a source block function, this keyword will
  33. appear in the exported version in the place of source name
  34. line. A source block is considered to be a source block function
  35. if the source name is present and is followed by a parenthesized
  36. argument list. The parentheses may be empty or contain
  37. whitespace. An example is the following which generates n random
  38. (uniform) numbers.
  39. #+source: rand(n)
  40. #+begin_src R
  41. runif(n)
  42. #+end_src
  43. ")
  44. (defvar org-babel-function-def-export-indent 4
  45. "When exporting a source block function, the block contents
  46. will be indented by this many characters. See
  47. `org-babel-function-def-export-name' for the definition of a
  48. source block function.")
  49. (defvar obe-marker nil)
  50. (defun org-babel-exp-src-blocks (body &rest headers)
  51. "Process src block for export. Depending on the 'export'
  52. headers argument in replace the source code block with...
  53. both ---- display the code and the results
  54. code ---- the default, display the code inside the block but do
  55. not process
  56. results - just like none only the block is run on export ensuring
  57. that it's results are present in the org-mode buffer
  58. none ----- do not display either code or results upon export"
  59. (interactive)
  60. (message "org-babel-exp processing...")
  61. (when (member (first headers) org-babel-interpreters)
  62. (save-excursion
  63. (goto-char (match-beginning 0))
  64. (let* ((info (org-babel-get-src-block-info))
  65. (params (third info)))
  66. ;; expand noweb references in the original file
  67. (setf (second info)
  68. (if (and (cdr (assoc :noweb params))
  69. (string= "yes" (cdr (assoc :noweb params))))
  70. (org-babel-expand-noweb-references
  71. info (get-file-buffer org-current-export-file))
  72. (second info)))
  73. (org-babel-exp-do-export info 'block)))))
  74. (defun org-babel-exp-inline-src-blocks (start end)
  75. "Process inline src blocks between START and END for export.
  76. See `org-babel-exp-src-blocks' for export options, currently the
  77. options and are taken from `org-babel-defualt-inline-header-args'."
  78. (interactive)
  79. (save-excursion
  80. (goto-char start)
  81. (while (and (< (point) end)
  82. (re-search-forward org-babel-inline-src-block-regexp end t))
  83. (let* ((info (save-match-data (org-babel-parse-inline-src-block-match)))
  84. (params (third info))
  85. (replacement
  86. (save-match-data
  87. (if (org-babel-in-example-or-verbatim)
  88. (buffer-substring (match-beginning 0) (match-end 0))
  89. ;; expand noweb references in the original file
  90. (setf (second info)
  91. (if (and (cdr (assoc :noweb params))
  92. (string= "yes" (cdr (assoc :noweb params))))
  93. (org-babel-expand-noweb-references
  94. info (get-file-buffer org-current-export-file))
  95. (second info)))
  96. (org-babel-exp-do-export info 'inline)))))
  97. (setq end (+ end (- (length replacement) (length (match-string 1)))))
  98. (replace-match replacement t t nil 1)))))
  99. (defun org-exp-res/src-name-cleanup ()
  100. "Cleanup leftover #+results and #+srcname lines as part of the
  101. org export cycle. This should only be called after all block
  102. processing has taken place."
  103. (interactive)
  104. (save-excursion
  105. (goto-char (point-min))
  106. (while (re-search-forward
  107. (concat
  108. "\\("org-babel-source-name-regexp"\\|"org-babel-result-regexp"\\)")
  109. nil t)
  110. (delete-region
  111. (progn (beginning-of-line) (point))
  112. (progn (end-of-line) (+ 1 (point)))))))
  113. (defun org-babel-in-example-or-verbatim ()
  114. "Return true if the point is currently in an escaped portion of
  115. an org-mode buffer code which should be treated as normal
  116. org-mode text."
  117. (or (org-in-indented-comment-line)
  118. (save-excursion
  119. (save-match-data
  120. (goto-char (point-at-bol))
  121. (looking-at "[ \t]*:[ \t]")))
  122. (org-in-regexps-block-p "^[ \t]*#\\+begin_src" "^[ \t]*#\\+end_src")))
  123. (defun org-babel-exp-lob-one-liners (start end)
  124. "Process #+lob (Library of Babel) calls between START and END for export.
  125. See `org-babel-exp-src-blocks' for export options. Currently the
  126. options are taken from `org-babel-default-header-args'."
  127. (interactive)
  128. (let (replacement)
  129. (save-excursion
  130. (goto-char start)
  131. (while (and (< (point) end)
  132. (re-search-forward org-babel-lob-one-liner-regexp nil t))
  133. (setq replacement
  134. (let ((lob-info (org-babel-lob-get-info)))
  135. (save-match-data
  136. (org-babel-exp-do-export
  137. (list "emacs-lisp" "results"
  138. (org-babel-merge-params
  139. org-babel-default-header-args
  140. (org-babel-parse-header-arguments
  141. (org-babel-clean-text-properties
  142. (concat ":var results="
  143. (mapconcat #'identity
  144. (butlast lob-info) " ")))))
  145. (car (last lob-info)))
  146. 'lob))))
  147. (setq end (+ end (- (length replacement) (length (match-string 0)))))
  148. (replace-match replacement t t)))))
  149. (defun org-babel-exp-do-export (info type)
  150. "Return a string containing the exported content of the current
  151. code block respecting the value of the :exports header argument."
  152. (flet ((silently () (let ((session (cdr (assoc :session (third info)))))
  153. (when (and session
  154. (not (equal "none" session))
  155. (not (assoc :noeval (third info))))
  156. (org-babel-exp-results info type 'silent))))
  157. (clean () (org-babel-remove-result info)))
  158. (case (intern (or (cdr (assoc :exports (third info))) "code"))
  159. ('none (silently) (clean) "")
  160. ('code (silently) (clean) (org-babel-exp-code info type))
  161. ('results (org-babel-exp-results info type))
  162. ('both (concat (org-babel-exp-code info type)
  163. "\n\n"
  164. (org-babel-exp-results info type))))))
  165. (defun org-babel-exp-code (info type)
  166. "Return the code the current code block in a manner suitable
  167. for exportation by org-mode. This function is called by
  168. `org-babel-exp-do-export'. The code block will not be
  169. evaluated."
  170. (let ((lang (first info))
  171. (body (second info))
  172. (switches (fourth info))
  173. (name (fifth info))
  174. (args (mapcar
  175. #'cdr
  176. (remove-if-not (lambda (el) (eq :var (car el))) (third info)))))
  177. (case type
  178. ('inline (format "=%s=" body))
  179. ('block
  180. (let ((str
  181. (format "#+BEGIN_SRC %s %s\n%s%s#+END_SRC\n" lang switches body
  182. (if (and body (string-match "\n$" body))
  183. "" "\n"))))
  184. (when name
  185. (add-text-properties
  186. 0 (length str)
  187. (list 'org-caption
  188. (format "%s(%s)"
  189. name
  190. (mapconcat #'identity args ", ")))
  191. str))
  192. str))
  193. ('lob
  194. (let ((call-line (and (string-match "results=" (car args))
  195. (substring (car args) (match-end 0)))))
  196. (cond
  197. ((eq backend 'html)
  198. (format "\n#+HTML: <label class=\"org-src-name\">%s</label>\n"
  199. call-line))
  200. ((t (format ": %s\n" call-line)))))))))
  201. (defun org-babel-exp-results (info type &optional silent)
  202. "Return the results of the current code block in a manner
  203. suitable for exportation by org-mode. This function is called by
  204. `org-babel-exp-do-export'. The code block will be evaluated.
  205. Optional argument SILENT can be used to inhibit insertion of
  206. results into the buffer."
  207. (let ((lang (first info))
  208. (body (second info))
  209. (params
  210. ;; lets ensure that we lookup references in the original file
  211. (mapcar
  212. (lambda (pair)
  213. (if (and org-current-export-file
  214. (eq (car pair) :var)
  215. (string-match org-babel-ref-split-regexp (cdr pair))
  216. (null (org-babel-ref-literal (match-string 2 (cdr pair)))))
  217. `(:var . ,(concat (match-string 1 (cdr pair))
  218. "=" org-current-export-file
  219. ":" (match-string 2 (cdr pair))))
  220. pair))
  221. (third info))))
  222. (case type
  223. ('inline
  224. (let ((raw (org-babel-execute-src-block
  225. nil info '((:results . "silent"))))
  226. (result-params (split-string (cdr (assoc :results params)))))
  227. (unless silent
  228. (cond ;; respect the value of the :results header argument
  229. ((member "file" result-params)
  230. (org-babel-result-to-file raw))
  231. ((or (member "raw" result-params) (member "org" result-params))
  232. (format "%s" raw))
  233. ((member "code" result-params)
  234. (format "src_%s{%s}" lang raw))
  235. (t
  236. (if (stringp raw)
  237. (if (= 0 (length raw)) "=(no results)="
  238. (format "%s" raw))
  239. (format "%S" raw)))))))
  240. ('block
  241. (org-babel-execute-src-block
  242. nil info (org-babel-merge-params
  243. params `((:results . ,(if silent "silent" "replace")))))
  244. "")
  245. ('lob
  246. (save-excursion
  247. (re-search-backward org-babel-lob-one-liner-regexp nil t)
  248. (org-babel-execute-src-block
  249. nil info (org-babel-merge-params
  250. params `((:results . ,(if silent "silent" "replace")))))
  251. "")))))
  252. (provide 'org-babel-exp)
  253. ;;; org-babel-exp.el ends here