edit-server.el 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. ;;
  2. ;; Emacs edit-server
  3. ;;
  4. ;; This provides an edit server to respond to requests from the Chrome
  5. ;; Emacs Chrome plugin. This is my first attempt at doing something
  6. ;; with sockets in Emacs. I based it on the following examples:
  7. ;;
  8. ;; http://www.emacswiki.org/emacs/EmacsEchoServer
  9. ;; http://nullprogram.com/blog/2009/05/17/
  10. ;;
  11. ;; To use it ensure the file is in your load-path and add something
  12. ;; like the following examples to your .emacs:
  13. ;;
  14. ;; To open pages for editing in new buffers in your existing Emacs
  15. ;; instance:
  16. ;;
  17. ;; (if (locate-library "edit-server")
  18. ;; (progn
  19. ;; (require 'edit-server)
  20. ;; (setq edit-server-new-frame nil)
  21. ;; (edit-server-start)))
  22. ;;
  23. ;; To open pages for editing in new frames using a running emacs
  24. ;; started in --daemon mode:
  25. ;;
  26. ;; (if (and (daemonp) (locate-library "edit-server"))
  27. ;; (progn
  28. ;; (require 'edit-server)
  29. ;; (edit-server-start)))
  30. ;;
  31. ;; (C) 2009 Alex Bennee (alex@bennee.com)
  32. ;; (C) 2010 Riccardo Murri (riccardo.murri@gmail.com)
  33. ;;
  34. ;; Licensed under GPLv3
  35. ;;
  36. ;; uncomment to debug
  37. ;(setq debug-on-error 't)
  38. ;(setq edebug-all-defs 't)
  39. ;; Customization
  40. (defcustom edit-server-port 9292
  41. "Local port the edit server listens to."
  42. :group 'edit-server
  43. :type 'integer)
  44. (defcustom edit-server-host nil
  45. "If not nil, accept connections from HOST address rather than just
  46. localhost. This may present a security issue."
  47. :group 'edit-server
  48. :type 'boolean)
  49. (defcustom edit-server-verbose nil
  50. "If not nil, log connections and progress also to the echo area."
  51. :group 'edit-server
  52. :type 'boolean)
  53. (defcustom edit-server-done-hook nil
  54. "Hook run when done editing a buffer for the Emacs HTTP edit-server.
  55. Current buffer holds the text that is about to be sent back to the client."
  56. :group 'edit-server
  57. :type 'hook)
  58. ; frame options
  59. (defcustom edit-server-new-frame t
  60. "If not nil, edit each buffer in a new frame (and raise it)."
  61. :group 'edit-server
  62. :type 'boolean)
  63. (defcustom edit-server-create-customized-frame t
  64. "If not nil, customize the new frame as specified by the parameters below.
  65. Otherwise, use the default frame parameters."
  66. :group 'edit-server
  67. :type 'boolean)
  68. (defcustom edit-server-new-frame-minibuffer t
  69. "Show the emacs frame's minibuffer if set to t; hide if nil."
  70. :group 'edit-server
  71. :type 'boolean)
  72. (defcustom edit-server-new-frame-menu-bar t
  73. "Show the emacs frame's menu-bar if set to t; hide if nil."
  74. :group 'edit-server
  75. :type 'boolean)
  76. (defcustom edit-server-new-frame-mode-line t
  77. "Show the emacs frame's mode-line if set to t; hide if nil."
  78. :group 'edit-server
  79. :type 'boolean)
  80. ;; Vars
  81. (defconst edit-server-process-buffer-name " *edit-server*"
  82. "Template name of the edit-server process buffers.")
  83. (defconst edit-server-log-buffer-name "*edit-server-log*"
  84. "Template name of the edit-server process buffers.")
  85. (defconst edit-server-edit-buffer-name "TEXTAREA"
  86. "Template name of the edit-server text editing buffers.")
  87. (defconst edit-server-new-frame-title "Emacs TEXTAREA"
  88. "Template name of the emacs frame's title.")
  89. (defconst edit-server-new-frame-width 80
  90. "The emacs frame's width.")
  91. (defconst edit-server-new-frame-height 25
  92. "The emacs frame's height.")
  93. (defvar edit-server-proc 'nil
  94. "Network process associated with the current edit, made local when
  95. the edit buffer is created")
  96. (defvar edit-server-frame 'nil
  97. "The frame created for a new edit-server process, made local when
  98. then edit buffer is created")
  99. (defvar edit-server-clients '()
  100. "List of all client processes associated with the server process.")
  101. (defvar edit-server-phase nil
  102. "Symbol indicating the state of the HTTP request parsing.")
  103. (defvar edit-server-received nil
  104. "Number of bytes received so far in the client buffer.
  105. Depending on the character encoding, may be different from the buffer length.")
  106. (defvar edit-server-request nil
  107. "The HTTP request (GET, HEAD, POST) received.")
  108. (defvar edit-server-content-length nil
  109. "The value gotten from the HTTP `Content-Length' header.")
  110. (defvar edit-server-url nil
  111. "The value gotten from the HTTP `x-url' header.")
  112. ;; Mode magic
  113. ;
  114. ; We want to re-map some of the keys to trigger edit-server-done
  115. ; instead of the usual emacs like behaviour. However using
  116. ; local-set-key will affect all buffers of the same mode, hence we
  117. ; define a special (derived) mode for handling editing of text areas.
  118. ;
  119. (define-derived-mode edit-server-text-mode text-mode "Edit Server Text Mode"
  120. "A derived version of text-mode with a few common Emacs keystrokes
  121. rebound to more functions that can deal with the response to the
  122. edit-server request.
  123. Any of the following keys will close the buffer and send the text
  124. to the HTTP client: C-x #, C-x C-s, C-c C-c.
  125. If any of the above isused with a prefix argument, the
  126. unmodified text is sent back instead.
  127. "
  128. :group 'edit-server)
  129. (define-key edit-server-text-mode-map (kbd "C-x #") 'edit-server-done)
  130. (define-key edit-server-text-mode-map (kbd "C-x C-s") 'edit-server-done)
  131. (define-key edit-server-text-mode-map (kbd "C-c C-c") 'edit-server-done)
  132. (define-key edit-server-text-mode-map (kbd "C-x C-c") 'edit-server-abort)
  133. ;; Edit Server socket code
  134. ;
  135. (defun edit-server-start (&optional verbose)
  136. "Start the edit server.
  137. If argument VERBOSE is non-nil, logs all server activity to buffer `*edit-server-log*'.
  138. When called interactivity, a prefix argument will cause it to be verbose.
  139. "
  140. (interactive "P")
  141. (if (process-status "edit-server")
  142. (message "An edit-server process is already running")
  143. (make-network-process
  144. :name "edit-server"
  145. :buffer edit-server-process-buffer-name
  146. :family 'ipv4
  147. :host (if edit-server-host
  148. edit-server-host
  149. 'local)
  150. :service edit-server-port
  151. :log 'edit-server-accept
  152. :server 't)
  153. (setq edit-server-clients '())
  154. (if verbose (get-buffer-create edit-server-log-buffer-name))
  155. (edit-server-log nil "Created a new edit-server process")))
  156. (defun edit-server-stop nil
  157. "Stop the edit server"
  158. (interactive)
  159. (while edit-server-clients
  160. (edit-server-kill-client (car edit-server-clients))
  161. (setq edit-server-clients (cdr edit-server-clients)))
  162. (if (process-status "edit-server")
  163. (delete-process "edit-server")
  164. (message "No edit server running"))
  165. (if (get-buffer edit-server-process-buffer-name)
  166. (kill-buffer edit-server-process-buffer-name)))
  167. (defun edit-server-log (proc fmt &rest args)
  168. "If a `*edit-server-log*' buffer exists, write STRING to it for logging purposes.
  169. If `edit-server-verbose' is non-nil, then STRING is also echoed to the message line."
  170. (let ((string (apply 'format fmt args)))
  171. (if edit-server-verbose
  172. (message string))
  173. (if (get-buffer edit-server-log-buffer-name)
  174. (with-current-buffer edit-server-log-buffer-name
  175. (goto-char (point-max))
  176. (insert (current-time-string)
  177. " "
  178. (if (processp proc)
  179. (concat
  180. (buffer-name (process-buffer proc))
  181. ": ")
  182. "") ; nil is not acceptable to 'insert
  183. string)
  184. (or (bolp) (newline))))))
  185. (defun edit-server-accept (server client msg)
  186. "Accept a new client connection."
  187. (let ((buffer (generate-new-buffer edit-server-process-buffer-name)))
  188. (buffer-disable-undo buffer)
  189. (set-process-buffer client buffer)
  190. (set-process-filter client 'edit-server-filter)
  191. (set-process-query-on-exit-flag client nil) ; kill-buffer kills the associated process
  192. (with-current-buffer buffer
  193. (set (make-local-variable 'edit-server-phase) 'wait)
  194. (set (make-local-variable 'edit-server-received) 0)
  195. (set (make-local-variable 'edit-server-request) nil))
  196. (set (make-local-variable 'edit-server-content-length) nil)
  197. (set (make-local-variable 'edit-server-url) nil))
  198. (add-to-list 'edit-server-clients client)
  199. (edit-server-log client msg))
  200. (defun edit-server-filter (proc string)
  201. "Process data received from the client."
  202. ;; there is no guarantee that data belonging to the same client
  203. ;; request will arrive all in one go; therefore, we must accumulate
  204. ;; data in the buffer and process it in different phases, which
  205. ;; requires us to keep track of the processing state.
  206. (with-current-buffer (process-buffer proc)
  207. (insert string)
  208. (setq edit-server-received
  209. (+ edit-server-received (string-bytes string)))
  210. (when (eq edit-server-phase 'wait)
  211. ;; look for a complete HTTP request string
  212. (save-excursion
  213. (goto-char (point-min))
  214. (when (re-search-forward "^\\([A-Z]+\\)\\s-+\\(\\S-+\\)\\s-+\\(HTTP/[0-9\.]+\\)\r?\n" nil t)
  215. (edit-server-log proc
  216. "Got HTTP `%s' request, processing in buffer `%s'..."
  217. (match-string 1) (current-buffer))
  218. (setq edit-server-request (match-string 1))
  219. (setq edit-server-content-length nil)
  220. (setq edit-server-phase 'head))))
  221. (when (eq edit-server-phase 'head)
  222. ;; look for "Content-length" header
  223. (save-excursion
  224. (goto-char (point-min))
  225. (when (re-search-forward "^Content-Length:\\s-+\\([0-9]+\\)" nil t)
  226. (setq edit-server-content-length (string-to-number (match-string 1)))))
  227. ;; look for "x-url" header
  228. (save-excursion
  229. (goto-char (point-min))
  230. (when (re-search-forward "^x-url: .*//\\(.*\\)/" nil t)
  231. (setq edit-server-url (match-string 1))))
  232. ;; look for head/body separator
  233. (save-excursion
  234. (goto-char (point-min))
  235. (when (re-search-forward "\\(\r?\n\\)\\{2\\}" nil t)
  236. ;; HTTP headers are pure ASCII (1 char = 1 byte), so we can subtract
  237. ;; the buffer position from the count of received bytes
  238. (setq edit-server-received
  239. (- edit-server-received (- (match-end 0) (point-min))))
  240. ;; discard headers - keep only HTTP content in buffer
  241. (delete-region (point-min) (match-end 0))
  242. (setq edit-server-phase 'body))))
  243. (when (eq edit-server-phase 'body)
  244. (if (and edit-server-content-length
  245. (> edit-server-content-length edit-server-received))
  246. (edit-server-log proc
  247. "Received %d bytes of %d ..."
  248. edit-server-received edit-server-content-length)
  249. ;; all content transferred - process request now
  250. (cond
  251. ((string= edit-server-request "POST")
  252. ;; create editing buffer, and move content to it
  253. (edit-server-create-edit-buffer proc))
  254. (t
  255. ;; send 200 OK response to any other request
  256. (edit-server-send-response proc "edit-server is running.\n" t)))
  257. ;; wait for another connection to arrive
  258. (setq edit-server-received 0)
  259. (setq edit-server-phase 'wait)))))
  260. (defun edit-server-create-frame(buffer)
  261. "Create a frame for the edit server"
  262. (if edit-server-new-frame
  263. (let* ((property-alist
  264. `((name . ,edit-server-new-frame-title)
  265. (width . ,edit-server-new-frame-width)
  266. (height . ,edit-server-new-frame-height)
  267. (minibuffer . ,edit-server-new-frame-minibuffer)
  268. (menu-bar-lines . ,edit-server-new-frame-menu-bar)))
  269. ; Aquamacs gets confused by make-frame-on-display
  270. (new-frame
  271. (if (featurep 'aquamacs)
  272. (make-frame property-alist)
  273. (make-frame-on-display (getenv "DISPLAY")
  274. (and edit-server-create-customized-frame
  275. property-alist)))))
  276. (if (not edit-server-new-frame-mode-line)
  277. (setq mode-line-format nil))
  278. (raise-frame new-frame)
  279. new-frame)
  280. (pop-to-buffer buffer)
  281. nil))
  282. (defun edit-server-create-edit-buffer(proc)
  283. "Create an edit buffer, place content in it and save the network
  284. process for the final call back"
  285. (let ((buffer (generate-new-buffer (if edit-server-url
  286. edit-server-url
  287. edit-server-edit-buffer-name))))
  288. (copy-to-buffer buffer (point-min) (point-max))
  289. (with-current-buffer buffer
  290. (not-modified)
  291. (edit-server-text-mode)
  292. (add-hook 'kill-buffer-hook 'edit-server-abort* nil t)
  293. (buffer-enable-undo)
  294. (set (make-local-variable 'edit-server-proc) proc)
  295. (set (make-local-variable 'edit-server-frame)
  296. (edit-server-create-frame buffer)))))
  297. (defun edit-server-send-response (proc &optional body close)
  298. "Send an HTTP 200 OK response back to process PROC.
  299. Optional second argument BODY specifies the response content:
  300. - If nil, the HTTP response will have null content.
  301. - If a string, the string is sent as response content.
  302. - Any other value will cause the contents of the current
  303. buffer to be sent.
  304. If optional third argument CLOSE is non-nil, then process PROC
  305. and its buffer are killed with `edit-server-kill-client'."
  306. (interactive)
  307. (if (processp proc)
  308. (let ((response-header (concat
  309. "HTTP/1.0 200 OK\n"
  310. (format "Server: Emacs/%s\n" emacs-version)
  311. "Date: "
  312. (format-time-string
  313. "%a, %d %b %Y %H:%M:%S GMT\n"
  314. (current-time)))))
  315. (process-send-string proc response-header)
  316. (process-send-string proc "\n")
  317. (cond
  318. ((stringp body) (process-send-string proc body))
  319. ((not body) nil)
  320. (t (process-send-region proc (point-min) (point-max))))
  321. (process-send-eof proc)
  322. (if close
  323. (edit-server-kill-client proc))
  324. (edit-server-log proc "Editing done, sent HTTP OK response."))
  325. (message "edit-server-send-response: invalid proc (bug?)")))
  326. (defun edit-server-kill-client (proc)
  327. "Kill client process PROC and remove it from the list."
  328. (let ((procbuf (process-buffer proc)))
  329. (delete-process proc)
  330. (kill-buffer procbuf)
  331. (setq edit-server-clients (delq proc edit-server-clients))))
  332. (defun edit-server-done (&optional abort nokill)
  333. "Finish editing: send HTTP response back, close client and editing buffers.
  334. The current contents of the buffer are sent back to the HTTP
  335. client, unless argument ABORT is non-nil, in which case then the
  336. original text is sent back.
  337. If optional second argument NOKILL is non-nil, then the editing
  338. buffer is not killed.
  339. When called interactively, use prefix arg to abort editing."
  340. (interactive "P")
  341. ;; Do nothing if the connection is closed by the browser (tab killed, etc.)
  342. (unless (eq (process-status edit-server-proc) 'closed)
  343. (let ((buffer (current-buffer))
  344. (proc edit-server-proc)
  345. (procbuf (process-buffer edit-server-proc)))
  346. ;; edit-server-* vars are buffer-local, so they must be used before issuing kill-buffer
  347. (if abort
  348. ;; send back original content
  349. (with-current-buffer procbuf
  350. (run-hooks 'edit-server-done-hook)
  351. (edit-server-send-response proc t))
  352. ;; send back edited content
  353. (save-restriction
  354. (widen)
  355. (buffer-disable-undo)
  356. ;; ensure any format encoding is done (like longlines)
  357. (dolist (format buffer-file-format)
  358. (format-encode-region (point-min) (point-max) format))
  359. ;; send back
  360. (run-hooks 'edit-server-done-hook)
  361. (edit-server-send-response edit-server-proc t)
  362. ;; restore formats (only useful if we keep the buffer)
  363. (dolist (format buffer-file-format)
  364. (format-decode-region (point-min) (point-max) format))
  365. (buffer-enable-undo)))
  366. (if edit-server-frame (delete-frame edit-server-frame))
  367. ;; delete-frame may change the current buffer
  368. (unless nokill (kill-buffer buffer))
  369. (edit-server-kill-client proc))))
  370. (defun edit-server-abort ()
  371. "Discard editing and send the original text back to the browser."
  372. (interactive)
  373. (edit-server-done t))
  374. (defun edit-server-abort* ()
  375. "Discard editing and send the original text back to the browser,
  376. but don't kill the editing buffer."
  377. (interactive)
  378. (edit-server-done t t))
  379. (provide 'edit-server)
  380. ;;; edit-server.el ends here