-
Notifications
You must be signed in to change notification settings - Fork 34
/
Copy pathlsp-docker.el
806 lines (713 loc) · 43.5 KB
/
lsp-docker.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
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
;;; lsp-docker.el --- LSP Docker integration -*- lexical-binding: t; -*-
;; Copyright (C) 2019 Ivan Yonchovski
;; Author: Ivan Yonchovski <yyoncho@gmail.com>
;; URL: https://github.com/emacs-lsp/lsp-docker
;; Keywords: languages langserver
;; Version: 1.0.0
;; Package-Requires: ((emacs "27.1") (dash "2.14.1") (lsp-mode "6.2.1") (f "0.20.0") (s "1.13.0") (yaml "0.2.0") (ht "2.0"))
;; 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 this program. If not, see <https://www.gnu.org/licenses/>.
;;; Commentary:
;; Run language servers in containers
;;; Code:
(require 'lsp-mode)
(require 'dash)
(require 'f)
(require 's)
(require 'yaml)
(require 'ht)
(defgroup lsp-docker nil
"Language Server Protocol dockerized servers support."
:group 'lsp-mode
:tag "Language Server in docker (lsp-docker)")
(defcustom lsp-docker-log-docker-supplemental-calls nil
"If non-nil, all docker command supplemental-calls will be logged to a buffer."
:group 'lsp-docker
:type 'boolean)
(defcustom lsp-docker-log-docker-supplemental-calls-buffer-name "*lsp-docker-supplemental-calls*"
"Log docker supplemental calls using this particular buffer."
:group 'lsp-docker
:type 'string)
;; top node keys
(defconst lsp-docker--lsp-key 'lsp
"Main key associated to the root-node of the containerized language servers")
;; 1st sub-node keys
(defconst lsp-docker--server-key 'server
"LSP sub-key holding a single (or a group of) server(s)")
(defconst lsp-docker--mappings-key 'mappings
"Collection of mappings between host-paths and
containerized-paths (host paths must be within the project)")
;; 2nd sub-node keys
;; supported keys in YAML configuration file(s)
(defconst lsp-docker--srv-cfg-type-key 'type
"The type of server (at the moment only `docker' is supported).")
(defconst lsp-docker--srv-cfg-subtype-key 'subtype
"For type container it can be:
- `container': attach to an already running container
- `image': when the image does not exist, try to build it based on the dockerfile
found in the project-scope An image might feature an optional tag, i.e.
`<image>:<tag>'. the If a tagless image is indicated `latest' will be assumed")
(defconst lsp-docker--srv-cfg-name-key 'name
"Depending on the `lsp-docker--srv-cfg-subtype-key' it holds the
name of the container/image for the described language server.")
(defconst lsp-docker--srv-cfg-server-key 'server
"Server ID of a registered LSP server. You can find the list of
registered servers evaluating: `(ht-keys lsp-clients)'.")
(defconst lsp-docker--srv-cfg-launch-parameters-key 'launch_parameters
"Command parameters (docker or podman) to launch the language server with.
Pay attention that these parameters have to be supported by the selected subtype.")
(defconst lsp-docker--srv-cfg-launch-command-key 'launch_command
"Command to launch the language server in stdio mode. This key is
not used when the `lsp-docker--srv-cfg-subtype-key' is set to
container, as the server command shall be the entrypoint.")
(defun lsp-docker--log-docker-supplemental-calls-p ()
"Return non-nil if should log docker invocation commands"
lsp-docker-log-docker-supplemental-calls)
(defun lsp-docker--uri->path (path-mappings docker-container-name uri)
"Turn docker URI into host path.
Argument PATH-MAPPINGS dotted pair of (host-path . container-path).
Argument DOCKER-CONTAINER-NAME name to use when running container.
Argument URI the uri to translate."
(let ((path (lsp--uri-to-path-1 uri)))
(-if-let ((local . remote) (-first (-lambda ((_ . docker-path))
(s-contains? docker-path path))
path-mappings))
(replace-regexp-in-string (format "\\(%s\\).*" remote) local path nil nil 1)
(format "/docker:%s:%s" docker-container-name path))))
(defun lsp-docker--path->uri (path-mappings path)
"Turn host PATH into docker uri.
Argument PATH-MAPPINGS dotted pair of (host-path . container-path).
Argument PATH the path to translate."
(lsp--path-to-uri-1
(-if-let ((local . remote) (-first (-lambda ((local-path . _))
(s-contains? local-path path))
path-mappings))
(replace-regexp-in-string (format "\\(%s\\).*" local) remote path nil nil 1)
(user-error "The path %s is not under path mappings" path))))
(defvar lsp-docker-container-name-suffix 0
"Used to prevent collision of container names.")
(defvar lsp-docker-command "docker"
"The docker command to use.")
(defun lsp-docker-launch-new-container (docker-container-name path-mappings launch-parameters docker-image-id server-command)
"Return the docker command to be executed on host.
Argument DOCKER-CONTAINER-NAME name to use for container.
Argument PATH-MAPPINGS dotted pair of (host-path . container-path).
Argument DOCKER-IMAGE-ID the docker container to run language servers with.
Argument LAUNCH-PARAMETERS parameters (for docker or podman) to run language servers with.
Argument SERVER-COMMAND the language server command to run inside the container."
(-remove #'s-blank?
(split-string
(format "%s run --name %s --rm -i %s %s %s %s"
lsp-docker-command
docker-container-name
(->> path-mappings
(-map (-lambda ((path . docker-path))
(format "-v %s:%s" path docker-path)))
(s-join " "))
(s-join " " launch-parameters)
docker-image-id
server-command)
" ")))
(defun lsp-docker-exec-in-container (docker-container-name server-command)
"Return command to exec into running container.
Argument DOCKER-CONTAINER-NAME name of container to exec into.
Argument SERVER-COMMAND the command to execute inside the running container."
(split-string
(format "%s exec -i %s %s" lsp-docker-command docker-container-name server-command)))
(defun lsp-docker--attach-container-name-global-suffix (container-name)
"Attach a user-specified or a default suffix (properly changing it) to the container name"
(if lsp-docker-container-name-suffix
(format "%s-%d"
container-name
(if (numberp lsp-docker-container-name-suffix)
(cl-incf lsp-docker-container-name-suffix)
lsp-docker-container-name-suffix))
container-name))
(cl-defun lsp-docker-register-client (&key server-id
docker-server-id
path-mappings
launch-parameters
docker-image-id
docker-container-name
priority
server-command
launch-server-cmd-fn)
"Registers docker clients with lsp"
(if-let ((client (copy-lsp--client (gethash server-id lsp-clients))))
(progn
(let ((docker-container-name-full (lsp-docker--attach-container-name-global-suffix docker-container-name)))
(setf (lsp--client-server-id client) docker-server-id
(lsp--client-uri->path-fn client) (-partial #'lsp-docker--uri->path
path-mappings
docker-container-name-full)
(lsp--client-path->uri-fn client) (-partial #'lsp-docker--path->uri path-mappings)
(lsp--client-new-connection client) (plist-put
(lsp-stdio-connection
(lambda ()
(funcall (or launch-server-cmd-fn #'lsp-docker-launch-new-container)
docker-container-name-full
path-mappings
launch-parameters
docker-image-id
server-command)))
:test? (lambda (&rest _)
(-any?
(-lambda ((dir))
(f-ancestor-of? dir (buffer-file-name)))
path-mappings)))
(lsp--client-priority client) (or priority (lsp--client-priority client))))
(lsp-register-client client))
(user-error "No such client %s" server-id)))
(defvar lsp-docker-default-client-packages
'(lsp-bash
lsp-clangd
lsp-css
lsp-dockerfile
lsp-go
lsp-html
lsp-javascript
lsp-pylsp)
"Default list of client packages to load.")
(defvar lsp-docker-default-client-configs
(list
(list :server-id 'bash-ls :docker-server-id 'bashls-docker :server-command "bash-language-server start")
(list :server-id 'clangd :docker-server-id 'clangd-docker :server-command "ccls")
(list :server-id 'css-ls :docker-server-id 'cssls-docker :server-command "css-languageserver --stdio")
(list :server-id 'dockerfile-ls :docker-server-id 'dockerfilels-docker :server-command "docker-langserver --stdio")
(list :server-id 'gopls :docker-server-id 'gopls-docker :server-command "gopls")
(list :server-id 'html-ls :docker-server-id 'htmls-docker :server-command "html-languageserver --stdio")
(list :server-id 'pylsp :docker-server-id 'pyls-docker :server-command "pylsp")
(list :server-id 'ts-ls :docker-server-id 'tsls-docker :server-command "typescript-language-server --stdio"))
"Default list of client configurations.")
(cl-defun lsp-docker-init-clients (&key
path-mappings
(docker-image-id "emacslsp/lsp-docker-langservers")
(docker-container-name "lsp-container")
(priority 10)
(client-packages lsp-docker-default-client-packages)
(client-configs lsp-docker-default-client-configs))
"Loads the required client packages and registers the required clients to run with docker.
:path-mappings is an alist of local paths and their mountpoints
in the docker container.
Example: '((\"/path/to/projects\" . \"/projects\"))
:docker-image-id is the identifier for the docker image to be
used for all clients, as a string.
:docker-container-name is the name to use for the container when
it is started.
:priority is the priority with which to register the docker
clients with lsp. (See the library ‘lsp-clients’ for details.)
:client-packages is a list of libraries to load before registering the clients.
:client-configs is a list of configurations for the various
clients you wish to use with ‘lsp-docker’. Each element takes
the form
'(:server-id 'example-ls
:docker-server-id 'examplels-docker
:docker-image-id \"examplenamespace/examplels-docker:x.y\"
:docker-container-name \"examplels-container\"
:server-command \"run_example_ls.sh\")
where
:server-id is the ID of the language server, as defined in the
library ‘lsp-clients’.
:docker-server-id is any arbitrary unique symbol used internally
by ‘lsp’ to distinguish it from non-docker clients for the same
server.
:docker-image-id is an optional property to override this
function's :docker-image-id argument for just this client. If
you specify this, you MUST also specify :docker-container-name.
:docker-container-name is an optional property to override this
function's :docker-container-name argument for just this client.
This MUST be specified if :docker-image-id is specified, but is
otherwise optional.
:server-command is a string specifying the command to run inside
the docker container to run the language server."
(seq-do (lambda (package) (require package nil t)) client-packages)
(let ((default-docker-image-id docker-image-id)
(default-docker-container-name docker-container-name))
(seq-do (-lambda ((&plist :server-id :docker-server-id :docker-image-id :docker-container-name :server-command))
(when (and docker-image-id (not docker-container-name))
(user-error "Invalid client definition for server ID %S. You must specify a container name when specifying an image ID."
server-id))
(lsp-docker-register-client
:server-id server-id
:priority priority
:docker-server-id docker-server-id
:docker-image-id (or docker-image-id default-docker-image-id)
:docker-container-name (if docker-image-id
docker-container-name
default-docker-container-name)
:server-command server-command
:path-mappings path-mappings
:launch-parameters nil
:launch-server-cmd-fn #'lsp-docker-launch-new-container))
client-configs)))
(defvar lsp-docker-default-priority
100
"Default lsp-docker containerized servers priority (it needs to
be bigger than default servers in order to override them)")
(defcustom lsp-docker-persistent-default-config
(ht (lsp-docker--server-key (ht (lsp-docker--srv-cfg-type-key "docker")
(lsp-docker--srv-cfg-subtype-key "image")
(lsp-docker--srv-cfg-name-key "emacslsp/lsp-docker-langservers")
(lsp-docker--srv-cfg-server-key nil)
(lsp-docker--srv-cfg-launch-command-key nil)))
(lsp-docker--mappings-key (vector
(ht ('source ".")
('destination "/projects")))))
"Default configuration for all language servers with persistent configurations"
:type 'hash-table
:group 'lsp-docker)
(defun lsp-docker-get-config-from-project-config-file (project-config-file-path)
"Get the LSP configuration based on a project configuration file"
(if (f-exists? project-config-file-path)
(if-let* ((whole-config (yaml-parse-string (f-read project-config-file-path)))
(lsp-config (gethash lsp-docker--lsp-key whole-config)))
;; use default values for missing fields in the provided configuration
(if (vectorp (gethash lsp-docker--server-key lsp-config))
lsp-config ; DO NOT merge to the persistent configuration when a multi-server one is detected
(ht-merge (ht-copy lsp-docker-persistent-default-config) lsp-config)))))
(defun lsp-docker--find-project-config-file-from-lsp ()
"Get the LSP configuration file path (project-local configuration, using lsp-mode)"
(let ((config-file-path-candidates (list)))
(when (lsp-workspace-root)
(push (f-join (lsp-workspace-root) ".lsp-docker.yml") config-file-path-candidates)
(push (f-join (lsp-workspace-root) ".lsp-docker.yaml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") ".lsp-docker.yml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") ".lsp-docker.yaml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "lsp-docker.yml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "lsp-docker.yaml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "config.yml") config-file-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "config.yaml") config-file-path-candidates)
(--first (f-exists? it) config-file-path-candidates))))
(defun lsp-docker--find-project-dockerfile-from-lsp ()
"Get the LSP server building Dockerfile path using lsp-mode"
(let ((dockerfile-path-candidates (list)))
(when (lsp-workspace-root)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "Dockerfile") dockerfile-path-candidates)
(push (f-join (f-join (lsp-workspace-root) ".lsp-docker") "Dockerfile.lsp") dockerfile-path-candidates)
(--first (f-exists? it) dockerfile-path-candidates))))
(defun lsp-docker--find-building-path-from-dockerfile (dockerfile-path)
"Get the LSP server building folder path using an explicit dockerfile path"
(when dockerfile-path
(f-dirname (f-dirname dockerfile-path))))
(defun lsp-docker-get-config-from-lsp ()
"Get the LSP configuration based on a project-local configuration (using lsp-mode)"
(let ((project-config-file-path (lsp-docker--find-project-config-file-from-lsp)))
(if project-config-file-path
(lsp-docker-get-config-from-project-config-file project-config-file-path)
(ht-copy lsp-docker-persistent-default-config))))
(defvar lsp-docker-supported-server-types-subtypes
(ht ('docker (list 'container 'image)))
"A list of all supported server types and subtypes, currently only docker is supported")
(defun lsp-docker-get-server-type-subtype (server-config)
"Get the server type & sub-type from the SERVER-CONFIG hash-table"
(let* ((lsp-server-type (gethash lsp-docker--srv-cfg-type-key server-config))
(lsp-server-subtype (gethash lsp-docker--srv-cfg-subtype-key server-config)))
(cons (if (stringp lsp-server-type)
(intern lsp-server-type)
lsp-server-type)
(if (stringp lsp-server-subtype)
(intern lsp-server-subtype)
lsp-server-subtype))))
(defun lsp-docker-get-server-container-name (server-config)
"Get the server container name from the SERVER-CONFIG hash-table"
(let* ((lsp-server-subtype (gethash 'subtype server-config)))
(if (equal lsp-server-subtype "container")
(gethash 'name server-config))))
(defun lsp-docker-get-server-image-name (server-config)
"Get the server image name from the SERVER-CONFIG hash-table"
(let* ((lsp-server-subtype (gethash 'subtype server-config)))
(if (equal lsp-server-subtype "image")
(gethash 'name server-config))))
(defun lsp-docker--get-server-launch-parameters (server-config)
"Get the server launch parameters from the SERVER-CONFIG hash-table"
(let ((launch-parameters (gethash lsp-docker--srv-cfg-launch-parameters-key server-config)))
(if (or (vectorp launch-parameters)
(not launch-parameters))
launch-parameters
(user-error "Cannot find the right launch parameters"))))
(defun lsp-docker-get-server-id (server-config)
"Get the server id from the SERVER-CONFIG hash-table"
(let ((server-id (gethash lsp-docker--srv-cfg-server-key server-config)))
(if (stringp server-id)
(intern server-id)
server-id)))
(defun lsp-docker--get-base-client (base-server-id)
"Get the base lsp client associated to BASE-SERVER-ID key for
dockerized client to be built upon"
(if-let* ((base-client (gethash base-server-id lsp-clients)))
base-client
(user-error "Cannot find the specified base lsp client (%s)!
Make sure the '%s' sub-key is set to one of the lsp registered clients:\n\n%s"
base-server-id lsp-docker--srv-cfg-server-key (ht-keys lsp-clients))))
(defun lsp-docker-get-path-mappings (config project-directory)
"Get the server path mappings from the top project hash-table CONFIG"
(if-let ((lsp-mappings-info (gethash lsp-docker--mappings-key config)))
(--map (cons (f-canonical (f-expand (gethash 'source it)
project-directory))
(gethash 'destination it))
lsp-mappings-info)
(user-error "No path mappings specified!")))
(defun lsp-docker-get-launch-command (server-config)
"Get the server launch command from the SERVER-CONFIG hash-table"
(gethash lsp-docker--srv-cfg-launch-command-key server-config))
(defun lsp-docker-check-server-type-subtype (supported-server-types-subtypes server-type-subtype)
"Verify that the combination of server (type . subtype) is supported by the current implementation"
(if (not server-type-subtype)
(user-error "No server type and subtype specified!"))
(if (ht-find (lambda (type subtypes)
(let ((server-type (car server-type-subtype))
(server-subtype (cdr server-type-subtype)))
(and (equal server-type type)
(-contains? subtypes server-subtype))))
supported-server-types-subtypes)
server-type-subtype
(user-error "No compatible server type and subtype found!")))
(defun lsp-docker-check-path-mappings (path-mappings)
"Verify that specified path mappings are all inside the project directory"
(--all? (or (f-descendant-of? (f-canonical (car it)) (f-canonical (lsp-workspace-root)))
(f-same? (f-canonical (car it)) (f-canonical (lsp-workspace-root))))
path-mappings))
(defun lsp-docker-launch-existing-container (docker-container-name &rest _unused)
"Return the docker command to be executed on host.
Argument DOCKER-CONTAINER-NAME name to use for container."
(split-string
(format "%s start -ia %s"
lsp-docker-command
docker-container-name)
" "))
(defun lsp-docker-create-activation-function-by-project-dir (project-dir)
`(lambda (&rest unused)
(let ((current-project-root (lsp-workspace-root))
(registered-project-root ,project-dir))
(f-same? current-project-root registered-project-root))))
(defun lsp-docker--create-activation-function-by-project-dir-and-base-client (project-dir base-lsp-client)
`(lambda (current-file-name current-major-mode)
(let ((current-project-root (lsp-workspace-root))
(registered-project-root ,project-dir)
(base-activation-fn ,(lsp--client-activation-fn base-lsp-client))
(base-major-modes ',(lsp--client-major-modes base-lsp-client)))
(and (f-same? current-project-root registered-project-root)
(or (if (functionp base-activation-fn)
(funcall base-activation-fn current-file-name current-major-mode)
nil)
(-contains? base-major-modes current-major-mode))))))
(defun lsp-docker-generate-docker-server-id (server-config project-root)
"Generate the docker-server-id from the SERVER-CONFIG"
(let ((original-server-id (symbol-name (lsp-docker-get-server-id server-config)))
(project-path-server-id-part (s-chop-prefix "-" (s-replace-all '(("/" . "-") ("." . "")) project-root))))
(intern (s-join "-" (list project-path-server-id-part original-server-id "docker")))))
(defun lsp-docker--generate-docker-server-container-name (server-config project-root)
"Generate the docker-container-name from the SERVER-CONFIG"
(let ((docker-server-id (lsp-docker-generate-docker-server-id server-config project-root)))
(if (symbolp docker-server-id)
(symbol-name docker-server-id)
docker-server-id)))
(defun lsp-docker--finalize-docker-server-container-name (config-specified-server-name server-config project-root)
"Get or generate the container name.
If CONFIG-SPECIFIED-SERVER-NAME is non-nil, return it as
container name. Otherwise generate a unique container name from
SERVER-CONFIG and PROJECT-ROOT.
"
(cond ((stringp config-specified-server-name) config-specified-server-name)
('t (lsp-docker--attach-container-name-global-suffix (lsp-docker--generate-docker-server-container-name server-config project-root)))))
(defun lsp-docker--encode-single-quoted-parameters (raw-token-command)
"Encode single quoted tokens (with base64 encoding) so they won't be split"
(let* ((tokens-to-encode (--remove (s-blank-str? (cadr it)) (s-match-strings-all "'\\([^']+\\)'" raw-token-command)))
(replacement-pairs (--mapcat (list (cons (car it) (format "'%s'" (base64-encode-string (cadr it))))) tokens-to-encode)))
(s-replace-all replacement-pairs raw-token-command)))
(defun lsp-docker--decode-single-quoted-parameters (encoded-token-command)
"Decode single quoted tokens (base64-encoded) so they can be used again"
(let* ((tokens-to-decode (--remove (s-blank-str? (cadr it)) (s-match-strings-all "'\\([^']+\\)'" encoded-token-command)))
(replacement-pairs (--mapcat (list (cons (car it) (format "'%s'" (base64-decode-string (cadr it))))) tokens-to-decode)))
(s-replace-all replacement-pairs encoded-token-command)))
(defun lsp-docker--decode-single-quoted-tokens (command-tokens)
"Decode single quoted tokens (base64-encoded) from a token list"
(--map-when (s-match "'\\([^']+\\)'" it) (format "'%s'" (base64-decode-string (cadr (s-match "'\\([^']+\\)'" it)))) command-tokens))
(defun lsp-docker--run-docker-command (command-arguments)
"Run a command (with a configurable command itself: docker or
podman) and get its exit code and output as a pair (exit-code .
output)"
(lsp-docker--run-external-command (format "%s %s" lsp-docker-command command-arguments)))
(defun lsp-docker--get-build-command (image-name dockerfile-path)
"Get a building command string"
(format "%s build --tag %s --file %s %s" lsp-docker-command image-name dockerfile-path (lsp-docker--find-building-path-from-dockerfile dockerfile-path)))
(defun lsp-docker--run-image-build (image-name dockerfile-path buffer-name)
"Build the specified image using a particular dockerfile (with its output redirected to a specified buffer)"
(-let ((
(command-program . command-arguments)
(lsp-docker--decode-single-quoted-tokens (s-split " " (lsp-docker--encode-single-quoted-parameters (lsp-docker--get-build-command image-name dockerfile-path))))))
(with-current-buffer (get-buffer-create buffer-name)
(message "Building the image %s, please open the %s buffer for details" image-name buffer-name)
(apply #'call-process command-program nil (current-buffer) nil command-arguments))))
(defun lsp-docker--run-external-command (command)
"Run a command and get its output and exit code"
(-let ((
(command-program . command-arguments)
(lsp-docker--decode-single-quoted-tokens (s-split " " (lsp-docker--encode-single-quoted-parameters command)))))
(progn
(lsp-docker--conditionally-log-docker-supplemental-call command-program command-arguments)
(lsp-docker--launch-command-internal command-program command-arguments))))
(defun lsp-docker--launch-command-internal (command-program command-arguments)
"Run a command using 'call-process' function and return a pair of exit code and raw output"
(with-temp-buffer
(cons
(apply #'call-process command-program nil (current-buffer) nil command-arguments)
(buffer-string))))
(defun lsp-docker--conditionally-log-docker-supplemental-call (command-program command-arguments)
"Log a command into a buffer set in lsp-docker settings group"
(if (lsp-docker--log-docker-supplemental-calls-p)
(with-current-buffer (get-buffer-create lsp-docker-log-docker-supplemental-calls-buffer-name)
(goto-char (point-max))
(insert (format "LOG: calling %s %s\n" command-program (s-join " " command-arguments))))))
(defun lsp-docker--get-existing-images ()
"Get available docker images already existing on the host"
(-let ((
(exit-code . raw-output)
(lsp-docker--run-docker-command "image list --format '{{.Repository}}:{{.Tag}}'")))
(if (equal exit-code 0)
;; filter out the list of tagged images from cmd output
(--remove (s-blank? it) (--map (s-chop-suffix "'" (s-chop-prefix "'" it)) (s-lines raw-output)))
(user-error "Cannot get the existing images list from the host, exit code: %d" exit-code))))
(defun lsp-docker--get-existing-containers ()
"Get available docker images already existing on the host"
(-let ((
(exit-code . raw-output)
(lsp-docker--run-docker-command "container list --all --format '{{.Names}}'")))
(if (equal exit-code 0)
(--remove (s-blank? it) (--map (s-chop-suffix "'" (s-chop-prefix "'" it)) (s-lines raw-output)))
(user-error "Cannot get the existing containers list from the host, exit code: %d" exit-code))))
(defun lsp-docker--check-image-exists (image-name)
"Check that the specified image already exists on the host"
;; automatically add "latest" tag when `image-name' is an untagged image name
(let ((target-image (if (not (string-match "[:]" image-name))
(format "%s:latest" image-name)
image-name)))
(--any? (s-equals? it target-image) (lsp-docker--get-existing-images))))
(defun lsp-docker--check-container-exists (container-name)
"Check that the specified container already exists on the host"
(--any? (s-equals? it container-name) (lsp-docker--get-existing-containers)))
(defun lsp-docker--generate-build-buffer-name (image-name dockerfile-path)
"Generate a buffer name used when building the specified image"
(let ((image-part image-name)
(dockerfile-path-part (s-chop-prefix "-" (s-replace-all '(("/" . "-") ("." . "")) dockerfile-path))))
(s-join "-" (list image-part dockerfile-path-part "build"))))
(defun lsp-docker--build-image-if-necessary (image-name dockerfile-path)
"Check that the specified image exists, otherwise build it (if possible)"
(unless (lsp-docker--check-image-exists image-name)
(if dockerfile-path
(if (y-or-n-p (format "Image %s is missing but can be built (Dockerfile was found), do you want to build it?" image-name))
(let ((build-buffer-name (lsp-docker--generate-build-buffer-name image-name dockerfile-path)))
(lsp-docker--run-image-build image-name dockerfile-path build-buffer-name))
(user-error "Cannot register a server with a missing image!"))
(user-error "Cannot find the image %s but cannot build it too (missing Dockerfile)" image-name))))
(defun lsp-docker--create-building-process-sentinel (
server-id
docker-server-id
path-mappings
launch-parameters
image-name
docker-container-name
activation-fn
server-command)
`(lambda (proc _message)
(when (eq (process-status proc) 'exit)
(lsp-docker-register-client-with-activation-fn
:server-id ',server-id
:docker-server-id ',docker-server-id
:path-mappings ',path-mappings
:launch-parameters ,launch-parameters
:docker-image-id ',image-name
:docker-container-name ',docker-container-name
:activation-fn ,activation-fn
:priority lsp-docker-default-priority
:server-command ',server-command
:launch-server-cmd-fn #'lsp-docker-launch-new-container))))
(cl-defun lsp-docker--build-image-and-register-server-async (&key image-name
dockerfile-path
server-id
docker-server-id
path-mappings
launch-parameters
docker-container-name
activation-fn
server-command
;; TODO: keep these inputs for future feature
;; implementation, see
;; https://github.com/sfavazza/lsp-docker/pull/1#discussion_r1367081991
;; project-root
;; docker-image-id
;; priority
;; launch-server-cmd-fn
)
"Build an image asynchronously and register it afterwards"
(unless (lsp-docker--check-image-exists image-name) ;; Check again whether we have to build a new image
(if dockerfile-path
(if (y-or-n-p (format "Image %s is missing but can be built (Dockerfile was found), do you want to build it?" image-name))
(let* ((build-buffer-name (lsp-docker--generate-build-buffer-name image-name dockerfile-path))
(build-command (lsp-docker--get-build-command image-name dockerfile-path))
(build-command-decoded (lsp-docker--decode-single-quoted-tokens (s-split " " (lsp-docker--encode-single-quoted-parameters build-command)))))
(with-current-buffer (get-buffer-create build-buffer-name)
(lsp-docker--conditionally-log-docker-supplemental-call (car build-command-decoded) (cdr build-command-decoded))
(message "Building the image %s, please open the %s buffer for details" image-name build-buffer-name)
(make-process
:name "lsp-docker-build"
:buffer (current-buffer)
:command build-command-decoded
:sentinel (lsp-docker--create-building-process-sentinel
server-id
docker-server-id
path-mappings
launch-parameters
image-name
docker-container-name
activation-fn
server-command))))
(user-error "Cannot register a server with a missing image!"))
(user-error "Cannot find the image %s but cannot build it too (missing Dockerfile)" image-name))))
(cl-defun lsp-docker-register-client-with-activation-fn (&key server-id
docker-server-id
path-mappings
launch-parameters
docker-image-id
docker-container-name
activation-fn
priority
server-command
launch-server-cmd-fn)
"Registers docker clients with lsp (by persisting configuration)"
(if-let ((client (copy-lsp--client (gethash server-id lsp-clients))))
(progn
(setf (lsp--client-server-id client) docker-server-id
(lsp--client-uri->path-fn client) (-partial #'lsp-docker--uri->path
path-mappings
docker-container-name)
(lsp--client-activation-fn client) activation-fn
(lsp--client-path->uri-fn client) (-partial #'lsp-docker--path->uri path-mappings)
(lsp--client-new-connection client) (plist-put
(lsp-stdio-connection
(lambda ()
(funcall (or launch-server-cmd-fn #'lsp-docker-launch-new-container)
docker-container-name
path-mappings
launch-parameters
docker-image-id
server-command)))
:test? (lambda (&rest _)
t))
(lsp--client-priority client) (or priority (lsp--client-priority client)))
(lsp-register-client client)
(message "Registered a language server with id: %s and container name: %s" docker-server-id docker-container-name))
(user-error "No such client %s" server-id)))
(defun lsp-docker--register-single-server (server-config project-root path-mappings)
"Register a single dockerized language server.
Its description is provided via the SERVER-CONFIG hash-table. It
must represents the fields defined under the `server' (single
server configuration) or `multi-server/<dockerized-server-name>'
(multi-server configuration) node. The PROJECT-ROOT must be a
path pointing to the top-level folder of the project the
configuration file resides into. The PATH-MAPPINGS provides a
hash-table to translate the paths between the host and the
dockerized server."
(let* ((server-type-subtype (lsp-docker-get-server-type-subtype server-config))
(config-specified-server-container-name (lsp-docker-get-server-container-name server-config))
(server-image-name (lsp-docker-get-server-image-name server-config))
(regular-server-id (lsp-docker-get-server-id server-config))
(server-id (lsp-docker-generate-docker-server-id server-config (lsp-workspace-root)))
(server-launch-parameters (lsp-docker--get-server-launch-parameters server-config))
(server-launch-command (lsp-docker-get-launch-command server-config))
(base-client (lsp-docker--get-base-client regular-server-id))
(activation-fn (lsp-docker--create-activation-function-by-project-dir-and-base-client
(lsp-workspace-root)
base-client))
(server-container-name (lsp-docker--finalize-docker-server-container-name
config-specified-server-container-name server-config project-root)))
(if (and (lsp-docker-check-server-type-subtype lsp-docker-supported-server-types-subtypes
server-type-subtype)
(lsp-docker-check-path-mappings path-mappings))
(let ((container-type (car server-type-subtype))
(container-subtype (cdr server-type-subtype)))
(pcase container-type
('docker (pcase container-subtype
('image (if (lsp-docker--check-image-exists server-image-name)
(lsp-docker-register-client-with-activation-fn
:server-id regular-server-id
:docker-server-id server-id
:path-mappings path-mappings
:launch-parameters server-launch-parameters
:docker-image-id server-image-name
:docker-container-name server-container-name
:activation-fn activation-fn
:priority lsp-docker-default-priority
:server-command server-launch-command
:launch-server-cmd-fn #'lsp-docker-launch-new-container)
(lsp-docker--build-image-and-register-server-async
:image-name server-image-name
:dockerfile-path (lsp-docker--find-project-dockerfile-from-lsp)
:server-id regular-server-id
:docker-server-id server-id
:path-mappings path-mappings
:launch-parameters server-launch-parameters
:docker-container-name server-container-name
:activation-fn activation-fn
:server-command server-launch-command)))
('container (if (lsp-docker--check-container-exists server-container-name)
(lsp-docker-register-client-with-activation-fn
:server-id regular-server-id
:docker-server-id server-id
:path-mappings path-mappings
:launch-parameters nil
:docker-image-id nil
:docker-container-name server-container-name
:activation-fn activation-fn
:priority lsp-docker-default-priority
:server-command server-launch-command
:launch-server-cmd-fn #'lsp-docker-launch-existing-container)
(user-error "Invalid LSP docker config: cannot find the specified container: %s" server-container-name)))
(user-error "Invalid LSP docker config: unsupported server type and/or subtype")))
(user-error "Invalid LSP docker config: unsupported server type and/or subtype")))
(user-error "Language server registration failed, check input parameters"))))
(defun lsp-docker-register ()
"Register one or more dockerized language servers for the current project"
(interactive)
(if (lsp-workspace-root)
(let* ((config (lsp-docker-get-config-from-lsp))
(project-root (lsp-workspace-root))
(path-mappings (lsp-docker-get-path-mappings config (lsp-workspace-root)))
(server-config (gethash lsp-docker--server-key config)))
;; check whether a single or multiple servers are described in the configuration
(cond
((vectorp server-config)
(message "registering multiple servers")
;; NOTE: if multiple language server descriptions share the same name "server" field, the latest entry
;; will be enforced.
(--map (lsp-docker--register-single-server it
project-root
path-mappings)
server-config))
(server-config
(message "registering a single server")
(lsp-docker--register-single-server server-config
project-root
path-mappings))
(t
(user-error "no `%s' node found in configuration, see README for reference"
lsp-docker--server-key))))
(user-error
(format (concat "Current file: %s is not in a registered project! "
"Try adding your project with `lsp-workspace-folders-add'")
(buffer-file-name)))))
(defun lsp-docker-start ()
"Register and launch a server to use LSP mode in a container for the current project"
(interactive)
(lsp-docker-register)
(lsp))
(provide 'lsp-docker)
;;; lsp-docker.el ends here