123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- ;;
- ;; Emacs edit-server
- ;;
- ;; 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/
- ;;
- ;; (C) 2009 Alex Bennee (alex@bennee.com)
- ;; (C) 2010 Riccardo Murri (riccardo.murri@gmail.com)
- ;;
- ;; Licensed under GPLv3
- ;;
- ; still debugging
- (setq debug-on-error 't)
- (setq edebug-all-defs 't)
- ;; Customization
- (defcustom edit-server-port 9292
- "Local port the edit server listens to."
- :group 'edit-server
- :type 'integer)
- (defcustom edit-server-new-frame t
- "If not nil, edit each buffer in a new frame (and raise it)."
- :group 'edit-server
- :type 'boolean)
- ;; Vars
- (defvar edit-server-proc 'nil
- "Network process associated with the current edit, made local when
- the edit buffer is created")
- (defvar edit-server-frame 'nil
- "The frame created for a new edit-server process, made local when
- then edit buffer is created")
- (defvar edit-server-clients '()
- "List of all client processes associated with the server process.")
- (defconst edit-server-buffer-name " *edit-server*"
- "Template name of the edit-server process buffers")
- ;; 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-text-mode-map
- (let ((map (make-sparse-keymap)))
- (define-key map (kbd "C-x #") 'edit-server-done)
- ;; XXX: should rather invoke 'edit-server-abort when the buffer is deleted,
- ;; and send the original content back to the HTTP client
- (define-key map (kbd "C-x k") 'edit-server-abort)
- (define-key map (kbd "C-x C-s") 'edit-server-done)
- map)
- "Keymap for `edit-server-text-mode'.")
- ;; XXX: make it a minor mode? server-mode is minor
- (define-derived-mode edit-server-text-mode text-mode "Edit Server Text Mode"
- "A derived version of text-mode with a few common Emacs keystrokes
- rebound to more functions that can deal with the response to the
- edit-server request")
- ;; Edit Server socket code
- ;
- (defun edit-server-start nil
- "Start the edit server"
- (interactive)
- (if (process-status "edit-server")
- (message "An edit-server process is already running")
- (make-network-process
- :name "edit-server"
- :buffer edit-server-buffer-name
- :family 'ipv4
- :host 'local ; only listen to local connections
- :service edit-server-port
- :log 'edit-server-accept
- :server 't)
- (setq edit-server-clients '())
- (message "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"))
- (if (get-buffer edit-server-buffer-name)
- (kill-buffer edit-server-buffer-name)))
- ; Write log entries
- (defun edit-server-log (string &optional client)
- "If a `*edit-server-log*' buffer exists, write STRING to it for logging purposes."
- (if (get-buffer " *edit-server-log*")
- (with-current-buffer " *edit-server-log*"
- (goto-char (point-max))
- (insert (current-time-string)
- (if client (format " %s:" client) " ")
- string)
- (or (bolp) (newline)))))
- (defun edit-server-accept (server client msg)
- "Accept a new client connection."
- (let ((buffer (generate-new-buffer edit-server-buffer-name)))
- (buffer-disable-undo buffer)
- (set-process-buffer client buffer)
- (set-process-filter client 'edit-server-filter)
- (set-process-query-on-exit-flag client nil) ; kill-buffer kills the associated process
- (with-current-buffer buffer
- (set (make-local-variable 'edit-server-phase) 'wait)
- (set (make-local-variable 'edit-server-received) 0)
- (set (make-local-variable 'edit-server-request) nil))
- (set (make-local-variable 'edit-server-content-length) nil))
- (add-to-list 'edit-server-clients client))
- (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)
- (message "edit-server: 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 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))
- (setq edit-server-phase 'body))))
-
- (when (eq edit-server-phase 'body)
- (if (and edit-server-content-length
- (> edit-server-content-length edit-server-received))
- (message "edit-server: Received %d bytes of %d ..."
- edit-server-received edit-server-content-length)
- ;; all content trasnferred - process request now
- (cond
- ((string= edit-server-request "POST")
- ;; create editing buffer, and move content to it
- (edit-server-create-edit-buffer proc))
- (t
- ;; send 200 OK response to any other request
- (edit-server-send-response proc "edit-server is running.\n" t)))
- ;; wait for another connection to arrive
- (setq edit-server-received 0)
- (setq edit-server-phase 'wait)))))
- (defun edit-server-create-edit-buffer(proc)
- "Create an edit buffer, place content in it and save the network
- process for the final call back"
- (let ((buffer (generate-new-buffer "TEXTAREA")))
- (copy-to-buffer buffer (point-min) (point-max))
- (with-current-buffer buffer
- (edit-server-text-mode)
- (buffer-enable-undo)
- (set (make-local-variable 'edit-server-proc) proc)
- (set (make-local-variable 'edit-server-frame)
- (if edit-server-new-frame (make-frame) nil))
- (if edit-server-new-frame
- (raise-frame edit-server-frame)
- (pop-to-buffer buffer)))))
- (defun edit-server-send-response (proc &optional string close)
- "Send an HTTP 200 OK response back to process PROC.
- Optional second argument STRING specifies the response content.
- If optional third argument CLOSE is non-nil, then process PROC
- and its buffer are killed with `edit-server-kill-client'."
- (interactive)
- (message "edit-server-send-response")
- (if proc
- (let ((response-header (concat
- "HTTP/1.0 200 OK\n"
- "Server: Emacs\n"
- "Date: "
- (format-time-string
- "%a, %d %b %Y %H:%M:%S GMT\n"
- (current-time)))))
- (process-send-string proc response-header)
- (process-send-string proc "\n")
- (if string
- (process-send-string proc string))
- (process-send-eof proc)
- (if close
- (edit-server-kill-client proc)))
- (message "edit-server-send-response: null 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)
- (kill-buffer procbuf)
- (setq edit-server-clients (delq procbuf edit-server-clients))))
- (defun edit-server-done ()
- "Once someone is done with editing their text edit-server-done is
- called and the response is sent back to the browser"
- (interactive)
- (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
- (edit-server-send-response edit-server-proc (buffer-string))
- (if edit-server-frame (delete-frame edit-server-frame))
- ;; delete-frame may change the current buffer
- (kill-buffer buffer)
- (edit-server-kill-client proc)))
- (defun edit-server-abort ()
- "Send the original text back to the browser."
- (interactive)
- (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
- (with-current-buffer procbuf
- (edit-server-send-response proc (buffer-string)))
- (if edit-server-frame (delete-frame edit-server-frame))
- ;; delete-frame may change the current buffer
- (kill-buffer buffer)
- (edit-server-kill-client proc)))
|