-
Notifications
You must be signed in to change notification settings - Fork 78
/
docker-utils.el
185 lines (153 loc) · 7.28 KB
/
docker-utils.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
;;; docker-utils.el --- Random utilities -*- lexical-binding: t -*-
;; Author: Philippe Vaucher <philippe.vaucher@gmail.com>
;; This file is NOT part of GNU Emacs.
;; 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, 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; see the file COPYING. If not, write to the
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
;; Boston, MA 02110-1301, USA.
;;; Commentary:
;;; Code:
(require 's)
(require 'aio)
(require 'dash)
(require 'json)
(require 'tramp)
(require 'tablist)
(require 'transient)
(defun docker-utils-get-marked-items-ids ()
"Get the id part of `tablist-get-marked-items'."
(-map #'car (tablist-get-marked-items)))
(defun docker-utils-ensure-items ()
"Ensure at least one item is selected."
(when (null (docker-utils-get-marked-items-ids))
(user-error "This action cannot be used in an empty list")))
(defun docker-utils-generate-new-buffer-name (program &rest args)
"Wrapper around `generate-new-buffer-name' using PROGRAM and ARGS."
(generate-new-buffer-name (format "* %s %s *" program (s-join " " args))))
(defun docker-utils-generate-new-buffer (program &rest args)
"Wrapper around `generate-new-buffer' using PROGRAM and ARGS."
(generate-new-buffer (apply #'docker-utils-generate-new-buffer-name program args)))
(defmacro docker-utils-with-buffer (name &rest body)
"Wrapper around `with-current-buffer'.
Execute BODY in a buffer named with the help of NAME."
(declare (indent defun))
`(with-current-buffer (docker-utils-generate-new-buffer "docker" ,name)
(setq buffer-read-only nil)
(erase-buffer)
,@body
(setq buffer-read-only t)
(goto-char (point-min))
(pop-to-buffer (current-buffer))))
(defmacro docker-utils-transient-define-prefix (name arglist &rest args)
"Wrapper around `transient-define-prefix' forwarding NAME, ARGLIST and ARGS and calling `docker-utils-ensure-items'."
`(transient-define-prefix ,name ,arglist
,@args
(interactive)
(docker-utils-ensure-items)
(transient-setup ',name)))
(defmacro docker-utils-define-transient-arguments (name)
"Define the transient arguments function using NAME that return the latest transient value or its default."
`(defun ,(intern (format "%s-arguments" name)) ()
,(format "Return the latest used arguments in the `%s' transient." name)
(let ((history (alist-get ',name transient-history))
(default (transient-default-value (get ',name 'transient--prefix))))
(if (equal 0 (length history))
(car default)
(car history)))))
(defmacro docker-utils-refresh-entries (promise)
"Update the current buffer with the results of PROMISE."
`(let ((buffer (current-buffer))
(entries (aio-await ,promise)))
(with-current-buffer buffer
(setq tabulated-list-entries entries)
(tabulated-list-print t))))
(defvar docker-pop-to-buffer-action nil
"Action to use internally when `docker-utils-pop-to-buffer' calls `pop-to-buffer'.")
(defun docker-utils-pop-to-buffer (name)
"Like `pop-to-buffer', but suffix NAME with the host if on a remote host."
(pop-to-buffer
(if (file-remote-p default-directory)
(with-parsed-tramp-file-name default-directory nil (concat name " - " host))
name)
docker-pop-to-buffer-action))
(defun docker-utils-unit-multiplier (str)
"Return the correct multiplier for STR."
(expt 1024 (-elem-index (upcase str) '("B" "KB" "MB" "GB" "TB" "PB" "EB"))))
(defun docker-utils-human-size-to-bytes (str)
"Parse STR and return size in bytes."
(let* ((parts (s-match "^\\([0-9\\.]+\\)\\([A-Z]+\\)?$" str))
(value (string-to-number (-second-item parts)))
(multiplier (docker-utils-unit-multiplier (-third-item parts))))
(* value multiplier)))
(defun docker-utils-human-size-predicate (a b)
"Sort A and B by image size."
(< (docker-utils-human-size-to-bytes a) (docker-utils-human-size-to-bytes b)))
(defun docker-utils-columns-list-format (columns-spec)
"Convert COLUMNS-SPEC (a list of plists) to 'tabulated-list-format', i.e. a vector of (name width sort-fn)."
(apply 'vector
(--map-indexed
(-let* (((&plist :name name :width width :sort sort-fn-inner) it)
(sort-fn (if sort-fn-inner
(let ((idx it-index)) ;; Rebind the closure var!
;; Sort fn is called with (id [entries..])
;; Extract the column value and pass to inner function
(-on sort-fn-inner (lambda (x) (elt (cadr x) idx))))
t)))
(list name width sort-fn))
columns-spec)))
(defun docker-utils-make-format-string (id-template column-spec)
"Make the format string to pass to docker-ls commands.
ID-TEMPLATE is the Go template used to extract the property that
identifies the object (usually its id).
COLUMN-SPEC is the value of docker-X-columns."
(let* ((templates (--map (plist-get it :template) column-spec))
(delimited (string-join templates ",")))
(format "[%s,%s]" id-template delimited)))
(defun docker-utils-parse (column-specs line)
"Convert a LINE from \"docker ls\" to a `tabulated-list-entries' entry.
LINE is expected to be a JSON formatted array, and COLUMN-SPECS is the relevant
defcustom (e.g. `docker-image-columns`) used to apply any custom format functions."
(condition-case nil
(let* ((data (json-read-from-string line)))
;; apply format function, if any
(--each-indexed
column-specs
(let ((fmt-fn (plist-get it :format))
(data-index (+ it-index 1)))
(when fmt-fn (aset data data-index (apply fmt-fn (list (aref data data-index)))))))
(list (aref data 0) (seq-drop data 1)))
(json-readtable-error
(error "Could not read following string as json:\n%s" line))))
(defun docker-utils-columns-setter (sym new-value)
"Convert NEW-VALUE into a list of plists, then assign to SYM.
If NEW-VALUE already looks like a list of plists, no conversion is performed and
NEW-VALUE is assigned to SYM unchanged. This is expected to be used as the
value of :set in a defcustom."
(let ((is-plist (plist-member (car new-value) :name))
(new-value-plist (--map
(-interleave '(:name :width :template :sort :format) it)
new-value)))
(set sym (if is-plist new-value new-value-plist))))
(defun docker-utils-columns-getter (sym)
"Convert the value of SYM for displaying in the customization menu.
Just strips the plist symbols and returns only values.
This has no effect on the actual value of the variable."
(--map
(-map (-partial #'plist-get it) '(:name :width :template :sort :format))
(symbol-value sym)))
(defun docker-utils-package-p (package)
"Check if PACKAGE is available."
(or (featurep package)
(ignore-errors (require package))))
(provide 'docker-utils)
;;; docker-utils.el ends here