Permalink
Fetching contributors…
Cannot retrieve contributors at this time
671 lines (586 sloc) 24.8 KB
;;; edit-server.el --- server that responds to edit requests from Chrome
;; Copyright (C) 2009-2013 Alex Bennée
;; Copyright (C) 2010-2011 Riccardo Murri
;; Author: Alex Bennée <alex@bennee.com>
;; Maintainer: Alex Bennée <alex@bennee.com>
;; Version: 1.14
;; 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)
(defcustom edit-server-edit-mode-hook nil
"Hook run when we enter edit-server-edit-mode. This is the final step of
an edit session and is called when all frames and displays have been
set-up. You should not set any additional major modes here though as they
may conflict with the edit-server-edit-mode, use the
edit-server-start-hook instead."
: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 . "Edit with Emacs FRAME")
(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*"
"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-request-url nil
"The HTTP request URL received.")
(make-variable-buffer-local 'edit-server-request-url)
(put 'edit-server-request-url '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
(when (and
(numberp arg)
(> arg 0))
(run-hooks 'edit-server-edit-mode-hook)))
(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)))
;;;###autoload
(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 (process-status "edit-server")
(message "An edit-server process is already running")
(if (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 "Failed to start an edit-server")
(ad-activate 'save-buffers-kill-emacs)
(setq edit-server-clients '())
(when 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)))
(when edit-server-verbose
(message string))
(when (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)))
(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
edit-server-request-url 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)
(setq edit-server-request (match-string 1)
edit-server-request-url (match-string 2)
edit-server-content-length nil
edit-server-phase 'head)
(edit-server-log
proc "Got HTTP `%s' request of url `%s', processing in buffer `%s'..."
edit-server-request edit-server-request-url (current-buffer)))))
(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-match "foreground" edit-server-request-url)
(edit-server-foreground-request (current-buffer))
(edit-server-send-response proc "edit-server received foreground request.\n")
(edit-server-kill-client proc))
((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-foreground-request (buffer)
"Bring Emacs into the foreground after a request from Chrome.
`buffer' is the process buffer which contains any potential contents
to be passed into the kill ring.
The new frame will have a specific frame parameter of
`edit-server-forground-frame' set to 't"
(when (bufferp buffer)
(with-current-buffer buffer
(kill-ring-save (point-min) (point-max))))
(when edit-server-new-frame
(set-frame-parameter
(make-frame-on-display (getenv "DISPLAY")
edit-server-new-frame-alist)
'edit-server-forground-frame 't)))
(defun edit-server-show-edit-buffer (buffer)
"Show edit `BUFFER' by creating a frame or raising the selected
frame. If a frame was created it returns `FRAME'."
(let ((edit-frame nil))
(when edit-server-new-frame
(setq edit-frame
(if (memq window-system '(ns mac nil))
;; Aquamacs, Emacs NS, Emacs (experimental) Mac port, termcap.
;; matching (nil) avoids use of DISPLAY from TTY environments.
(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 edit-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))))
(pop-to-buffer buffer)
(raise-frame edit-frame)
(select-frame-set-input-focus (window-frame (selected-window)))
edit-frame))
(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'"
(funcall (or (assoc-default edit-server-url
edit-server-url-major-mode-alist 'string-match)
edit-server-default-major-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"
;; FIXME: `existing' is useless: see issue #104.
(let* ((existing-buffer (and (stringp existing) (get-buffer existing)))
(buffer (or existing-buffer (generate-new-buffer
(or edit-server-url
edit-server-edit-buffer-name)))))
(edit-server-log proc
"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 were 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
;; don't run abort twice in a row.
(remove-hook 'kill-buffer-hook 'edit-server-abort*)
(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