310 lines
13 KiB
EmacsLisp
310 lines
13 KiB
EmacsLisp
|
;;; org-babel-oz.el --- org-babel functions for Oz evaluation
|
||
|
|
||
|
;; Copyright (C) 2009 Torsten Anders and Eric Schulte
|
||
|
|
||
|
;; Author: Torsten Anders and Eric Schulte
|
||
|
;; Keywords: literate programming, reproducible research
|
||
|
;; Homepage: http://orgmode.org
|
||
|
;; Version: 0.01
|
||
|
|
||
|
;;; License:
|
||
|
|
||
|
;; This program 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, or (at your option)
|
||
|
;; any later version.
|
||
|
;;
|
||
|
;; This program 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; see the file COPYING. If not, write to the
|
||
|
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
|
;; Boston, MA 02110-1301, USA.
|
||
|
|
||
|
;;; Commentary:
|
||
|
|
||
|
;; Org-Babel support for evaluating Oz source code.
|
||
|
;;
|
||
|
;; Oz code is always send to the Oz Programming Environment (OPI), the
|
||
|
;; Emacs mode and compiler interface for Oz programs. Therefore, only
|
||
|
;; session mode is supported. In practice, non-session code blocks are
|
||
|
;; handled equally well by the session mode. However, only a single
|
||
|
;; session is supported. Consequently, the :session header argument is
|
||
|
;; ignored.
|
||
|
;;
|
||
|
;; The Org-babel header argument :results is interpreted as
|
||
|
;; follows. :results output requires the respective code block to be
|
||
|
;; an Oz statement and :results value requires an Oz
|
||
|
;; expression. Currently, results are only supported for expressions
|
||
|
;; (i.e. the result of :results output is always nil).
|
||
|
;;
|
||
|
;; Expression evaluation happens synchronously. Therefore there is an
|
||
|
;; additional header argument :wait-time <number>, which specifies the
|
||
|
;; maximum time to wait for the result of a given expression. nil
|
||
|
;; means to wait as long as it takes to get a result (potentially wait
|
||
|
;; forever).
|
||
|
;;
|
||
|
;; NOTE: Currently the copyright of this file may not be in a state to
|
||
|
;; permit inclusion as core software into Emacs or Org-mode.
|
||
|
|
||
|
;;; Requirements:
|
||
|
|
||
|
;; - Mozart Programming System, the implementation of the Oz
|
||
|
;; programming language (http://www.mozart-oz.org/), which includes
|
||
|
;; the major mode mozart for editing Oz programs.
|
||
|
;;
|
||
|
;; - StartOzServer.oz which is located in the contrib/scripts
|
||
|
;; directory of the Org-mode repository
|
||
|
|
||
|
;;; TODO:
|
||
|
|
||
|
;; - Decide: set communication to \\switch -threadedqueries?
|
||
|
;;
|
||
|
;; - Only start Oz compiler when required, e.g., load Org-babel only when needed?
|
||
|
;;
|
||
|
;; - Avoid synchronous evaluation to avoid blocking Emacs (complex
|
||
|
;; Strasheela programs can take long to find a result..). In order
|
||
|
;; to cleanly map code blocks to their associated results (which can
|
||
|
;; arrive then in any order) I could use IDs
|
||
|
;; (e.g. integers). However, how do I do concurrency in Emacs Lisp,
|
||
|
;; and how can I define org-babel-execute:oz concurrently.
|
||
|
;;
|
||
|
;; - Expressions are rarely used in Oz at the top-level, and using
|
||
|
;; them in documentation and Literate Programs will cause
|
||
|
;; confusion. Idea: hide expression from reader and instead show
|
||
|
;; them statement (e.g., MIDI output statement) and then include
|
||
|
;; result in Org file. Implementation: for expressions (:results
|
||
|
;; value) support an additional header argument that takes arbitrary
|
||
|
;; Oz code. This code is not seen by the reader, but will be used
|
||
|
;; for the actual expression at the end. Alternative: feed all
|
||
|
;; relevant code as statement (:results output), then add expression
|
||
|
;; as extra code block which outputs, e.g., file name (so the file
|
||
|
;; name must be accessible by global var), but the code of this
|
||
|
;; extra codeblock is not seen. Hm, in that case it might be even
|
||
|
;; more easy to manually add this link to the Org file.
|
||
|
;;
|
||
|
|
||
|
|
||
|
(require 'org-babel)
|
||
|
;;; major mode for editing Oz programs
|
||
|
(require 'mozart)
|
||
|
|
||
|
;; Add Oz to the list of supported languages. Org-babel will match
|
||
|
;; the string below against the declared language of the source-code
|
||
|
;; block.
|
||
|
(org-babel-add-interpreter "oz")
|
||
|
|
||
|
;; specify the name and file extension for Oz
|
||
|
(add-to-list 'org-babel-tangle-langs '("oz" "oz" nil nil))
|
||
|
|
||
|
;;
|
||
|
;; Interface to communicate with Oz.
|
||
|
;; (1) For statements without any results: oz-send-string
|
||
|
;; (2) For expressions with a single result: oz-send-string-expression
|
||
|
;; (defined in org-babel-oz-ResultsValue.el)
|
||
|
;;
|
||
|
|
||
|
;; oz-send-string-expression implements an additional very direct
|
||
|
;; communication between Org-babel and the Oz compiler. Communication
|
||
|
;; with the Oz server works already without this code via the function
|
||
|
;; oz-send-string from mozart.el.in, but this function does not get
|
||
|
;; back any results from Oz to Emacs. The following code creates a
|
||
|
;; socket for sending code to the OPI compiler and results are
|
||
|
;; returned by the same socket. On the Oz side, a socket is opened and
|
||
|
;; conected to the compiler of the OPI (via oz-send-string). On the
|
||
|
;; Emacs side, a connection to this socket is created for feeding code
|
||
|
;; and receiving results. This additional communication channel to the
|
||
|
;; OPI compiler ensures that results are returned cleanly (e.g., only
|
||
|
;; the result of the sent code is returned, no parsing or any
|
||
|
;; processing of *Oz Emulator* is required).
|
||
|
;;
|
||
|
;; There is no buffer, nor sentinel involved. Oz code is send
|
||
|
;; directly, and results from Oz are send back, but Emacs Lisp
|
||
|
;; requires a filter function for processing results.
|
||
|
|
||
|
(defvar org-babel-oz-server-dir
|
||
|
(file-name-as-directory
|
||
|
(expand-file-name
|
||
|
"scripts"
|
||
|
(file-name-as-directory
|
||
|
(expand-file-name
|
||
|
"../../.."
|
||
|
(file-name-directory (or load-file-name buffer-file-name))))))
|
||
|
"Path to the contrib/scripts directory in which
|
||
|
StartOzServer.oz is located.")
|
||
|
|
||
|
(defvar org-babel-oz-port 6001
|
||
|
"Port for communicating with Oz compiler.")
|
||
|
(defvar org-babel-oz-OPI-socket nil
|
||
|
"Socket for communicating with OPI.")
|
||
|
|
||
|
(defvar org-babel-oz-collected-result nil
|
||
|
"Aux var to hand result from org-babel-oz-filter to oz-send-string-expression.")
|
||
|
(defun org-babel-oz-filter (proc string)
|
||
|
"Processes output from socket org-babel-oz-OPI-socket."
|
||
|
;; (setq org-babel-oz-collected-results (cons string org-babel-oz-collected-results))
|
||
|
(setq org-babel-oz-collected-result string)
|
||
|
)
|
||
|
|
||
|
|
||
|
(defun org-babel-oz-create-socket ()
|
||
|
(message "Create OPI socket for evaluating expressions")
|
||
|
;; Start Oz directly
|
||
|
(run-oz)
|
||
|
;; Create socket on Oz side (after Oz was started).
|
||
|
(oz-send-string (concat "\\insert '" org-babel-oz-server-dir "StartOzServer.oz'"))
|
||
|
;; Wait until socket is created before connecting to it.
|
||
|
;; Quick hack: wait 3 sec
|
||
|
;;
|
||
|
;; extending time to 30 secs does not help when starting Emacs for
|
||
|
;; the first time (and computer does nothing else)
|
||
|
(sit-for 3)
|
||
|
;; connect to OPI socket
|
||
|
(setq org-babel-oz-OPI-socket
|
||
|
;; Creates a socket. I/O interface of Emacs sockets as for processes.
|
||
|
(open-network-stream "*Org-babel-OPI-socket*" nil "localhost" org-babel-oz-port))
|
||
|
;; install filter
|
||
|
(set-process-filter org-babel-oz-OPI-socket #'org-babel-oz-filter)
|
||
|
)
|
||
|
|
||
|
;; communication with org-babel-oz-OPI-socket is asynchronous, but
|
||
|
;; oz-send-string-expression turns is into synchronous...
|
||
|
(defun oz-send-string-expression (string &optional wait-time)
|
||
|
"Similar to oz-send-string, oz-send-string-expression sends a string to the OPI compiler. However, string must be expression and this function returns the result of the expression (as string). oz-send-string-expression is synchronous, wait-time allows to specify a maximum wait time. After wait-time is over with no result, the function returns nil."
|
||
|
(if (not org-babel-oz-OPI-socket)
|
||
|
(org-babel-oz-create-socket))
|
||
|
(let ((polling-delay 0.1)
|
||
|
result)
|
||
|
(process-send-string org-babel-oz-OPI-socket string)
|
||
|
;; wait for result
|
||
|
(if wait-time
|
||
|
(let ((waited 0))
|
||
|
(unwind-protect
|
||
|
(progn
|
||
|
(while
|
||
|
;; stop loop if org-babel-oz-collected-result \= nil or waiting time is over
|
||
|
(not (or (not (equal org-babel-oz-collected-result nil))
|
||
|
(> waited wait-time)))
|
||
|
(progn
|
||
|
(sit-for polling-delay)
|
||
|
;; (message "org-babel-oz: next polling iteration")
|
||
|
(setq waited (+ waited polling-delay))))
|
||
|
;; (message "org-babel-oz: waiting over, got result or waiting timed out")
|
||
|
;; (message (format "wait-time: %s, waited: %s" wait-time waited))
|
||
|
(setq result org-babel-oz-collected-result)
|
||
|
(setq org-babel-oz-collected-result nil))))
|
||
|
(unwind-protect
|
||
|
(progn
|
||
|
(while (equal org-babel-oz-collected-result nil)
|
||
|
(sit-for polling-delay))
|
||
|
(setq result org-babel-oz-collected-result)
|
||
|
(setq org-babel-oz-collected-result nil))))
|
||
|
result))
|
||
|
|
||
|
|
||
|
(defun org-babel-execute:oz (body params)
|
||
|
"Execute a block of Oz code with org-babel. This function is
|
||
|
called by `org-babel-execute-src-block' via multiple-value-bind."
|
||
|
(let* ((processed-params (org-babel-process-params params))
|
||
|
;; (session (org-babel-ruby-initiate-session (first processed-params)))
|
||
|
(vars (second processed-params))
|
||
|
;; (result-params (third processed-params))
|
||
|
(result-type (fourth processed-params))
|
||
|
(full-body (if vars
|
||
|
;; only add var declarations if any variables are there
|
||
|
(concat
|
||
|
;; prepend code to define all arguments passed to the code block
|
||
|
"local\n"
|
||
|
(mapconcat
|
||
|
(lambda (pair)
|
||
|
(format "%s=%s"
|
||
|
(car pair)
|
||
|
(org-babel-oz-var-to-oz (cdr pair))))
|
||
|
vars "\n") "\n"
|
||
|
"in\n"
|
||
|
body
|
||
|
"end\n")
|
||
|
body))
|
||
|
(wait-time (plist-get params :wait-time))
|
||
|
;; set the session if the session variable is non-nil
|
||
|
;; (session-buffer (org-babel-oz-initiate-session session))
|
||
|
;; (session (org-babel-prep-session:oz session params))
|
||
|
)
|
||
|
;; actually execute the source-code block
|
||
|
(case result-type
|
||
|
(output
|
||
|
(progn
|
||
|
(message "Org-babel: executing Oz statement")
|
||
|
(oz-send-string full-body)))
|
||
|
(value
|
||
|
(progn
|
||
|
(message "Org-babel: executing Oz expression")
|
||
|
(oz-send-string-expression full-body (if wait-time
|
||
|
wait-time
|
||
|
1)))))
|
||
|
))
|
||
|
|
||
|
;; This function should be used to assign any variables in params in
|
||
|
;; the context of the session environment.
|
||
|
(defun org-babel-prep-session:oz (session params)
|
||
|
"Prepare SESSION according to the header arguments specified in PARAMS."
|
||
|
(error "org-babel-prep-session:oz unimplemented"))
|
||
|
;; TODO: testing... (copied from org-babel-haskell.el)
|
||
|
;; (defun org-babel-prep-session:oz (session params)
|
||
|
;; "Prepare SESSION according to the header arguments specified in PARAMS."
|
||
|
;; (save-window-excursion
|
||
|
;; (org-babel-oz-initiate-session session)
|
||
|
;; (let* ((vars (org-babel-ref-variables params))
|
||
|
;; (var-lines (mapconcat ;; define any variables
|
||
|
;; (lambda (pair)
|
||
|
;; (format "%s=%s"
|
||
|
;; (car pair)
|
||
|
;; (org-babel-ruby-var-to-ruby (cdr pair))))
|
||
|
;; vars "\n"))
|
||
|
;; (vars-file (concat (make-temp-file "org-babel-oz-vars") ".oz")))
|
||
|
;; (when vars
|
||
|
;; (with-temp-buffer
|
||
|
;; (insert var-lines) (write-file vars-file)
|
||
|
;; (oz-mode)
|
||
|
;; ;; (inferior-oz-load-file) ; ??
|
||
|
;; ))
|
||
|
;; (current-buffer))))
|
||
|
;;
|
||
|
|
||
|
|
||
|
;; TODO: testing... (simplified version of def in org-babel-prep-session:ocaml)
|
||
|
;;
|
||
|
;; BUG: does not work yet. Error: ad-Orig-error: buffer none doesn't exist or has no process
|
||
|
;; UNUSED DEF
|
||
|
(defun org-babel-oz-initiate-session (&optional session)
|
||
|
"If there is not a current inferior-process-buffer in SESSION
|
||
|
then create. Return the initialized session."
|
||
|
(unless (string= session "none")
|
||
|
;; TODO: make it possible to have multiple sessions
|
||
|
(save-window-excursion
|
||
|
;; (run-oz)
|
||
|
(get-buffer oz-compiler-buffer))))
|
||
|
|
||
|
(defun org-babel-oz-var-to-oz (var)
|
||
|
"Convert an elisp var into a string of Oz source code
|
||
|
specifying a var of the same value."
|
||
|
(if (listp var)
|
||
|
;; (concat "[" (mapconcat #'org-babel-oz-var-to-oz var ", ") "]")
|
||
|
(eval var)
|
||
|
(format "%s" var) ; don't preserve string quotes.
|
||
|
;; (format "%s" var)
|
||
|
))
|
||
|
|
||
|
;; TODO:
|
||
|
(defun org-babel-oz-table-or-string (results)
|
||
|
"If the results look like a table, then convert them into an
|
||
|
Emacs-lisp table, otherwise return the results as a string."
|
||
|
(error "org-babel-oz-table-or-string unimplemented"))
|
||
|
|
||
|
|
||
|
(provide 'org-babel-oz)
|
||
|
;;; org-babel-oz.el ends here
|