| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283 | ;;; ob-clojure.el --- Babel Functions for Clojure    -*- lexical-binding: t; -*-;; Copyright (C) 2009-2020 Free Software Foundation, Inc.;; Author: Joel Boehland, Eric Schulte, Oleh Krehel, Frederick Giasson;;;; Keywords: literate programming, reproducible research;; Homepage: https://orgmode.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:;; Support for evaluating clojure code;; Requirements:;; - clojure (at least 1.2.0);; - clojure-mode;; - either cider or SLIME;; For Cider, see https://github.com/clojure-emacs/cider;; For SLIME, the best way to install these components is by following;; the directions as set out by Phil Hagelberg (Technomancy) on the;; web page: http://technomancy.us/126;;; Code:(require 'cl-lib)(require 'ob)(require 'org-macs)(declare-function cider-jack-in "ext:cider" (&optional prompt-project cljs-too))(declare-function cider-current-connection "ext:cider-client" (&optional type))(declare-function cider-current-ns "ext:cider-client" ())(declare-function cider-repls "ext:cider-connection" (&optional type ensure))(declare-function nrepl--merge "ext:nrepl-client" (dict1 dict2))(declare-function nrepl-dict-get "ext:nrepl-client" (dict key))(declare-function nrepl-dict-put "ext:nrepl-client" (dict key value))(declare-function nrepl-request:eval "ext:nrepl-client" (input callback connection &optional ns line column additional-params tooling))(declare-function nrepl-sync-request:eval "ext:nrepl-client" (input connection &optional ns tooling))(declare-function slime-eval "ext:slime" (sexp &optional package))(defvar nrepl-sync-request-timeout)(defvar cider-buffer-ns)(defvar sesman-system)(defvar cider-version)(defvar org-babel-tangle-lang-exts)(add-to-list 'org-babel-tangle-lang-exts '("clojure" . "clj"))(defvar org-babel-default-header-args:clojure '())(defvar org-babel-header-args:clojure '((ns . :any)					(package . :any)))(defcustom org-babel-clojure-sync-nrepl-timeout 10  "Timeout value, in seconds, of a Clojure sync call.If the value is nil, timeout is disabled."  :group 'org-babel  :type 'integer  :version "26.1"  :package-version '(Org . "9.1")  :safe #'wholenump)(defcustom org-babel-clojure-backend  (cond ((featurep 'cider) 'cider)	(t 'slime))  "Backend used to evaluate Clojure code blocks."  :group 'org-babel  :type '(choice	  (const :tag "cider" cider)	  (const :tag "SLIME" slime)))(defcustom org-babel-clojure-default-ns "user"  "Default Clojure namespace for source block when finding ns failed."  :type 'string  :group 'org-babel)(defun org-babel-clojure-cider-current-ns ()  "Like `cider-current-ns' except `cider-find-ns'."  (or cider-buffer-ns      (let ((repl-buf (cider-current-connection)))	(and repl-buf (buffer-local-value 'cider-buffer-ns repl-buf)))      org-babel-clojure-default-ns))(defun org-babel-expand-body:clojure (body params)  "Expand BODY according to PARAMS, return the expanded body."  (let* ((vars (org-babel--get-vars params))	 (ns (or (cdr (assq :ns params))		 (org-babel-clojure-cider-current-ns)))	 (result-params (cdr (assq :result-params params)))	 (print-level nil)	 (print-length nil)	 (body (org-trim		(concat		 ;; Source block specified namespace :ns.		 (and (cdr (assq :ns params)) (format "(ns %s)\n" ns))		 ;; Variables binding.		 (if (null vars) (org-trim body)		   (format "(let [%s]\n%s)"			   (mapconcat			    (lambda (var)			      (format "%S (quote %S)" (car var) (cdr var)))			    vars			    "\n      ")			   body))))))    (if (or (member "code" result-params)	    (member "pp" result-params))	(format "(clojure.pprint/pprint (do %s))" body)      body)))(defun org-babel-execute:clojure (body params)  "Execute a block of Clojure code with Babel.The underlying process performed by the code block can be outputusing the :show-process parameter."  (let* ((expanded (org-babel-expand-body:clojure body params))	 (response (list 'dict))         result)    (cl-case org-babel-clojure-backend      (cider       (require 'cider)       (let ((result-params (cdr (assq :result-params params)))	     (show (cdr (assq :show-process params))))         (if (member show '(nil "no"))	     ;; Run code without showing the process.	     (progn	       (setq response		     (let ((nrepl-sync-request-timeout			    org-babel-clojure-sync-nrepl-timeout))		       (nrepl-sync-request:eval expanded						(cider-current-connection))))	       (setq result		     (concat		      (nrepl-dict-get response				      (if (or (member "output" result-params)					      (member "pp" result-params))					  "out"					"value"))		      (nrepl-dict-get response "ex")		      (nrepl-dict-get response "root-ex")		      (nrepl-dict-get response "err"))))	   ;; Show the process in an output buffer/window.           (let ((process-buffer (switch-to-buffer-other-window				  "*Clojure Show Process Sub Buffer*"))		 status)	     ;; Run the Clojure code in nREPL.	     (nrepl-request:eval	      expanded	      (lambda (resp)		(when (member "out" resp)		  ;; Print the output of the nREPL in the output buffer.		  (princ (nrepl-dict-get resp "out") process-buffer))		(when (member "ex" resp)		  ;; In case there is an exception, then add it to the		  ;; output buffer as well.		  (princ (nrepl-dict-get resp "ex") process-buffer)		  (princ (nrepl-dict-get resp "root-ex") process-buffer))		(when (member "err" resp)		  ;; In case there is an error, then add it to the		  ;; output buffer as well.		  (princ (nrepl-dict-get resp "err") process-buffer))		(nrepl--merge response resp)		;; Update the status of the nREPL output session.		(setq status (nrepl-dict-get response "status")))	      (cider-current-connection))	     ;; Wait until the nREPL code finished to be processed.	     (while (not (member "done" status))	       (nrepl-dict-put response "status" (remove "need-input" status))	       (accept-process-output nil 0.01)	       (redisplay))	     ;; Delete the show buffer & window when the processing is	     ;; finalized.	     (mapc #'delete-window		   (get-buffer-window-list process-buffer nil t))	     (kill-buffer process-buffer)	     ;; Put the output or the value in the result section of	     ;; the code block.	     (setq result		   (concat		    (nrepl-dict-get response				    (if (or (member "output" result-params)					    (member "pp" result-params))					"out"				      "value"))		    (nrepl-dict-get response "ex")		    (nrepl-dict-get response "root-ex")		    (nrepl-dict-get response "err")))))))      (slime       (require 'slime)       (with-temp-buffer	 (insert expanded)	 (setq result	       (slime-eval		`(swank:eval-and-grab-output		  ,(buffer-substring-no-properties (point-min) (point-max)))		(cdr (assq :package params)))))))    (org-babel-result-cond (cdr (assq :result-params params))      result      (condition-case nil (org-babel-script-escape result)	(error result)))))(defun org-babel-clojure-initiate-session (&optional session _params)  "Initiate a session named SESSION according to PARAMS."  (when (and session (not (string= session "none")))    (save-window-excursion      (cond       ((org-babel-comint-buffer-livep session) nil)       ;; CIDER jack-in to the Clojure project directory.       ((eq org-babel-clojure-backend 'cider)        (require 'cider)        (let ((session-buffer	       (save-window-excursion		 (if (version< cider-version "0.18.0")		     ;; Older CIDER (without sesman) still need to use		     ;; old way.		     (cider-jack-in nil) ;jack-in without project		   ;; New CIDER (with sesman to manage sessions).		   (unless (cider-repls)		     (let ((sesman-system 'CIDER))		       (call-interactively 'sesman-link-with-directory))))                 (current-buffer))))          (when (org-babel-comint-buffer-livep session-buffer)            (sit-for .25)	    session-buffer)))       ((eq org-babel-clojure-backend 'slime)        (error "Session evaluation with SLIME is not supported"))       (t        (error "Session initiate failed")))      (get-buffer session))))(defun org-babel-prep-session:clojure (session params)  "Prepare SESSION according to the header arguments specified in PARAMS."  (let ((session (org-babel-clojure-initiate-session session))        (var-lines (org-babel-variable-assignments:clojure params)))    (when session      (org-babel-comint-in-buffer session	(dolist (var var-lines)	  (insert var)	  (comint-send-input nil t)	  (org-babel-comint-wait-for-output session)	  (sit-for .1)	  (goto-char (point-max)))))    session))(defun org-babel-clojure-var-to-clojure (var)  "Convert src block's VAR to Clojure variable."  (cond   ((listp var)    (replace-regexp-in-string "(" "'(" var))   ((stringp var)    ;; Wrap Babel passed-in header argument value with quotes in Clojure.    (format "\"%s\"" var))   (t    (format "%S" var))))(defun org-babel-variable-assignments:clojure (params)  "Return a list of Clojure statements assigning the block's variables in PARAMS."  (mapcar   (lambda (pair)     (format "(def %s %s)"             (car pair)             (org-babel-clojure-var-to-clojure (cdr pair))))   (org-babel--get-vars params)))(provide 'ob-clojure);;; ob-clojure.el ends here
 |