-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspiral-socket.el
385 lines (329 loc) · 15.3 KB
/
spiral-socket.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
;;; spiral-socket.el --- SPIRAL socket connection utilities -*- lexical-binding: t; -*-
;;
;; Filename: spiral-socket.el
;; Description:
;; Author: Daniel Barreto <[email protected]>
;; Maintainer: Daniel Barreto <[email protected]>
;; Copyright (C) 2017 Daniel Barreto
;; Created: Sat Dec 2 13:55:17 2017 (+0100)
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Commentary:
;;
;; SPIRAL socket connection utilities
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; This program 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 of the License, or (at
;; your option) any later version.
;;
;; This program 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 GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Code:
(require 'ansi-color)
(require 'find-func)
(require 'spiral-loop)
(require 'spiral-mode)
(require 'spiral-util)
(defcustom spiral-blob-path nil
"Path to UNREPL 'blob' file.
When nil, SPIRAL will try to find the blob file in the same directory
where the spiral package ins installed."
:type 'file
:group 'spiral)
(defcustom spiral-default-socket-repl-command "boot"
"The default command to be used when creating a Clojure Socket REPL."
:type '(choice (const "clj")
(const "boot")
(const "lein"))
:group 'spiral)
(defcustom spiral-preferred-build-tool nil
"Allow choosing a build system when there are many.
When there are artifacts from multiple build systems (\"lein\", or
\"boot\") the user is prompted to select one of them. When non-nil, this
variable will suppress this behavior and will select whatever build system
is indicated by the variable if present. Note, this is only when CIDER
cannot decide which of many build systems to use and will never override a
command when there is no ambiguity."
:type '(choice "clj"
"boot"
"lein"
(const :tag "Always ask" nil))
:group 'spiral)
(defcustom spiral-start-socket-repl-expr
"(do (require 'clojure.core.server) (let [srv (clojure.core.server/start-server {:name :repl :port 0 :accept 'clojure.core.server/repl :server-daemon false})] (println (pr-str [:spiral/server-ready (.getLocalPort srv)]))))"
"Expression to evaluate to initiate a Socket REPL running on a random port."
:type 'string
:group 'spiral)
(defcustom spiral-lein-command "lein"
"The command used to execute Leiningen."
:type 'string
:group 'spiral)
(defcustom spiral-lein-global-options nil
"Command global options used to execute Leiningen (e.g.: -o for offline)."
:type 'string
:group 'spiral
:safe #'stringp)
(defcustom spiral-lein-parameters "trampoline run"
"Params passed to Lein to start a Socket REPL server via `spiral-connect'.
Command resolution will append
\"-m clojure.main -e `spiral-start-socket-repl-expr'\" at the end of these
parameters. See `spiral-socket--repl-cmd' for more info."
:type 'string
:group 'spiral
:safe #'stringp)
(defcustom spiral-boot-command "boot"
"The command used to execute Boot."
:type 'string
:group 'spiral)
(defcustom spiral-boot-global-options nil
"Command global options used to execute Boot (e.g.: -c for checkouts)."
:type 'string
:group 'spiral)
(defcustom spiral-boot-parameters "wait"
"Params passed to boot to start a Socket REPL server via `spiral-connect'.
Command resolution will prepend \"-i `spiral-start-socket-repl-expr' before
these parameters. See `spiral-socket--repl-cmd' for more info."
:type 'string
:group 'spiral)
(defcustom spiral-clj-command "clojure"
"The command used to execute Clojure."
:type 'string
:group 'spiral
:safe #'stringp)
(defcustom spiral-clj-global-options nil
"Command global options to execute Clojure."
:type 'string
:group 'spiral)
(defvar-local spiral-server-host-port nil
"Tuple with host and port of a Socket REPL.
Only used when spinning a new Socket REPL and waiting for it to boot so
that its corresponding connection pool can be created.")
(defvar-local spiral-server-initializing t
"Boolean flag that indicates if a server Socket REPL has started yet.")
(define-error 'spiral-connection-error "There was a problem connecting")
;; Utilities
;; -------------------------------------------------------------------
;; project type dependent
(defun spiral-socket--identify-buildtools-present ()
"Identify build systems present by their build files in PROJECT-DIR.
BORROWED FROM CIDER."
(let* ((default-directory (spiral-clojure-dir))
(build-files '(("lein" . "project.clj")
("boot" . "build.boot")
("clj" . "deps.edn"))))
(delq nil
(mapcar (lambda (candidate)
(when (file-exists-p (cdr candidate))
(car candidate)))
build-files))))
(defun spiral-socket--clojure-project-type ()
"Determine the type of Clojure project.
BORROWED FROM CIDER.
If more than one project file types are present, check for a preferred
build tool in `spiral-preferred-build-tool', otherwise prompt the user to
choose."
(let* ((choices (spiral-socket--identify-buildtools-present))
(multiple-project-choices (> (length choices) 1))
(default (car choices)))
(cond ((and multiple-project-choices
(member spiral-preferred-build-tool choices))
spiral-preferred-build-tool)
(multiple-project-choices
(completing-read (format "Which command should be used (default %s): " default)
choices nil t nil nil default))
(choices
(car choices))
(t spiral-default-socket-repl-command))))
(defun spiral-socket--repl-cmd (project-type)
"Return a string specifying a Socket REPL cmd using PROJECT-TYPE on PORT.
PROJECT-TYPE is a symbol. It can be either `boot' or `lein'."
(pcase project-type
("clj" (format "%s%s -e \"%s\""
spiral-clj-command
(if spiral-clj-global-options
(concat " " spiral-clj-global-options)
"")
spiral-start-socket-repl-expr))
("boot" (format "%s%s call -e \"%s\" %s"
spiral-boot-command
(if spiral-boot-global-options
(concat " " spiral-boot-global-options)
"")
spiral-start-socket-repl-expr
spiral-boot-parameters))
("lein" (format "%s%s %s -m clojure.main -e \"%s\""
spiral-lein-command
(if spiral-lein-global-options
(concat " " spiral-lein-global-options)
" ")
spiral-lein-parameters
spiral-start-socket-repl-expr))
(_ (user-error (format
(concat
"SPIRAL couldn't figure out your build tool (%s).\n"
"Maybe it is missing or maybe there is no support for it yet.")
project-type)))))
;; blob
(defun spiral-socket--find-default-blob ()
"Look for the blob file in the same dir where `spiral-connect' is defined."
(expand-file-name "blob.clj" (spiral-dir)))
(defun spiral-socket--blob ()
"Return the SPIRAL blob as a string."
(let ((blob-path (or spiral-blob-path
(spiral-socket--find-default-blob))))
(with-temp-buffer
(insert-file-contents blob-path)
(buffer-string))))
;; network
(defun spiral-socket--server-initialized-p (output)
"Check if OUTPUT can be interpreted as a Socket REPL initialization.
Return new Socket REPL port"
(when-let (server-ready-match (string-match "\\[:spiral/server-ready \\([0-9]+\\)\\]" output))
(string-to-number (match-string 1 output))))
(defun spiral-socket--get-network-buffer (type server-host &optional server-port)
"Return a buffer for a connection of type TYPE.
The buffer will hold a `spiral-conn-id' variable of the form:
SERVER-HOST:SERVER-PORT.
TYPE is a keyword, it can either be `:client', `:side-loader' or
`:aux', and it is only used to generate a correct name for the buffer.
This function makes sure to get or create a buffer to be used for the
output of a network connection process."
(let* ((buff-name (if (eql type :server)
"*spiral-server[init]*"
(format "*spiral-%s[%s:%S]*"
(spiral-keyword-name type)
server-host server-port)))
(buff (get-buffer-create buff-name)))
(when (and server-host server-port)
(with-current-buffer buff
(erase-buffer)
(setq-local spiral-server-host-port (cons server-host server-port))))
buff))
(defun spiral-socket--server-message-handler (connected-callback)
"Create a proper 'process filter' function for a newly created Socket REPL.
The returned function receives two arguments: PROCESS and OUTPUT, and waits
for the Socket REPL initialization to execute CONNECTED-CALLBACK.
PROJECT-TYPE is used to figure out when has the REPL been initialized."
(let ((calling-buffer (current-buffer)))
(lambda (process output)
(let ((server-buffer (process-buffer process)))
(when (and (buffer-live-p server-buffer)
(buffer-local-value 'spiral-server-initializing server-buffer))
(with-current-buffer server-buffer
(when-let (port (spiral-socket--server-initialized-p
(concat (buffer-substring-no-properties (point-min) (point-max))
output)))
(message "Clojure Socket REPL server started on %d." port)
(rename-buffer
(replace-regexp-in-string "init" (number-to-string port) (buffer-name))
t)
(setq-local spiral-server-initializing nil)
(with-current-buffer calling-buffer
(funcall connected-callback process "localhost" port)))
(goto-char (point-max))
(insert (if (string-match "^\s*\\[" output)
(ansi-color-apply output)
output))))))))
(defun spiral-socket--server-sentinel (process event)
"Handle a Socket REPL server PROCESS EVENT."
(let* ((server-buffer (process-buffer process))
(conn-id (buffer-local-value 'spiral-conn-id server-buffer)))
(spiral-disconnect conn-id)
(unless (string-match-p "^killed\\|^interrupt\\|^hangup\\|^broken" event)
(if (and server-buffer (buffer-live-p server-buffer))
(with-current-buffer server-buffer
(setq-local spiral-server-initializing nil)
(error "Could not start Socket REPL server: %s"
(buffer-substring-no-properties (point-min) (point-max))))
(error "Could not start Socket REPL server")))))
(defun spiral-socket--client-sentinel (process event)
"Handle a Socket REPL client PROCESS EVENT.
When a 'disconnect' EVENT comes from PROCESS, quit the PROCESS' project and
shares the message with the user."
(when (string-match-p "^killed\\|^interrupt\\|^hangup\\|broken" event)
(spiral-disconnect (buffer-local-value 'spiral-conn-id (process-buffer process))
event)))
;; Main connection functions
;; -------------------------------------------------------------------
(defun spiral-socket-create-server (connected-callback)
"Create a new Clojure Socket REPL and return its process.
CONNECTED-CALLBACK is a function to be executed when the new Socket REPL
has started, it should receive three parameters: PROCESS, HOST, and PORT."
(let* ((default-directory (spiral-clojure-dir))
(project-type (spiral-socket--clojure-project-type))
(cmd (spiral-socket--repl-cmd project-type))
(pretty-cmd (propertize cmd 'face 'font-lock-keyword-face))
(server-buf (let ((b (spiral-socket--get-network-buffer :server "localhost")))
(with-current-buffer b
(setq-local spiral-server-initializing t))
b))
(server-proc (start-file-process-shell-command "spiral-socket-server"
server-buf
cmd)))
(message "Starting a new Socket REPL via %s" pretty-cmd)
;; Set process configuration
(set-process-coding-system server-proc 'utf-8-unix 'utf-8-unix)
(set-process-sentinel server-proc #'spiral-socket--server-sentinel)
(set-process-filter
server-proc
(spiral-socket--server-message-handler connected-callback))
;; If initialization takes too much time, we want to show the server buffer
;; to users so they know why its taking that long (or at least to give a
;; sense of 'work being done...')
(run-at-time 5 nil (lambda ()
(with-current-buffer server-buf
(when spiral-server-initializing
(save-excursion
(goto-char (point-min))
(insert (propertize "Things seem to be taking a bit too long... here's the command's output" 'face 'font-lock-comment-face)
"\n"
pretty-cmd
"\n\n"))
(pop-to-buffer server-buf)))))
;; Finally return the process
server-proc))
(defun spiral-socket-connect (type host port dispatcher-fn
&optional upgrade-msg)
"Create a new TYPE connection process to HOST:PORT.
TYPE is a keyword: `:client', `:aux', or `:side-loader'.
UPGRADE-MSG is a STR to be sent to the newly created process.
DISPATCHER-FN is a function that will receive UNREPL EDN messages and will
dispatch to handlers according to the message's tag.
Returns a pair (TYPE . new-process)."
(let* ((conn-id (spiral-make-conn-id host port))
(new-proc (condition-case err
(make-network-process
:name (format "spiral-%s" (spiral-keyword-name type))
:buffer (spiral-socket--get-network-buffer type host port)
:host host
:service port
:filter #'spiral-loop-handle-proc-message)
(file-error
(signal 'spiral-connection-error
(list conn-id
(cadr (cdr err))))))))
;; Setup process
(process-send-string new-proc (or upgrade-msg (spiral-socket--blob)))
(when (eql type :client)
(set-process-sentinel new-proc #'spiral-socket--client-sentinel))
(with-current-buffer (process-buffer new-proc)
(clojure-mode)
(setq-local spiral-conn-id conn-id)
(setq-local spiral-loop-process-type type)
(setq-local spiral-loop-process-dispatcher dispatcher-fn))
;; Finally return the process
new-proc))
(provide 'spiral-socket)
;;; spiral-socket.el ends here