123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634 |
- ;;; edit-server.el --- server that responds to edit requests from Chrome
- ;; Copyright (C) 2009-2011 Alex Bennee
- ;; Copyright (C) 2010-2011 Riccardo Murri
- ;; Author: Alex Bennee <alex@bennee.com>
- ;; Maintainer: Alex Bennee <alex@bennee.com>
- ;; Version: 1.10
- ;; Homepage: https://github.com/stsquad/emacs_chrome
- ;; This file is not part of GNU Emacs.
- ;; This file 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 file 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 this program. If not, see <http://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;; This provides an edit server to respond to requests from the Chrome
- ;; Emacs Chrome plugin. This is my first attempt at doing something
- ;; with sockets in Emacs. I based it on the following examples:
- ;;
- ;; http://www.emacswiki.org/emacs/EmacsEchoServer
- ;; http://nullprogram.com/blog/2009/05/17/
- ;;
- ;; To use it ensure the file is in your load-path and add something
- ;; like the following examples to your .emacs:
- ;;
- ;; To open pages for editing in new buffers in your existing Emacs
- ;; instance:
- ;;
- ;; (when (require 'edit-server nil t)
- ;; (setq edit-server-new-frame nil)
- ;; (edit-server-start))
- ;;
- ;; To open pages for editing in new frames using a running emacs
- ;; started in --daemon mode:
- ;;
- ;; (when (and (require 'edit-server nil t) (daemonp))
- ;; (edit-server-start))
- ;;
- ;; Buffers are edited in `text-mode' by default; to use a different
- ;; major mode, change `edit-server-default-major-mode' or customize
- ;; `edit-server-url-major-mode-alist' to specify major modes based
- ;; on the remote URL:
- ;;
- ;; (setq edit-server-url-major-mode-alist
- ;; '(("github\\.com" . markdown-mode)))
- ;;
- ;; Alternatively, set the mode in `edit-server-start-hook'. For
- ;; example:
- ;;
- ;; (add-hook 'edit-server-start-hook
- ;; (lambda ()
- ;; (when (string-match "github.com" (buffer-name))
- ;; (markdown-mode))))
- ;;; Code:
- ;; uncomment to debug
- ;; (setq debug-on-error t)
- ;; (setq edebug-all-defs t)
- (unless (featurep 'make-network-process)
- (error "Incompatible version of [X]Emacs - lacks make-network-process"))
- ;;; Customization
- (defcustom edit-server-port 9292
- "Local port the edit server listens to."
- :group 'edit-server
- :type 'integer)
- (defcustom edit-server-host nil
- "If not nil, accept connections from HOST address rather than just
- localhost. This may present a security issue."
- :group 'edit-server
- :type 'boolean)
- (defcustom edit-server-verbose nil
- "If not nil, log connections and progress also to the echo area."
- :group 'edit-server
- :type 'boolean)
- (defcustom edit-server-done-hook nil
- "Hook run when done editing a buffer for the Emacs HTTP edit-server.
- Current buffer holds the text that is about to be sent back to the client."
- :group 'edit-server
- :type 'hook)
- (defcustom edit-server-start-hook nil
- "Hook run when starting a editing buffer. Current buffer is
- the fully prepared editing buffer. Use this hook to enable
- buffer-specific modes or add key bindings."
- :group 'edit-server
- :type 'hook)
- ;; frame options
- (defcustom edit-server-new-frame t
- "If not nil, edit each buffer in a new frame (and raise it)."
- :group 'edit-server
- :type 'boolean)
- (defcustom edit-server-new-frame-alist
- '((name . "Emacs TEXTAREA")
- (width . 80)
- (height . 25)
- (minibuffer . t)
- (menu-bar-lines . t))
- "Frame parameters for new frames. See `default-frame-alist' for examples.
- If nil, the new frame will use the existing `default-frame-alist' values."
- :group 'edit-server
- :type '(repeat (cons :format "%v"
- (symbol :tag "Parameter")
- (sexp :tag "Value"))))
- (defcustom edit-server-default-major-mode
- 'text-mode
- "The default major mode to use in editing buffers, if no other
- mode is selected by `edit-server-url-major-mode-alist'."
- :group 'edit-server
- :type 'function)
- (defcustom edit-server-url-major-mode-alist
- nil
- "A-list of patterns and corresponding functions; when the first
- pattern is encountered which matches `edit-server-url', the
- corresponding function will be called in order to set the desired
- major mode. If no pattern matches,
- `edit-server-default-major-mode' will be used."
- :group 'edit-server
- :type '(alist :key-type (string :tag "Regexp")
- :value-type (function :tag "Major mode function")))
- (defcustom edit-server-new-frame-mode-line t
- "Show the emacs frame's mode-line if set to t; hide if nil."
- :group 'edit-server
- :type 'boolean)
- ;;; Variables
- (defconst edit-server-process-buffer-name " *edit-server*"
- "Template name of the edit-server process buffers.")
- (defconst edit-server-log-buffer-name "*edit-server-log*"
- "Template name of the edit-server process buffers.")
- (defconst edit-server-edit-buffer-name "TEXTAREA"
- "Template name of the edit-server text editing buffers.")
- (defvar edit-server-clients ()
- "List of all client processes associated with the server process.")
- ;; Buffer local variables
- ;;
- ;; These are all required to associate the edit buffer with the
- ;; correct connection to the client and allow for the buffer to be sent
- ;; back when ready. They are `permanent-local` to avoid being reset if
- ;; the buffer changes major modes.
- (defvar edit-server-proc nil
- "Network process associated with the current edit.")
- (make-variable-buffer-local 'edit-server-proc)
- (put 'edit-server-proc 'permanent-local t)
- (defvar edit-server-frame nil
- "The frame created for a new edit-server process.")
- (make-variable-buffer-local 'edit-server-frame)
- (put 'edit-server-frame 'permanent-local t)
- (defvar edit-server-phase nil
- "Symbol indicating the state of the HTTP request parsing.")
- (make-variable-buffer-local 'edit-server-phase)
- (put 'edit-server-phase 'permanent-local t)
- (defvar edit-server-received nil
- "Number of bytes received so far in the client buffer.
- Depending on the character encoding, may be different from the buffer length.")
- (make-variable-buffer-local 'edit-server-received)
- (put 'edit-server-received 'permanent-local t)
- (defvar edit-server-request nil
- "The HTTP request (GET, HEAD, POST) received.")
- (make-variable-buffer-local 'edit-server-request)
- (put 'edit-server-request 'permanent-local t)
- (defvar edit-server-content-length nil
- "The value gotten from the HTTP `Content-Length' header.")
- (make-variable-buffer-local 'edit-server-content-length)
- (put 'edit-server-content-length 'permanent-local t)
- (defvar edit-server-url nil
- "The value gotten from the HTTP `x-url' header.")
- (make-variable-buffer-local 'edit-server-url)
- (put 'edit-server-url 'permanent-local t)
- (defvar edit-server-file nil
- "The value gotten from the HTTP `x-file' header.")
- (make-variable-buffer-local 'edit-server-file)
- (put 'edit-server-file 'permanent-local t)
- ;;; Mode magic
- ;;
- ;; We want to re-map some of the keys to trigger edit-server-done
- ;; instead of the usual emacs like behaviour. However using
- ;; local-set-key will affect all buffers of the same mode, hence we
- ;; define a special (derived) mode for handling editing of text areas.
- (defvar edit-server-edit-mode-map
- (let ((map (make-sparse-keymap)))
- (define-key map (kbd "C-x C-s") 'edit-server-save)
- (define-key map (kbd "C-x #") 'edit-server-done)
- (define-key map (kbd "C-c C-c") 'edit-server-done)
- (define-key map (kbd "C-x C-c") 'edit-server-abort)
- map)
- "Keymap for minor mode `edit-server-edit-mode'.
- Redefine a few common Emacs keystrokes to functions that can
- deal with the response to the edit-server request.
- Any of the following keys will close the buffer and send the text
- to the HTTP client: C-x #, C-c C-c.
- Pressing C-x C-s will save the current state to the kill-ring.
- If any of the above isused with a prefix argument, the
- unmodified text is sent back instead.")
- (define-minor-mode edit-server-edit-mode
- "Minor mode enabled on buffers opened by `edit-server-accept'.
- Its sole purpose is currently to enable
- `edit-server-edit-mode-map', which overrides common keystrokes to
- send a response back to the client."
- :group 'edit-server
- :lighter " EditSrv"
- :init-value nil
- :keymap edit-server-edit-mode-map)
- (defun turn-on-edit-server-edit-mode-if-server ()
- "Turn on `edit-server-edit-mode' if in an edit-server buffer."
- (when edit-server-proc
- (edit-server-edit-mode t)))
- (define-globalized-minor-mode global-edit-server-edit-mode
- edit-server-edit-mode turn-on-edit-server-edit-mode-if-server)
- (global-edit-server-edit-mode t)
- ;; Edit Server socket code
- ;; Avoid an unnecessary prompt about active processes when exiting
- ;; emacs with no active edit-server clients
- ;; https://github.com/stsquad/emacs_chrome/issues/67
- ;;
- ;; According to http://emacswiki.org/emacs/AdviceVsHooks advice is a
- ;; sledgehammer which should be avoided if possible. However, the
- ;; only hooks run by save-buffers-kill-emacs are defined by
- ;; kill-emacs-query-functions and run *after* the check for active
- ;; processes which results in precisely the interactive prompt we want
- ;; to avoid when edit-server has no active clients. So it seems that
- ;; advice is the only solution until save-buffers-kill-emacs offers an
- ;; earlier hook.
- (defadvice save-buffers-kill-emacs
- (before edit-server-stop-before-kill-emacs)
- "Call `edit-server-stop' if there are no active clients, to
- avoid the user being prompted to kill the edit-server process."
- (or edit-server-clients (edit-server-stop)))
- (defun edit-server-start (&optional verbose)
- "Start the edit server.
- If argument VERBOSE is non-nil, logs all server activity to buffer
- `*edit-server-log*'. When called interactivity, a prefix argument
- will cause it to be verbose."
- (interactive "P")
- (if (or (process-status "edit-server")
- (null (condition-case err
- (let ((proc (make-network-process
- :name "edit-server"
- :buffer edit-server-process-buffer-name
- :family 'ipv4
- :host (or edit-server-host 'local)
- :service edit-server-port
- :log 'edit-server-accept
- :server t
- :noquery t)))
- (set-process-coding-system proc 'utf-8 'utf-8)
- proc)
- (file-error nil))))
- (message "An edit-server process is already running")
- (ad-activate 'save-buffers-kill-emacs)
- (setq edit-server-clients '())
- (if verbose (get-buffer-create edit-server-log-buffer-name))
- (edit-server-log nil "Created a new edit-server process")))
- (defun edit-server-stop nil
- "Stop the edit server"
- (interactive)
- (while edit-server-clients
- (edit-server-kill-client (car edit-server-clients))
- (setq edit-server-clients (cdr edit-server-clients)))
- (if (process-status "edit-server")
- (delete-process "edit-server")
- (message "No edit server running"))
- (when (get-buffer edit-server-process-buffer-name)
- (kill-buffer edit-server-process-buffer-name))
- (ad-disable-advice 'save-buffers-kill-emacs
- 'before 'edit-server-stop-before-kill-emacs)
- ;; Disabling advice doesn't take effect until you (re-)activate
- ;; all advice for the function.
- (ad-activate 'save-buffers-kill-emacs))
- (defun edit-server-log (proc fmt &rest args)
- "If a `*edit-server-log*' buffer exists, write STRING to it.
- This is done for logging purposes. If `edit-server-verbose' is
- non-nil, then STRING is also echoed to the message line."
- (let ((string (apply 'format fmt args)))
- (if edit-server-verbose
- (message string))
- (if (get-buffer edit-server-log-buffer-name)
- (with-current-buffer edit-server-log-buffer-name
- (goto-char (point-max))
- (insert (current-time-string)
- " "
- (if (processp proc)
- (concat
- (buffer-name (process-buffer proc))
- ": ")
- "") ; nil is not acceptable to 'insert
- string)
- (or (bolp) (newline))))))
- (defun edit-server-accept (server client msg)
- "Accept a new client connection."
- (let ((buffer (generate-new-buffer edit-server-process-buffer-name)))
- (when (fboundp 'set-buffer-multibyte)
- (set-buffer-multibyte t)) ; djb
- (buffer-disable-undo buffer)
- (set-process-buffer client buffer)
- (set-process-filter client 'edit-server-filter)
- ;; kill-buffer kills the associated process
- (set-process-query-on-exit-flag client nil)
- (with-current-buffer buffer
- (setq edit-server-phase 'wait
- edit-server-received 0
- edit-server-request nil))
- (setq edit-server-content-length nil
- edit-server-url nil
- edit-server-file nil))
- (add-to-list 'edit-server-clients client)
- (edit-server-log client msg))
- (defun edit-server-filter (proc string)
- "Process data received from the client."
- ;; there is no guarantee that data belonging to the same client
- ;; request will arrive all in one go; therefore, we must accumulate
- ;; data in the buffer and process it in different phases, which
- ;; requires us to keep track of the processing state.
- (with-current-buffer (process-buffer proc)
- (insert string)
- (setq edit-server-received
- (+ edit-server-received (string-bytes string)))
- (when (eq edit-server-phase 'wait)
- ;; look for a complete HTTP request string
- (save-excursion
- (goto-char (point-min))
- (when (re-search-forward
- "^\\([A-Z]+\\)\\s-+\\(\\S-+\\)\\s-+\\(HTTP/[0-9\.]+\\)\r?\n"
- nil t)
- (edit-server-log
- proc "Got HTTP `%s' request, processing in buffer `%s'..."
- (match-string 1) (current-buffer))
- (setq edit-server-request (match-string 1))
- (setq edit-server-content-length nil)
- (setq edit-server-phase 'head))))
- (when (eq edit-server-phase 'head)
- ;; look for "Content-length" header
- (save-excursion
- (goto-char (point-min))
- (when (re-search-forward "^Content-Length:\\s-+\\([0-9]+\\)" nil t)
- (setq edit-server-content-length
- (string-to-number (match-string 1)))))
- ;; look for "x-url" header
- (save-excursion
- (goto-char (point-min))
- (when (re-search-forward "^x-url: .*/\\{2,3\\}\\([^?\r\n
]+\\)" nil t)
- (setq edit-server-url (match-string 1))))
- ;; look for "x-file" header
- (save-excursion
- (goto-char (point-min))
- (when (re-search-forward "^x-file: \\([^?\r\n
]+\\)" nil t)
- (edit-server-log proc "Found x-file: %s" (match-string 1))
- (setq edit-server-file (match-string 1))))
- ;; look for head/body separator
- (save-excursion
- (goto-char (point-min))
- (when (re-search-forward "\\(\r?\n\\)\\{2\\}" nil t)
- ;; HTTP headers are pure ASCII (1 char = 1 byte), so we can subtract
- ;; the buffer position from the count of received bytes
- (setq edit-server-received
- (- edit-server-received (- (match-end 0) (point-min))))
- ;; discard headers - keep only HTTP content in buffer
- (delete-region (point-min) (match-end 0))
- (edit-server-log proc
- "Processed headers, length: %s, url: %s, file: %s"
- edit-server-content-length edit-server-url edit-server-file)
- (setq edit-server-phase 'body))))
- (when (eq edit-server-phase 'body)
- (if (and edit-server-content-length
- (> edit-server-content-length edit-server-received))
- (edit-server-log proc
- "Received %d bytes of %d ..."
- edit-server-received edit-server-content-length)
- ;; all content transferred - process request now
- (cond
- ((string= edit-server-request "POST")
- ;; create editing buffer, and move content to it
- (edit-server-find-or-create-edit-buffer proc edit-server-file))
- (t
- ;; send 200 OK response to any other request
- (edit-server-send-response proc "edit-server is running.\n")
- (edit-server-kill-client proc)))
- ;; wait for another connection to arrive
- (setq edit-server-received 0)
- (setq edit-server-phase 'wait)))))
- (defun edit-server-show-edit-buffer (buffer)
- "Show edit buffer by creating a frame or raising the selected
- frame."
- (if edit-server-new-frame
- (let ((new-frame
- (if (memq window-system '(ns mac))
- ;; Aquamacs, Emacs NS, Emacs (experimental) Mac port
- (make-frame edit-server-new-frame-alist)
- (make-frame-on-display (getenv "DISPLAY")
- edit-server-new-frame-alist))))
- (unless edit-server-new-frame-mode-line
- (setq mode-line-format nil))
- (select-frame new-frame)
- (when (and (eq window-system 'x)
- (fboundp 'x-send-client-message))
- (x-send-client-message nil 0 nil
- "_NET_ACTIVE_WINDOW" 32
- '(1 0 0)))
- (raise-frame new-frame)
- (set-window-buffer (frame-selected-window new-frame) buffer)
- new-frame)
- (select-frame-set-input-focus (window-frame (selected-window)))
- (pop-to-buffer buffer)
- (raise-frame)
- nil))
- (defun edit-server-choose-major-mode ()
- "Use `edit-server-url-major-mode-alist' to choose a major mode
- initialization function based on `edit-server-url', or fall back
- to `edit-server-default-major-mode'"
- (let ((pairs edit-server-url-major-mode-alist)
- (mode edit-server-default-major-mode))
- (while pairs
- (let ((entry (car pairs)))
- (if (string-match (car entry) edit-server-url)
- (setq mode (cdr entry)
- pairs nil)
- (setq pairs (cdr pairs)))))
- (funcall mode)))
- (defun edit-server-find-or-create-edit-buffer(proc &optional existing)
- "Find and existing or create an new edit buffer, place content in it
- and save the network process for the final call back"
- (let* ((existing-buffer (get-buffer (or (and (stringp existing) existing) "")))
- (buffer (or existing-buffer (generate-new-buffer
- (or edit-server-url
- edit-server-edit-buffer-name)))))
- (edit-server-log proc (format
- "using buffer %s for edit (existing-buffer
- is %s)"
- buffer existing-buffer))
- ;; set multi-byte for proper UTF-8 handling (djb)
- (when (fboundp 'set-buffer-multibyte)
- (with-current-buffer buffer
- (set-buffer-multibyte t)))
- ;; I seem to be working around a bug here :-/
- ;;
- ;; For some reason the copy-to-buffer doesn't blat the existing contents.
- ;; This screws up formatting as the contents where decoded before being
- ;; sent back to the browser. As a kludge I save the returned contents
- ;; in the kill-ring.
- (when existing-buffer
- (kill-ring-save (point-min) (point-max)))
- (edit-server-log proc "copying new data into buffer")
- (copy-to-buffer buffer (point-min) (point-max))
-
- (with-current-buffer buffer
- (setq edit-server-url (with-current-buffer (process-buffer proc) edit-server-url))
- (edit-server-choose-major-mode)
- ;; Allow `edit-server-start-hook' to override the major mode.
- ;; (re)setting the minor mode seems to clear the buffer-local
- ;; variables that we depend upon for the response, so call the
- ;; hooks early on
- (run-hooks 'edit-server-start-hook)
- (set-buffer-modified-p 'nil)
- (add-hook 'kill-buffer-hook 'edit-server-abort* nil t)
- (buffer-enable-undo)
- (setq edit-server-proc proc
- edit-server-frame (edit-server-show-edit-buffer buffer))
- (edit-server-edit-mode))))
- (defun edit-server-send-response (proc &optional body progress)
- "Send an HTTP 200 OK response back to process PROC.
- Optional second argument BODY specifies the response content:
- - If nil, the HTTP response will have null content.
- - If a string, the string is sent as response content.
- - Any other value will cause the contents of the current
- buffer to be sent.
- If optional third argument progress is non-nil, then the response
- will include x-file and x-open headers to allow continuation of editing."
- (interactive)
- (edit-server-log proc "sending edit-server response")
- (if (processp proc)
- (let ((response-header (concat
- "HTTP/1.0 200 OK\n"
- (format "Server: Emacs/%s\n" emacs-version)
- "Date: "
- (format-time-string
- "%a, %d %b %Y %H:%M:%S GMT\n"
- (current-time))
- (when progress
- (format "x-file: %s\nx-open: true\n" (buffer-name))))))
- (process-send-string proc response-header)
- (process-send-string proc "\n")
- (cond
- ((stringp body)
- (process-send-string proc (encode-coding-string body 'utf-8)))
- ((not body) nil)
- (t
- (encode-coding-region (point-min) (point-max) 'utf-8)
- (process-send-region proc (point-min) (point-max))))
- (process-send-eof proc)
- (edit-server-log proc "Editing done, sent HTTP OK response."))
- (message "edit-server-send-response: invalid proc (bug?)")))
- (defun edit-server-kill-client (proc)
- "Kill client process PROC and remove it from the list."
- (let ((procbuf (process-buffer proc)))
- (delete-process proc)
- (when (buffer-live-p procbuf)
- (kill-buffer procbuf))
- (setq edit-server-clients (delq proc edit-server-clients))))
- (defun edit-server-done (&optional abort nokill)
- "Finish editing: send HTTP response back, close client and editing buffers.
- The current contents of the buffer are sent back to the HTTP
- client, unless argument ABORT is non-nil, in which case then the
- original text is sent back.
- If optional second argument NOKILL is non-nil, then the editing
- buffer is not killed and the buffer name is passed to calling process.
- When called interactively, use prefix arg to abort editing."
- (interactive "P")
- ;; Do nothing if the connection is closed by the browser (tab killed, etc.)
- (unless (eq (process-status edit-server-proc) 'closed)
- (let ((buffer (current-buffer))
- (proc edit-server-proc)
- (procbuf (process-buffer edit-server-proc)))
- ;; edit-server-* vars are buffer-local,
- ;; so they must be used before issuing kill-buffer
- (if abort
- ;; send back original content
- (with-current-buffer procbuf
- (run-hooks 'edit-server-done-hook)
- (edit-server-send-response proc t))
- ;; send back edited content
- (save-restriction
- (widen)
- (buffer-disable-undo)
- ;; ensure any format encoding is done (like longlines)
- (dolist (format buffer-file-format)
- (format-encode-region (point-min) (point-max) format))
- ;; send back
- (run-hooks 'edit-server-done-hook)
- (edit-server-send-response proc t nokill)
- (edit-server-log proc "sent response to browser")))
- (when edit-server-frame
- (delete-frame edit-server-frame))
- ;; delete-frame may change the current buffer
- (unless nokill
- (kill-buffer buffer))
- (edit-server-kill-client proc))))
- ;; edit-server-save uses the iterative edit-server option (send a
- ;; buffer back to the client which then returns new request to
- ;; continue the session). The edit-server then switches back to the
- ;; buffer referenced by the x-file header.
- ;;
- (defun edit-server-save ()
- "Save the current state of the edit buffer but don't close it."
- (interactive)
- (edit-server-done nil t))
- (defun edit-server-abort ()
- "Discard editing and send the original text back to the browser."
- (interactive)
- (edit-server-done t))
- (defun edit-server-abort* ()
- "Discard editing and send the original text back to the browser,
- but don't kill the editing buffer."
- (interactive)
- (edit-server-done t t))
- (provide 'edit-server)
- ;;; edit-server.el ends here
|