edit-server.el 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. ;;; edit-server.el --- server that responds to edit requests from Chrome
  2. ;; Copyright (C) 2009-2011 Alex Bennee
  3. ;; Copyright (C) 2010-2011 Riccardo Murri
  4. ;; Author: Alex Bennee <alex@bennee.com>
  5. ;; Maintainer: Alex Bennee <alex@bennee.com>
  6. ;; Version: 1.10
  7. ;; Homepage: https://github.com/stsquad/emacs_chrome
  8. ;; This file is not part of GNU Emacs.
  9. ;; This file is free software; you can redistribute it and/or modify
  10. ;; it under the terms of the GNU General Public License as published by
  11. ;; the Free Software Foundation; either version 3, or (at your option)
  12. ;; any later version.
  13. ;; This file is distributed in the hope that it will be useful,
  14. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. ;; GNU General Public License for more details.
  17. ;; You should have received a copy of the GNU General Public License
  18. ;; along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. ;;; Commentary:
  20. ;; This provides an edit server to respond to requests from the Chrome
  21. ;; Emacs Chrome plugin. This is my first attempt at doing something
  22. ;; with sockets in Emacs. I based it on the following examples:
  23. ;;
  24. ;; http://www.emacswiki.org/emacs/EmacsEchoServer
  25. ;; http://nullprogram.com/blog/2009/05/17/
  26. ;;
  27. ;; To use it ensure the file is in your load-path and add something
  28. ;; like the following examples to your .emacs:
  29. ;;
  30. ;; To open pages for editing in new buffers in your existing Emacs
  31. ;; instance:
  32. ;;
  33. ;; (when (require 'edit-server nil t)
  34. ;; (setq edit-server-new-frame nil)
  35. ;; (edit-server-start))
  36. ;;
  37. ;; To open pages for editing in new frames using a running emacs
  38. ;; started in --daemon mode:
  39. ;;
  40. ;; (when (and (require 'edit-server nil t) (daemonp))
  41. ;; (edit-server-start))
  42. ;;
  43. ;; Buffers are edited in `text-mode' by default; to use a different
  44. ;; major mode, change `edit-server-default-major-mode' or customize
  45. ;; `edit-server-url-major-mode-alist' to specify major modes based
  46. ;; on the remote URL:
  47. ;;
  48. ;; (setq edit-server-url-major-mode-alist
  49. ;; '(("github\\.com" . markdown-mode)))
  50. ;;
  51. ;; Alternatively, set the mode in `edit-server-start-hook'. For
  52. ;; example:
  53. ;;
  54. ;; (add-hook 'edit-server-start-hook
  55. ;; (lambda ()
  56. ;; (when (string-match "github.com" (buffer-name))
  57. ;; (markdown-mode))))
  58. ;;; Code:
  59. ;; uncomment to debug
  60. ;; (setq debug-on-error t)
  61. ;; (setq edebug-all-defs t)
  62. (unless (featurep 'make-network-process)
  63. (error "Incompatible version of [X]Emacs - lacks make-network-process"))
  64. ;;; Customization
  65. (defcustom edit-server-port 9292
  66. "Local port the edit server listens to."
  67. :group 'edit-server
  68. :type 'integer)
  69. (defcustom edit-server-host nil
  70. "If not nil, accept connections from HOST address rather than just
  71. localhost. This may present a security issue."
  72. :group 'edit-server
  73. :type 'boolean)
  74. (defcustom edit-server-verbose nil
  75. "If not nil, log connections and progress also to the echo area."
  76. :group 'edit-server
  77. :type 'boolean)
  78. (defcustom edit-server-done-hook nil
  79. "Hook run when done editing a buffer for the Emacs HTTP edit-server.
  80. Current buffer holds the text that is about to be sent back to the client."
  81. :group 'edit-server
  82. :type 'hook)
  83. (defcustom edit-server-start-hook nil
  84. "Hook run when starting a editing buffer. Current buffer is
  85. the fully prepared editing buffer. Use this hook to enable
  86. buffer-specific modes or add key bindings."
  87. :group 'edit-server
  88. :type 'hook)
  89. ;; frame options
  90. (defcustom edit-server-new-frame t
  91. "If not nil, edit each buffer in a new frame (and raise it)."
  92. :group 'edit-server
  93. :type 'boolean)
  94. (defcustom edit-server-new-frame-alist
  95. '((name . "Emacs TEXTAREA")
  96. (width . 80)
  97. (height . 25)
  98. (minibuffer . t)
  99. (menu-bar-lines . t))
  100. "Frame parameters for new frames. See `default-frame-alist' for examples.
  101. If nil, the new frame will use the existing `default-frame-alist' values."
  102. :group 'edit-server
  103. :type '(repeat (cons :format "%v"
  104. (symbol :tag "Parameter")
  105. (sexp :tag "Value"))))
  106. (defcustom edit-server-default-major-mode
  107. 'text-mode
  108. "The default major mode to use in editing buffers, if no other
  109. mode is selected by `edit-server-url-major-mode-alist'."
  110. :group 'edit-server
  111. :type 'function)
  112. (defcustom edit-server-url-major-mode-alist
  113. nil
  114. "A-list of patterns and corresponding functions; when the first
  115. pattern is encountered which matches `edit-server-url', the
  116. corresponding function will be called in order to set the desired
  117. major mode. If no pattern matches,
  118. `edit-server-default-major-mode' will be used."
  119. :group 'edit-server
  120. :type '(alist :key-type (string :tag "Regexp")
  121. :value-type (function :tag "Major mode function")))
  122. (defcustom edit-server-new-frame-mode-line t
  123. "Show the emacs frame's mode-line if set to t; hide if nil."
  124. :group 'edit-server
  125. :type 'boolean)
  126. ;;; Variables
  127. (defconst edit-server-process-buffer-name " *edit-server*"
  128. "Template name of the edit-server process buffers.")
  129. (defconst edit-server-log-buffer-name "*edit-server-log*"
  130. "Template name of the edit-server process buffers.")
  131. (defconst edit-server-edit-buffer-name "TEXTAREA"
  132. "Template name of the edit-server text editing buffers.")
  133. (defvar edit-server-clients ()
  134. "List of all client processes associated with the server process.")
  135. ;; Buffer local variables
  136. ;;
  137. ;; These are all required to associate the edit buffer with the
  138. ;; correct connection to the client and allow for the buffer to be sent
  139. ;; back when ready. They are `permanent-local` to avoid being reset if
  140. ;; the buffer changes major modes.
  141. (defvar edit-server-proc nil
  142. "Network process associated with the current edit.")
  143. (make-variable-buffer-local 'edit-server-proc)
  144. (put 'edit-server-proc 'permanent-local t)
  145. (defvar edit-server-frame nil
  146. "The frame created for a new edit-server process.")
  147. (make-variable-buffer-local 'edit-server-frame)
  148. (put 'edit-server-frame 'permanent-local t)
  149. (defvar edit-server-phase nil
  150. "Symbol indicating the state of the HTTP request parsing.")
  151. (make-variable-buffer-local 'edit-server-phase)
  152. (put 'edit-server-phase 'permanent-local t)
  153. (defvar edit-server-received nil
  154. "Number of bytes received so far in the client buffer.
  155. Depending on the character encoding, may be different from the buffer length.")
  156. (make-variable-buffer-local 'edit-server-received)
  157. (put 'edit-server-received 'permanent-local t)
  158. (defvar edit-server-request nil
  159. "The HTTP request (GET, HEAD, POST) received.")
  160. (make-variable-buffer-local 'edit-server-request)
  161. (put 'edit-server-request 'permanent-local t)
  162. (defvar edit-server-content-length nil
  163. "The value gotten from the HTTP `Content-Length' header.")
  164. (make-variable-buffer-local 'edit-server-content-length)
  165. (put 'edit-server-content-length 'permanent-local t)
  166. (defvar edit-server-url nil
  167. "The value gotten from the HTTP `x-url' header.")
  168. (make-variable-buffer-local 'edit-server-url)
  169. (put 'edit-server-url 'permanent-local t)
  170. ;;; Mode magic
  171. ;;
  172. ;; We want to re-map some of the keys to trigger edit-server-done
  173. ;; instead of the usual emacs like behaviour. However using
  174. ;; local-set-key will affect all buffers of the same mode, hence we
  175. ;; define a special (derived) mode for handling editing of text areas.
  176. (defvar edit-server-edit-mode-map
  177. (let ((map (make-sparse-keymap)))
  178. (define-key map (kbd "C-x C-s") 'edit-server-save)
  179. (define-key map (kbd "C-x #") 'edit-server-done)
  180. (define-key map (kbd "C-c C-c") 'edit-server-done)
  181. (define-key map (kbd "C-x C-c") 'edit-server-abort)
  182. map)
  183. "Keymap for minor mode `edit-server-edit-mode'.
  184. Redefine a few common Emacs keystrokes to functions that can
  185. deal with the response to the edit-server request.
  186. Any of the following keys will close the buffer and send the text
  187. to the HTTP client: C-x #, C-c C-c.
  188. Pressing C-x C-s will save the current state to the kill-ring.
  189. If any of the above isused with a prefix argument, the
  190. unmodified text is sent back instead.")
  191. (define-minor-mode edit-server-edit-mode
  192. "Minor mode enabled on buffers opened by `edit-server-accept'.
  193. Its sole purpose is currently to enable
  194. `edit-server-edit-mode-map', which overrides common keystrokes to
  195. send a response back to the client."
  196. :group 'edit-server
  197. :lighter " EditSrv"
  198. :init-value nil
  199. :keymap edit-server-edit-mode-map)
  200. (defun turn-on-edit-server-edit-mode-if-server ()
  201. "Turn on `edit-server-edit-mode' if in an edit-server buffer."
  202. (when edit-server-proc
  203. (edit-server-edit-mode t)))
  204. (define-globalized-minor-mode global-edit-server-edit-mode
  205. edit-server-edit-mode turn-on-edit-server-edit-mode-if-server)
  206. (global-edit-server-edit-mode t)
  207. ;; Edit Server socket code
  208. (defun edit-server-start (&optional verbose)
  209. "Start the edit server.
  210. If argument VERBOSE is non-nil, logs all server activity to buffer
  211. `*edit-server-log*'. When called interactivity, a prefix argument
  212. will cause it to be verbose."
  213. (interactive "P")
  214. (if (or (process-status "edit-server")
  215. (null (condition-case err
  216. (make-network-process
  217. :name "edit-server"
  218. :buffer edit-server-process-buffer-name
  219. :family 'ipv4
  220. :host (or edit-server-host 'local)
  221. :service edit-server-port
  222. :log 'edit-server-accept
  223. :server t
  224. :noquery t)
  225. (file-error nil))))
  226. (message "An edit-server process is already running")
  227. (setq edit-server-clients '())
  228. (if verbose (get-buffer-create edit-server-log-buffer-name))
  229. (edit-server-log nil "Created a new edit-server process")))
  230. (defun edit-server-stop nil
  231. "Stop the edit server"
  232. (interactive)
  233. (while edit-server-clients
  234. (edit-server-kill-client (car edit-server-clients))
  235. (setq edit-server-clients (cdr edit-server-clients)))
  236. (if (process-status "edit-server")
  237. (delete-process "edit-server")
  238. (message "No edit server running"))
  239. (when (get-buffer edit-server-process-buffer-name)
  240. (kill-buffer edit-server-process-buffer-name)))
  241. (defun edit-server-log (proc fmt &rest args)
  242. "If a `*edit-server-log*' buffer exists, write STRING to it.
  243. This is done for logging purposes. If `edit-server-verbose' is
  244. non-nil, then STRING is also echoed to the message line."
  245. (let ((string (apply 'format fmt args)))
  246. (if edit-server-verbose
  247. (message string))
  248. (if (get-buffer edit-server-log-buffer-name)
  249. (with-current-buffer edit-server-log-buffer-name
  250. (goto-char (point-max))
  251. (insert (current-time-string)
  252. " "
  253. (if (processp proc)
  254. (concat
  255. (buffer-name (process-buffer proc))
  256. ": ")
  257. "") ; nil is not acceptable to 'insert
  258. string)
  259. (or (bolp) (newline))))))
  260. (defun edit-server-accept (server client msg)
  261. "Accept a new client connection."
  262. (let ((buffer (generate-new-buffer edit-server-process-buffer-name)))
  263. (when (fboundp 'set-buffer-multibyte)
  264. (set-buffer-multibyte t)) ; djb
  265. (buffer-disable-undo buffer)
  266. (set-process-buffer client buffer)
  267. (set-process-filter client 'edit-server-filter)
  268. ;; kill-buffer kills the associated process
  269. (set-process-query-on-exit-flag client nil)
  270. (with-current-buffer buffer
  271. (setq edit-server-phase 'wait
  272. edit-server-received 0
  273. edit-server-request nil))
  274. (setq edit-server-content-length nil
  275. edit-server-url nil))
  276. (add-to-list 'edit-server-clients client)
  277. (edit-server-log client msg))
  278. (defun edit-server-filter (proc string)
  279. "Process data received from the client."
  280. ;; there is no guarantee that data belonging to the same client
  281. ;; request will arrive all in one go; therefore, we must accumulate
  282. ;; data in the buffer and process it in different phases, which
  283. ;; requires us to keep track of the processing state.
  284. (with-current-buffer (process-buffer proc)
  285. (insert string)
  286. (setq edit-server-received
  287. (+ edit-server-received (string-bytes string)))
  288. (when (eq edit-server-phase 'wait)
  289. ;; look for a complete HTTP request string
  290. (save-excursion
  291. (goto-char (point-min))
  292. (when (re-search-forward
  293. "^\\([A-Z]+\\)\\s-+\\(\\S-+\\)\\s-+\\(HTTP/[0-9\.]+\\)\r?\n"
  294. nil t)
  295. (edit-server-log
  296. proc "Got HTTP `%s' request, processing in buffer `%s'..."
  297. (match-string 1) (current-buffer))
  298. (setq edit-server-request (match-string 1))
  299. (setq edit-server-content-length nil)
  300. (setq edit-server-phase 'head))))
  301. (when (eq edit-server-phase 'head)
  302. ;; look for "Content-length" header
  303. (save-excursion
  304. (goto-char (point-min))
  305. (when (re-search-forward "^Content-Length:\\s-+\\([0-9]+\\)" nil t)
  306. (setq edit-server-content-length
  307. (string-to-number (match-string 1)))))
  308. ;; look for "x-url" header
  309. (save-excursion
  310. (goto-char (point-min))
  311. (when (re-search-forward "^x-url: .*//\\(.*\\)/" nil t)
  312. (setq edit-server-url (match-string 1))))
  313. ;; look for head/body separator
  314. (save-excursion
  315. (goto-char (point-min))
  316. (when (re-search-forward "\\(\r?\n\\)\\{2\\}" nil t)
  317. ;; HTTP headers are pure ASCII (1 char = 1 byte), so we can subtract
  318. ;; the buffer position from the count of received bytes
  319. (setq edit-server-received
  320. (- edit-server-received (- (match-end 0) (point-min))))
  321. ;; discard headers - keep only HTTP content in buffer
  322. (delete-region (point-min) (match-end 0))
  323. (setq edit-server-phase 'body))))
  324. (when (eq edit-server-phase 'body)
  325. (if (and edit-server-content-length
  326. (> edit-server-content-length edit-server-received))
  327. (edit-server-log proc
  328. "Received %d bytes of %d ..."
  329. edit-server-received edit-server-content-length)
  330. ;; all content transferred - process request now
  331. (cond
  332. ((string= edit-server-request "POST")
  333. ;; create editing buffer, and move content to it
  334. (edit-server-create-edit-buffer proc))
  335. (t
  336. ;; send 200 OK response to any other request
  337. (edit-server-send-response proc "edit-server is running.\n" t)))
  338. ;; wait for another connection to arrive
  339. (setq edit-server-received 0)
  340. (setq edit-server-phase 'wait)))))
  341. (defun edit-server-create-frame(buffer)
  342. "Create a frame for the edit server"
  343. (if edit-server-new-frame
  344. (let ((new-frame
  345. (if (memq window-system '(ns mac))
  346. ;; Aquamacs, Emacs NS, Emacs (experimental) Mac port
  347. (make-frame edit-server-new-frame-alist)
  348. (make-frame-on-display (getenv "DISPLAY")
  349. edit-server-new-frame-alist))))
  350. (unless edit-server-new-frame-mode-line
  351. (setq mode-line-format nil))
  352. (select-frame new-frame)
  353. (when (and (eq window-system 'x)
  354. (fboundp 'x-send-client-message))
  355. (x-send-client-message nil 0 nil
  356. "_NET_ACTIVE_WINDOW" 32
  357. '(1 0 0)))
  358. (raise-frame new-frame)
  359. (set-window-buffer (frame-selected-window new-frame) buffer)
  360. new-frame)
  361. (pop-to-buffer buffer)
  362. (raise-frame)
  363. nil))
  364. (defun edit-server-choose-major-mode ()
  365. "Use `edit-server-url-major-mode-alist' to choose a major mode
  366. initialization function based on `edit-server-url', or fall back
  367. to `edit-server-default-major-mode'"
  368. (let ((pairs edit-server-url-major-mode-alist)
  369. (mode edit-server-default-major-mode))
  370. (while pairs
  371. (let ((entry (car pairs)))
  372. (if (string-match (car entry) edit-server-url)
  373. (setq mode (cdr entry)
  374. pairs nil)
  375. (setq pairs (cdr pairs)))))
  376. (funcall mode)))
  377. (defun edit-server-create-edit-buffer(proc)
  378. "Create an edit buffer, place content in it and save the network
  379. process for the final call back"
  380. (let ((buffer (generate-new-buffer
  381. (or edit-server-url
  382. edit-server-edit-buffer-name))))
  383. (with-current-buffer buffer
  384. (when (fboundp 'set-buffer-multibyte)
  385. (set-buffer-multibyte t))) ; djb
  386. (copy-to-buffer buffer (point-min) (point-max))
  387. (with-current-buffer buffer
  388. (setq edit-server-url (with-current-buffer (process-buffer proc) edit-server-url))
  389. (edit-server-choose-major-mode)
  390. ;; Allow `edit-server-start-hook' to override the major mode.
  391. ;; (re)setting the minor mode seems to clear the buffer-local
  392. ;; variables that we depend upon for the response, so call the
  393. ;; hooks early on
  394. (run-hooks 'edit-server-start-hook)
  395. (not-modified)
  396. (add-hook 'kill-buffer-hook 'edit-server-abort* nil t)
  397. (buffer-enable-undo)
  398. (setq edit-server-proc proc
  399. edit-server-frame (edit-server-create-frame buffer))
  400. (edit-server-edit-mode))))
  401. (defun edit-server-send-response (proc &optional body close)
  402. "Send an HTTP 200 OK response back to process PROC.
  403. Optional second argument BODY specifies the response content:
  404. - If nil, the HTTP response will have null content.
  405. - If a string, the string is sent as response content.
  406. - Any other value will cause the contents of the current
  407. buffer to be sent.
  408. If optional third argument CLOSE is non-nil, then process PROC
  409. and its buffer are killed with `edit-server-kill-client'."
  410. (interactive)
  411. (if (processp proc)
  412. (let ((response-header (concat
  413. "HTTP/1.0 200 OK\n"
  414. (format "Server: Emacs/%s\n" emacs-version)
  415. "Date: "
  416. (format-time-string
  417. "%a, %d %b %Y %H:%M:%S GMT\n"
  418. (current-time)))))
  419. (process-send-string proc response-header)
  420. (process-send-string proc "\n")
  421. (cond
  422. ((stringp body)
  423. (process-send-string proc (encode-coding-string body 'utf-8)))
  424. ((not body) nil)
  425. (t
  426. (encode-coding-region (point-min) (point-max) 'utf-8)
  427. (process-send-region proc (point-min) (point-max))))
  428. (process-send-eof proc)
  429. (when close
  430. (edit-server-kill-client proc))
  431. (edit-server-log proc "Editing done, sent HTTP OK response."))
  432. (message "edit-server-send-response: invalid proc (bug?)")))
  433. (defun edit-server-kill-client (proc)
  434. "Kill client process PROC and remove it from the list."
  435. (let ((procbuf (process-buffer proc)))
  436. (delete-process proc)
  437. (when (buffer-live-p procbuf)
  438. (kill-buffer procbuf))
  439. (setq edit-server-clients (delq proc edit-server-clients))))
  440. (defun edit-server-done (&optional abort nokill)
  441. "Finish editing: send HTTP response back, close client and editing buffers.
  442. The current contents of the buffer are sent back to the HTTP
  443. client, unless argument ABORT is non-nil, in which case then the
  444. original text is sent back.
  445. If optional second argument NOKILL is non-nil, then the editing
  446. buffer is not killed.
  447. When called interactively, use prefix arg to abort editing."
  448. (interactive "P")
  449. ;; Do nothing if the connection is closed by the browser (tab killed, etc.)
  450. (unless (eq (process-status edit-server-proc) 'closed)
  451. (let ((buffer (current-buffer))
  452. (proc edit-server-proc)
  453. (procbuf (process-buffer edit-server-proc)))
  454. ;; edit-server-* vars are buffer-local,
  455. ;; so they must be used before issuing kill-buffer
  456. (if abort
  457. ;; send back original content
  458. (with-current-buffer procbuf
  459. (run-hooks 'edit-server-done-hook)
  460. (edit-server-send-response proc t))
  461. ;; send back edited content
  462. (save-restriction
  463. (widen)
  464. (buffer-disable-undo)
  465. ;; ensure any format encoding is done (like longlines)
  466. (dolist (format buffer-file-format)
  467. (format-encode-region (point-min) (point-max) format))
  468. ;; send back
  469. (run-hooks 'edit-server-done-hook)
  470. (edit-server-send-response edit-server-proc t)
  471. ;; restore formats (only useful if we keep the buffer)
  472. (dolist (format buffer-file-format)
  473. (format-decode-region (point-min) (point-max) format))
  474. (buffer-enable-undo)))
  475. (when edit-server-frame
  476. (delete-frame edit-server-frame))
  477. ;; delete-frame may change the current buffer
  478. (unless nokill
  479. (kill-buffer buffer))
  480. (edit-server-kill-client proc))))
  481. ;;
  482. ;; There are a couple of options for handling the save
  483. ;; action. The intent is to preserve data but not finish
  484. ;; editing. There are a couple of approaches that could
  485. ;; be taken:
  486. ;; a) Use the iterative edit-server option (send a buffer
  487. ;; back to the client which then returns new request
  488. ;; to continue the session).
  489. ;; b) Back-up the edit session to a file
  490. ;; c) Save the current buffer to the kill-ring
  491. ;;
  492. ;; I've attempted to do a) a couple of times but I keep running
  493. ;; into problems which I think are emacs bugs. So for now I'll
  494. ;; just push the current buffer to the kill-ring.
  495. (defun edit-server-save ()
  496. "Save the current state of the edit buffer."
  497. (interactive)
  498. (save-restriction
  499. (widen)
  500. (buffer-disable-undo)
  501. (copy-region-as-kill (point-min) (point-max))
  502. (buffer-enable-undo)))
  503. (defun edit-server-abort ()
  504. "Discard editing and send the original text back to the browser."
  505. (interactive)
  506. (edit-server-done t))
  507. (defun edit-server-abort* ()
  508. "Discard editing and send the original text back to the browser,
  509. but don't kill the editing buffer."
  510. (interactive)
  511. (edit-server-done t t))
  512. (provide 'edit-server)
  513. ;;; edit-server.el ends here