mirror of
https://github.com/Oxalide/vsphere-influxdb-go.git
synced 2023-10-10 13:36:51 +02:00
1568 lines
56 KiB
EmacsLisp
1568 lines
56 KiB
EmacsLisp
;;; govc.el --- Interface to govc for managing VMware ESXi and vCenter
|
||
|
||
;; Author: The govc developers
|
||
;; URL: https://github.com/vmware/govmomi/tree/master/govc/emacs
|
||
;; Keywords: convenience
|
||
;; Version: 0.14.0
|
||
;; Package-Requires: ((emacs "24.3") (dash "1.5.0") (s "1.9.0") (magit-popup "2.0.50") (json-mode "1.6.0"))
|
||
|
||
;; This file is NOT part of GNU Emacs.
|
||
|
||
;; Copyright (c) 2016 VMware, Inc. All Rights Reserved.
|
||
;;
|
||
;; Licensed under the Apache License, Version 2.0 (the "License");
|
||
;; you may not use this file except in compliance with the License.
|
||
;; You may obtain a copy of the License at
|
||
;;
|
||
;; http://www.apache.org/licenses/LICENSE-2.0
|
||
;;
|
||
;; Unless required by applicable law or agreed to in writing, software
|
||
;; distributed under the License is distributed on an "AS IS" BASIS,
|
||
;; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
;; See the License for the specific language governing permissions and
|
||
;; limitations under the License.
|
||
|
||
;;; Commentary:
|
||
|
||
;; The goal of this package is to provide a simple interface for commonly used
|
||
;; govc commands within Emacs. This includes table based inventory/state modes
|
||
;; for vms, hosts, datastores and pools. The keymap for each mode provides
|
||
;; shortcuts for easily feeding the data in view to other govc commands.
|
||
;;
|
||
;; Within the various govc modes, press `?' to see a popup menu of options.
|
||
;; A menu bar is enabled for certain modes, such as `govc-vm-mode' and `govc-host-mode'.
|
||
;; There is also a `govc' menu at all times under the `Tools' menu.
|
||
;;
|
||
;; The recommended way to install govc.el is via MELPA (http://melpa.org/).
|
||
|
||
;;; Code:
|
||
|
||
(eval-when-compile
|
||
(require 'cl))
|
||
(require 'dash)
|
||
(require 'diff)
|
||
(require 'dired)
|
||
(require 'json-mode)
|
||
(require 'magit-popup)
|
||
(require 'url-parse)
|
||
(require 's)
|
||
|
||
(autoload 'auth-source-search "auth-source")
|
||
|
||
(defgroup govc nil
|
||
"Emacs customization group for govc."
|
||
:group 'convenience)
|
||
|
||
(defcustom govc-keymap-prefix "C-c ;"
|
||
"Prefix for `govc-mode'."
|
||
:group 'govc)
|
||
|
||
(defcustom govc-command "govc"
|
||
"Executable path to the govc utility."
|
||
:type 'string
|
||
:group 'govc)
|
||
|
||
(defvar govc-command-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "h" 'govc-host)
|
||
(define-key map "p" 'govc-pool)
|
||
(define-key map "v" 'govc-vm)
|
||
(define-key map "s" 'govc-datastore)
|
||
(define-key map "?" 'govc-popup)
|
||
map)
|
||
"Keymap for `govc-mode' after `govc-keymap-prefix' was pressed.")
|
||
|
||
(defvar govc-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map (kbd govc-keymap-prefix) govc-command-map)
|
||
map)
|
||
"Keymap for `govc-mode'.")
|
||
|
||
;;;###autoload
|
||
(define-minor-mode govc-mode
|
||
"Running `govc-global-mode' creates key bindings to the various govc modes.
|
||
The default prefix is `C-c ;' and can be changed by setting `govc-keymap-prefix'.
|
||
|
||
\\{govc-mode-map\}"
|
||
nil govc-mode-line govc-mode-map
|
||
:group 'govc)
|
||
|
||
;;;###autoload
|
||
(define-globalized-minor-mode govc-global-mode govc-mode govc-mode)
|
||
|
||
(defcustom govc-mode-line
|
||
'(:eval (format " govc[%s]" (or (govc-session-name) "-")))
|
||
"Mode line lighter for govc."
|
||
:group 'govc
|
||
:type 'sexp
|
||
:risky t)
|
||
|
||
|
||
;;; Tabulated list mode extensions (derived from https://github.com/Silex/docker.el tabulated-list-ext.el)
|
||
(defun govc-tabulated-list-mark ()
|
||
"Mark and move to the next line."
|
||
(interactive)
|
||
(tabulated-list-put-tag (char-to-string dired-marker-char) t))
|
||
|
||
(defun govc-tabulated-list-unmark ()
|
||
"Unmark and move to the next line."
|
||
(interactive)
|
||
(tabulated-list-put-tag "" t))
|
||
|
||
(defun govc-tabulated-list-toggle-marks ()
|
||
"Toggle mark."
|
||
(interactive)
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(let ((cmd))
|
||
(while (not (eobp))
|
||
(setq cmd (char-after))
|
||
(tabulated-list-put-tag
|
||
(if (eq cmd dired-marker-char)
|
||
""
|
||
(char-to-string dired-marker-char)) t)))))
|
||
|
||
(defun govc-tabulated-list-unmark-all ()
|
||
"Unmark all."
|
||
(interactive)
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(while (not (eobp))
|
||
(tabulated-list-put-tag "" t))))
|
||
|
||
(defun govc-selection ()
|
||
"Get the current selection as a list of names."
|
||
(let ((selection))
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(while (not (eobp))
|
||
(when (eq (char-after) ?*)
|
||
(add-to-list 'selection (tabulated-list-get-id)))
|
||
(forward-line)))
|
||
(or selection (let ((id (tabulated-list-get-id)))
|
||
(if id
|
||
(list id))))))
|
||
|
||
(defun govc-do-selection (fn action)
|
||
"Call FN with `govc-selection' confirming ACTION."
|
||
(let* ((selection (govc-selection))
|
||
(count (length selection))
|
||
(prompt (if (= count 1)
|
||
(car selection)
|
||
(format "* [%d] marked" count))))
|
||
(if (yes-or-no-p (format "%s %s ?" action prompt))
|
||
(funcall fn selection))))
|
||
|
||
(defun govc-copy-selection ()
|
||
"Copy current selection or region to the kill ring."
|
||
(interactive)
|
||
(if (region-active-p)
|
||
(copy-region-as-kill (mark) (point) 'region)
|
||
(kill-new (message "%s" (s-join " " (--map (format "'%s'" it) (govc-selection)))))))
|
||
|
||
(defvar govc-font-lock-keywords
|
||
(list
|
||
(list dired-re-mark '(0 dired-mark-face))
|
||
(list "types.ManagedObjectReference\\(.*\\)" '(1 dired-directory-face))
|
||
(list "[^ ]*/$" '(0 dired-directory-face))
|
||
(list "\\.\\.\\.$" '(0 dired-symlink-face))))
|
||
|
||
(defvar govc-tabulated-list-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "m" 'govc-tabulated-list-mark)
|
||
(define-key map "u" 'govc-tabulated-list-unmark)
|
||
(define-key map "t" 'govc-tabulated-list-toggle-marks)
|
||
(define-key map "U" 'govc-tabulated-list-unmark-all)
|
||
(define-key map (kbd "M-&") 'govc-shell-command)
|
||
(define-key map (kbd "M-w") 'govc-copy-selection)
|
||
(define-key map (kbd "M-E") 'govc-copy-environment)
|
||
map)
|
||
"Keymap for `govc-tabulated-list-mode'.")
|
||
|
||
(define-derived-mode govc-tabulated-list-mode tabulated-list-mode "Tabulated govc"
|
||
"Generic table bindings to mark/unmark rows."
|
||
(setq-local font-lock-defaults
|
||
'(govc-font-lock-keywords t nil nil beginning-of-line)))
|
||
|
||
|
||
;;; Keymap helpers for generating menus and popups
|
||
(defun govc-keymap-list (keymap)
|
||
"Return a list of (key name function) for govc bindings in the given KEYMAP.
|
||
The name returned is the first word of the function `documentation'."
|
||
(let ((map))
|
||
(map-keymap
|
||
(lambda (k f)
|
||
(when (keymapp f)
|
||
(setq map (append map
|
||
(--map (and (setcar it (kbd (format "M-%s" (char-to-string (car it))))) it)
|
||
(govc-keymap-list f)))))
|
||
(when (and (symbolp f)
|
||
(s-starts-with? "govc-" (symbol-name f)))
|
||
(if (not (eq ?? k))
|
||
(add-to-list 'map (list k (car (split-string (documentation f))) f))))) keymap)
|
||
map))
|
||
|
||
(defun govc-keymap-menu (keymap)
|
||
"Return a list of [key function t] for govc bindings in the given KEYMAP.
|
||
For use with `easy-menu-define'."
|
||
(-map (lambda (item)
|
||
(vector (nth 1 item) (nth 2 item) t))
|
||
(govc-keymap-list keymap)))
|
||
|
||
(defun govc-key-description (key)
|
||
"Call `key-description' ensuring KEY is a sequence."
|
||
(key-description (if (integerp key) (list key) key)))
|
||
|
||
(defun govc-keymap-list-to-help (keymap)
|
||
"Convert KEYMAP to list of help text."
|
||
(--map (list (govc-key-description (car it))
|
||
(car (split-string (documentation (nth 2 it)) "\\.")))
|
||
keymap))
|
||
|
||
(defun govc-keymap-popup-help ()
|
||
"Default keymap help for `govc-keymap-popup'."
|
||
(append (govc-keymap-list-to-help (govc-keymap-list govc-tabulated-list-mode-map))
|
||
'(("g" "Refresh current buffer")
|
||
("C-h m" "Show all key bindings"))))
|
||
|
||
(defun govc-keymap-popup (keymap)
|
||
"Convert a `govc-keymap-list' using KEYMAP for use with `magit-define-popup'.
|
||
Keys in the ASCII range of 32-97 are mapped to popup commands, all others are listed as help text."
|
||
(let* ((maps (--separate (and (integerp (car it))
|
||
(>= (car it) 32)
|
||
(<= (car it) 97))
|
||
(govc-keymap-list keymap)))
|
||
(help (govc-keymap-list-to-help (cadr maps))))
|
||
(append
|
||
'("Commands")
|
||
(car maps)
|
||
(list (s-join "\n" (--map (format " %-6s %s" (car it) (cadr it))
|
||
(append help (govc-keymap-popup-help))))
|
||
nil))))
|
||
|
||
|
||
;;; govc process helpers
|
||
(defcustom govc-urls nil
|
||
"List of URLs for use with `govc-session'.
|
||
The `govc-session-name' displayed by `govc-mode-line' uses `url-target' (anchor)
|
||
if set, otherwise `url-host' is used.
|
||
|
||
Example:
|
||
```
|
||
(setq govc-urls '(\"root:vagrant@localhost:18443#Vagrant-ESXi\"
|
||
\"root:password@192.168.1.192#Intel-NUC\"
|
||
\"Administrator@vsphere.local:password!@vcva-clovervm\"))
|
||
```
|
||
To enter a URL that is not in the list, prefix `universal-argument', for example:
|
||
|
||
`\\[universal-argument] \\[govc-vm]'
|
||
|
||
To avoid putting your credentials in a variable, you can use the
|
||
auth-source search integration.
|
||
|
||
```
|
||
(setq govc-urls '(\"myserver-vmware-2\"))
|
||
```
|
||
|
||
And then put this line in your `auth-sources' (e.g. `~/.authinfo.gpg'):
|
||
```
|
||
machine myserver-vmware-2 login tzz password mypass url \"myserver-vmware-2.some.domain.here:443?insecure=true\"
|
||
```
|
||
|
||
Which will result in the URL \"tzz:mypass@myserver-vmware-2.some.domain.here:443?insecure=true\".
|
||
For more details on `auth-sources', see Info node `(auth) Help for users'.
|
||
|
||
When in `govc-vm' or `govc-host' mode, a default URL is composed with the
|
||
current session credentials and the IP address of the current vm/host and
|
||
the vm/host name as the session name. This makes it easier to connect to
|
||
nested ESX/vCenter VMs or directly to an ESX host."
|
||
:group 'govc
|
||
:type '(repeat (string :tag "vcenter URL or auth-source machine reference")))
|
||
|
||
(defvar-local govc-session-url nil
|
||
"ESX or vCenter URL set by `govc-session' via `govc-urls' selection.")
|
||
|
||
(defvar-local govc-session-path nil)
|
||
|
||
(defvar-local govc-session-insecure nil
|
||
"Skip verification of server certificate when true.
|
||
This variable is set to the value of the `GOVC_INSECURE' env var by default.
|
||
It can also be set per-url via the query string (insecure=true). For example:
|
||
```
|
||
(setq govc-urls '(\"root:password@hostname?insecure=true\"))
|
||
```")
|
||
|
||
(defvar-local govc-session-datacenter nil
|
||
"Datacenter to use for the current `govc-session'.
|
||
If the endpoint has a single Datacenter it will be used by default, otherwise
|
||
`govc-session' will prompt for selection. It can also be set per-url via the
|
||
query string. For example:
|
||
```
|
||
(setq govc-urls '(\"root:password@hostname?datacenter=dc1\"))
|
||
```")
|
||
|
||
(defvar-local govc-session-datastore nil
|
||
"Datastore to use for the current `govc-session'.
|
||
If the endpoint has a single Datastore it will be used by default, otherwise
|
||
`govc-session' will prompt for selection. It can also be set per-url via the
|
||
query string. For example:
|
||
```
|
||
(setq govc-urls '(\"root:password@hostname?datastore=vsanDatastore\"))
|
||
```")
|
||
|
||
(defvar-local govc-session-network nil
|
||
"Network to use for the current `govc-session'.")
|
||
|
||
(defvar-local govc-filter nil
|
||
"Resource path filter.")
|
||
|
||
(defvar-local govc-args nil
|
||
"Additional govc arguments.")
|
||
|
||
(defun govc-session-name ()
|
||
"Return a name for the current session.
|
||
Derived from `govc-session-url' if set, otherwise from the 'GOVC_URL' env var.
|
||
Return value is the url anchor if set, otherwise the hostname is returned."
|
||
(let* ((u (or govc-session-url (getenv "GOVC_URL")))
|
||
(url (if u (govc-url-parse u))))
|
||
(if url
|
||
(concat (or (url-target url) (url-host url)) govc-session-path))))
|
||
|
||
(defun govc-format-command (command &rest args)
|
||
"Format govc COMMAND ARGS."
|
||
(format "%s %s %s" govc-command command
|
||
(s-join " " (--map (format "'%s'" it)
|
||
(-flatten (-non-nil args))))))
|
||
|
||
(defconst govc-environment-map (--map (cons (concat "GOVC_" (upcase it))
|
||
(intern (concat "govc-session-" it)))
|
||
'("url" "insecure" "datacenter" "datastore" "network"))
|
||
|
||
"Map of `GOVC_*' environment variable names to `govc-session-*' symbol names.")
|
||
|
||
(defun govc-environment (&optional unset)
|
||
"Return `process-environment' for govc.
|
||
Optionally clear govc env if UNSET is non-nil."
|
||
(let ((process-environment (copy-sequence process-environment)))
|
||
(dolist (e govc-environment-map)
|
||
(setenv (car e) (unless unset (symbol-value (cdr e)))))
|
||
process-environment))
|
||
|
||
(defun govc-export-environment (arg)
|
||
"Set if ARG is \\[universal-argument], unset if ARG is \\[negative-argument]."
|
||
(if (equal arg '-)
|
||
(progn (setq process-environment (govc-environment t))
|
||
(cons "unset" (--map (car it)
|
||
govc-environment-map)))
|
||
(progn (setq process-environment (govc-environment))
|
||
(cons "export" (--map (format "%s='%s'" (car it) (or (symbol-value (cdr it)) ""))
|
||
govc-environment-map)))))
|
||
|
||
(defun govc-copy-environment (&optional arg)
|
||
"Export session to `process-environment' and `kill-ring'.
|
||
Optionally set `GOVC_*' vars in `process-environment' using prefix
|
||
\\[universal-argument] ARG or unset with prefix \\[negative-argument] ARG."
|
||
(interactive "P")
|
||
(message (kill-new (if arg (s-join " " (govc-export-environment arg)) govc-session-url))))
|
||
|
||
(defun govc-process (command handler)
|
||
"Run COMMAND, calling HANDLER upon successful exit of the process."
|
||
(message command)
|
||
(let ((process-environment (govc-environment))
|
||
(exit-code))
|
||
(add-to-list 'govc-command-history command)
|
||
(with-temp-buffer
|
||
(setq exit-code (call-process-shell-command command nil (current-buffer)))
|
||
(if (zerop exit-code)
|
||
(funcall handler)
|
||
(error (buffer-string))))))
|
||
|
||
(defun govc (command &rest args)
|
||
"Execute govc COMMAND with ARGS.
|
||
Return value is `buffer-string' split on newlines."
|
||
(govc-process (govc-format-command command args)
|
||
(lambda ()
|
||
(split-string (buffer-string) "\n" t))))
|
||
|
||
(defun govc-json (command &rest args)
|
||
"Execute govc COMMAND passing arguments ARGS.
|
||
Return value is `json-read'."
|
||
(govc-process (govc-format-command command (cons "-json" args))
|
||
(lambda ()
|
||
(goto-char (point-min))
|
||
(let ((json-object-type 'plist))
|
||
(json-read)))))
|
||
|
||
(defun govc-ls-datacenter ()
|
||
"List datacenters."
|
||
(govc "ls" "-t" "Datacenter" "./..."))
|
||
|
||
(defun govc-object-prompt (prompt ls)
|
||
"PROMPT for object name via LS function. Return object without PROMPT if there is just one instance."
|
||
(let ((objs (if (listp ls) ls (funcall ls))))
|
||
(if (eq 1 (length objs))
|
||
(car objs)
|
||
(completing-read prompt objs))))
|
||
|
||
(defun govc-url-parse (url)
|
||
"A `url-generic-parse-url' wrapper to handle URL with password, but no scheme.
|
||
Also fixes the case where user contains an '@'."
|
||
(let* ((full (s-contains? "://" url))
|
||
(u (url-generic-parse-url (concat (unless full "https://") url))))
|
||
(unless full
|
||
(setf (url-type u) nil)
|
||
(setf (url-fullness u) nil))
|
||
(if (s-contains? "@" (url-host u))
|
||
(let* ((h (split-string (url-host u) "@"))
|
||
(p (split-string (car h) ":")))
|
||
(setf (url-host u) (cadr h))
|
||
(setf (url-user u) (concat (url-user u) "@" (car p)))
|
||
(setf (url-password u) (cadr p))))
|
||
u))
|
||
|
||
(defun govc-url-default ()
|
||
"Default URL when creating a new session."
|
||
(if govc-session-url
|
||
(let ((url (govc-url-parse govc-session-url)))
|
||
(if (equal major-mode 'govc-host-mode)
|
||
(progn (setf (url-host url) (govc-table-column-value "Name"))
|
||
(setf (url-target url) nil))
|
||
(progn (setf (url-host url) (govc-table-column-value "IP address"))
|
||
(setf (url-target url) (govc-table-column-value "Name"))))
|
||
(setf (url-filename url) "") ; erase query string
|
||
(if (string-empty-p (url-user url))
|
||
(setf (url-user url) "root")) ; local workstation url has no user set
|
||
(url-recreate-url url))))
|
||
|
||
(defun govc-urls-completing-read ()
|
||
"A wrapper for `completing-read' to mask credentials in `govc-urls'."
|
||
(let ((alist))
|
||
(dolist (ent govc-urls)
|
||
(let ((u (govc-url-parse ent)))
|
||
(setf (url-password u) nil)
|
||
(add-to-list 'alist `(,(url-recreate-url u) . ,ent) t)))
|
||
(let ((u (completing-read "govc url: " (-map 'car alist))))
|
||
(cdr (assoc u alist)))))
|
||
|
||
(defun govc-session-url-lookup-auth-source (url-or-address)
|
||
"Check if URL-OR-ADDRESS is a logical name in the authinfo file.
|
||
Given URL-OR-ADDRESS `myserver-vmware-2' this function will find
|
||
a line like
|
||
machine myserver-vmware-2 login tzz password mypass url \"myserver-vmware-2.some.domain.here:443?insecure=true\"
|
||
|
||
and will return the URL \"tzz:mypass@myserver-vmware-2.some.domain.here:443?insecure=true\".
|
||
|
||
If the line is not found, the original URL-OR-ADDRESS is
|
||
returned, assuming that's what the user wanted."
|
||
(let ((found (nth 0 (auth-source-search :max 1
|
||
:host url-or-address
|
||
:require '(:user :secret :url)
|
||
:create nil))))
|
||
(if found
|
||
(format "%s:%s@%s"
|
||
(plist-get found :user)
|
||
(let ((secret (plist-get found :secret)))
|
||
(if (functionp secret)
|
||
(funcall secret)
|
||
secret))
|
||
(plist-get found :url))
|
||
url-or-address)))
|
||
|
||
(defun govc-session-set-url (url)
|
||
"Set `govc-session-url' to URL and optionally set other govc-session-* variables via URL query."
|
||
;; Replace the original URL with the auth-source lookup if there is no user.
|
||
(unless (url-user (govc-url-parse url))
|
||
(setq url (govc-session-url-lookup-auth-source url)))
|
||
|
||
(let ((q (cdr (url-path-and-query (govc-url-parse url)))))
|
||
(dolist (opt (if q (url-parse-query-string q)))
|
||
(let ((var (intern (concat "govc-session-" (car opt)))))
|
||
(if (boundp var)
|
||
(set var (cadr opt))))))
|
||
(setq govc-session-url url))
|
||
|
||
(defun govc-session ()
|
||
"Initialize a govc session."
|
||
(interactive)
|
||
(let ((url (if (or current-prefix-arg (eq 0 (length govc-urls)))
|
||
(read-string "govc url: " (govc-url-default))
|
||
(if (eq 1 (length govc-urls))
|
||
(car govc-urls)
|
||
(govc-urls-completing-read)))))
|
||
;; Wait until this point to clear so current session is preserved in the
|
||
;; event of `keyboard-quit' in `read-string'.
|
||
(setq govc-session-datacenter nil
|
||
govc-session-datastore nil
|
||
govc-session-network nil
|
||
govc-filter nil)
|
||
(govc-session-set-url url))
|
||
(unless govc-session-insecure
|
||
(setq govc-session-insecure (or (getenv "GOVC_INSECURE")
|
||
(completing-read "govc insecure: " '("true" "false")))))
|
||
(unless govc-session-datacenter
|
||
(setq govc-session-datacenter (govc-object-prompt "govc datacenter: " 'govc-ls-datacenter)))
|
||
(add-to-list 'govc-urls govc-session-url))
|
||
|
||
(defalias 'govc-current-session 'buffer-local-variables)
|
||
|
||
(defun govc-session-clone (session)
|
||
"Clone a session from SESSION buffer locals."
|
||
(dolist (v session)
|
||
(let ((s (car v)))
|
||
(when (s-starts-with? "govc-session-" (symbol-name s))
|
||
(set s (assoc-default s session))))))
|
||
|
||
(defvar govc-command-history nil
|
||
"History list for govc commands used by `govc-shell-command'.")
|
||
|
||
(defun govc-shell-command (&optional cmd)
|
||
"Shell CMD with current `govc-session' exported as GOVC_ env vars."
|
||
(interactive)
|
||
(let ((process-environment (govc-environment))
|
||
(current-prefix-arg "*govc*")
|
||
(url govc-session-url)
|
||
(shell-command-history govc-command-history))
|
||
(if cmd
|
||
(async-shell-command cmd current-prefix-arg)
|
||
(call-interactively 'async-shell-command))
|
||
(setq govc-command-history shell-command-history)
|
||
(with-current-buffer (get-buffer current-prefix-arg)
|
||
(setq govc-session-url url))))
|
||
|
||
(defcustom govc-max-events 100
|
||
"Limit events output to the last N events."
|
||
:type 'integer
|
||
:group 'govc)
|
||
|
||
(defun govc-events ()
|
||
"Events via govc events -n `govc-max-events'."
|
||
(interactive)
|
||
(govc-shell-command
|
||
(govc-format-command "events"
|
||
(list "-n" govc-max-events (if current-prefix-arg "-f") (govc-selection)))))
|
||
|
||
(defun govc-logs ()
|
||
"Logs via govc logs -n `govc-max-events'."
|
||
(interactive)
|
||
(govc-shell-command
|
||
(let ((host (govc-selection)))
|
||
(govc-format-command "logs"
|
||
(list "-n" govc-max-events (if current-prefix-arg "-f") (if host (list "-host" host)))))))
|
||
|
||
(defun govc-parse-info (output)
|
||
"Parse govc info command OUTPUT."
|
||
(let* ((entries)
|
||
(entry)
|
||
(entry-key))
|
||
(-each output
|
||
(lambda (line)
|
||
(let* ((ix (s-index-of ":" line))
|
||
(key (s-trim (substring line 0 ix)))
|
||
(val (s-trim (substring line (+ ix 1)))))
|
||
(unless entry-key
|
||
(setq entry-key key))
|
||
(when (s-equals? key entry-key)
|
||
(setq entry (make-hash-table :test 'equal))
|
||
(add-to-list 'entries entry))
|
||
(puthash key val entry))))
|
||
entries))
|
||
|
||
(defun govc-table-column-names ()
|
||
"Return a list of column names from `tabulated-list-format'."
|
||
(--map (car (aref tabulated-list-format it))
|
||
(number-sequence 0 (- (length tabulated-list-format) 1))))
|
||
|
||
(defun govc-table-column-value (key)
|
||
"Return current column value for given KEY."
|
||
(let ((names (govc-table-column-names))
|
||
(entry (tabulated-list-get-entry))
|
||
(value))
|
||
(dotimes (ix (- (length names) 1))
|
||
(if (s-equals? key (nth ix names))
|
||
(setq value (elt entry ix))))
|
||
value))
|
||
|
||
(defun govc-table-info (command &optional args)
|
||
"Convert `govc-parse-info' COMMAND ARGS output to `tabulated-list-entries' format."
|
||
(let ((names (govc-table-column-names)))
|
||
(-map (lambda (info)
|
||
(let ((id (or (gethash "Path" info)
|
||
(gethash (car names) info))))
|
||
(list id (vconcat
|
||
(--map (or (gethash it info) "-")
|
||
names)))))
|
||
(govc-parse-info (govc command args)))))
|
||
|
||
(defun govc-map-info (command &optional args)
|
||
"Populate key=val map table with govc COMMAND ARGS output."
|
||
(-map (lambda (line)
|
||
(let* ((ix (s-index-of ":" line))
|
||
(key (s-trim (substring line 0 ix)))
|
||
(val (s-trim (substring line (+ ix 1)))))
|
||
(list key (vector key val))))
|
||
(govc command args)))
|
||
|
||
(defun govc-map-info-table (entries)
|
||
"Tabulated `govc-map-info' data via ENTRIES."
|
||
(let ((session (govc-current-session))
|
||
(args (append govc-args (govc-selection)))
|
||
(buffer (get-buffer-create "*govc-info*")))
|
||
(pop-to-buffer buffer)
|
||
(tabulated-list-mode)
|
||
(setq govc-args args)
|
||
(govc-session-clone session)
|
||
(setq tabulated-list-format [("Name" 50)
|
||
("Value" 50)]
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries entries)
|
||
(tabulated-list-print)))
|
||
|
||
(defun govc-type-list-entries (command)
|
||
"Convert govc COMMAND type table output to `tabulated-list-entries'."
|
||
(-map (lambda (line)
|
||
(let* ((entry (s-split-up-to " " (s-collapse-whitespace line) 2))
|
||
(name (car entry))
|
||
(type (nth 1 entry))
|
||
(value (car (last entry))))
|
||
(list name (vector name type value))))
|
||
(govc command govc-args)))
|
||
|
||
(defun govc-json-info-selection (command)
|
||
"Run govc COMMAND -json on `govc-selection'."
|
||
(if current-prefix-arg
|
||
(--each (govc-selection) (govc-json-info command it))
|
||
(govc-json-info command (govc-selection))))
|
||
|
||
(defun govc-json-diff ()
|
||
"Diff two *govc-json* buffers in view."
|
||
(let ((buffers))
|
||
(-each (window-list-1)
|
||
(lambda (w)
|
||
(with-current-buffer (window-buffer w)
|
||
(if (and (eq major-mode 'json-mode)
|
||
(s-starts-with? "*govc-json*" (buffer-name)))
|
||
(push (current-buffer) buffers)))) )
|
||
(if (= (length buffers) 2)
|
||
(pop-to-buffer
|
||
(diff-no-select (car buffers) (cadr buffers))))))
|
||
|
||
(defun govc-json-info (command selection)
|
||
"Run govc COMMAND -json on SELECTION."
|
||
(govc-process (govc-format-command command "-json" govc-args selection)
|
||
(lambda ()
|
||
(let ((buffer (get-buffer-create (concat "*govc-json*" (if current-prefix-arg selection)))))
|
||
(copy-to-buffer buffer (point-min) (point-max))
|
||
(with-current-buffer buffer
|
||
(json-mode)
|
||
;; We use `json-mode-beautify' as `json-pretty-print-buffer' does not work for `govc-host-json-info'
|
||
(json-mode-beautify))
|
||
(display-buffer buffer))))
|
||
(if current-prefix-arg
|
||
(govc-json-diff)))
|
||
|
||
(defun govc-mode-new-session ()
|
||
"Connect new session for the current govc mode."
|
||
(interactive)
|
||
(call-interactively 'govc-session)
|
||
(revert-buffer))
|
||
|
||
(defun govc-host-with-session ()
|
||
"Host-mode with current session."
|
||
(interactive)
|
||
(govc-host nil (govc-current-session)))
|
||
|
||
(defun govc-vm-with-session ()
|
||
"VM-mode with current session."
|
||
(interactive)
|
||
(govc-vm nil (govc-current-session)))
|
||
|
||
(defun govc-datastore-with-session ()
|
||
"Datastore-mode with current session."
|
||
(interactive)
|
||
(govc-datastore nil (govc-current-session)))
|
||
|
||
(defun govc-pool-with-session ()
|
||
"Pool-mode with current session."
|
||
(interactive)
|
||
(govc-pool nil (govc-current-session)))
|
||
|
||
|
||
;;; govc object mode
|
||
(defvar-local govc-object-history '("-")
|
||
"History list of visited objects.")
|
||
|
||
(defun govc-object-collect ()
|
||
"Wrapper for govc object.collect."
|
||
(interactive)
|
||
(let ((id (car govc-args)))
|
||
(add-to-list 'govc-object-history id)
|
||
(setq govc-session-path id))
|
||
(govc-type-list-entries "object.collect"))
|
||
|
||
(defun govc-object-collect-selection (&optional json)
|
||
"Expand object selection via govc object.collect.
|
||
Optionally specify JSON encoding."
|
||
(interactive)
|
||
(let* ((entry (or (tabulated-list-get-entry) (error "No entry")))
|
||
(name (elt entry 0))
|
||
(type (elt entry 1))
|
||
(val (elt entry 2)))
|
||
|
||
(setq govc-args (list (car govc-args) name))
|
||
|
||
(cond
|
||
((s-blank? val))
|
||
((and (not json) (s-ends-with? "types.ManagedObjectReference" type))
|
||
(let ((ids (govc "ls" "-L" (split-string val ","))))
|
||
(setq govc-args (list (govc-object-prompt "moid: " ids)))))
|
||
((string= val "...")
|
||
(if (s-starts-with? "[]" type) (setq json t))))
|
||
|
||
(if json
|
||
(govc-json-info "object.collect" nil)
|
||
(tabulated-list-revert))))
|
||
|
||
(defun govc-object-collect-selection-json ()
|
||
"JSON object selection via govc object.collect."
|
||
(interactive)
|
||
(govc-object-collect-selection t))
|
||
|
||
(defun govc-object-next ()
|
||
"Next managed object reference."
|
||
(interactive)
|
||
(if (search-forward "types.ManagedObjectReference" nil t)
|
||
(progn (govc-tabulated-list-unmark-all)
|
||
(tabulated-list-put-tag (char-to-string dired-marker-char)))
|
||
(goto-char (point-min))))
|
||
|
||
(defun govc-object-collect-parent ()
|
||
"Parent object selection if reachable, otherwise prompt with `govc-object-history'."
|
||
(interactive)
|
||
(if (cadr govc-args)
|
||
(let ((prop (butlast (split-string (cadr govc-args) "\\."))))
|
||
(setq govc-args (list (car govc-args) (if prop (s-join "." prop)))))
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(if (re-search-forward "^[[:space:]]*parent" nil t)
|
||
(govc-object-collect-selection)
|
||
(let ((id (govc-object-prompt "moid: " govc-object-history)))
|
||
(setq govc-args (list id (if (string= id "-") "content")))))))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-object (&optional moid property session)
|
||
"Object browser aka MOB (Managed Object Browser).
|
||
Optionally starting at MOID and PROPERTY if given.
|
||
Inherit SESSION if given."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-object*")))
|
||
(if (called-interactively-p 'interactive)
|
||
(switch-to-buffer buffer)
|
||
(pop-to-buffer buffer))
|
||
(govc-object-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-args (list (or moid "-") property))
|
||
(tabulated-list-print)))
|
||
|
||
(defun govc-object-info ()
|
||
"Object browser via govc object.collect on `govc-selection'."
|
||
(interactive)
|
||
(if (equal major-mode 'govc-object-mode)
|
||
(progn
|
||
(setq govc-args (list (govc-object-prompt "moid: " govc-object-history)))
|
||
(tabulated-list-revert))
|
||
(govc-object (tabulated-list-get-id) nil (govc-current-session))))
|
||
|
||
(defvar govc-object-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "J" 'govc-object-collect-selection-json)
|
||
(define-key map "N" 'govc-object-next)
|
||
(define-key map "O" 'govc-object-info)
|
||
(define-key map (kbd "DEL") 'govc-object-collect-parent)
|
||
(define-key map (kbd "RET") 'govc-object-collect-selection)
|
||
(define-key map "?" 'govc-object-popup)
|
||
map)
|
||
"Keymap for `govc-object-mode'.")
|
||
|
||
(define-derived-mode govc-object-mode govc-tabulated-list-mode "Object"
|
||
"Major mode for handling a govc object."
|
||
(setq tabulated-list-format [("Name" 40 t)
|
||
("Type" 40 t)
|
||
("Value" 40 t)]
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-object-collect)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-object-popup
|
||
"Object popup."
|
||
:actions (govc-keymap-popup govc-object-mode-map))
|
||
|
||
|
||
;;; govc metric mode
|
||
(defun govc-metric-sample ()
|
||
"Sample metrics."
|
||
(interactive)
|
||
(govc-shell-command (govc-format-command "metric.sample" govc-args govc-filter (govc-selection))))
|
||
|
||
(defun govc-metric-sample-plot ()
|
||
"Plot metric sample."
|
||
(interactive)
|
||
(let* ((type (if (and (display-images-p) (not (eq current-prefix-arg '-))) 'png 'dumb))
|
||
(max (if (member "-i" govc-args) "60" "180"))
|
||
(args (append govc-args (list "-n" max "-plot" type govc-filter)))
|
||
(session (govc-current-session))
|
||
(metrics (govc-selection)))
|
||
(with-current-buffer (get-buffer-create "*govc*")
|
||
(govc-session-clone session)
|
||
(erase-buffer)
|
||
(delete-other-windows)
|
||
(if (eq type 'dumb)
|
||
(split-window-right)
|
||
(split-window-below))
|
||
(display-buffer-use-some-window (current-buffer) '((inhibit-same-window . t)))
|
||
(--each metrics
|
||
(let* ((cmd (govc-format-command "metric.sample" args it))
|
||
(data (govc-process cmd 'buffer-string)))
|
||
(if (eq type 'dumb)
|
||
(insert data)
|
||
(insert-image (create-image (string-as-unibyte data) type t))))))))
|
||
|
||
(defun govc-metric-select (metrics)
|
||
"Select metric names. METRICS is a regexp."
|
||
(interactive (list (read-regexp "Select metrics" (regexp-quote ".usage."))))
|
||
(save-excursion
|
||
(goto-char (point-min))
|
||
(while (not (eobp))
|
||
(if (string-match-p metrics (tabulated-list-get-id))
|
||
(govc-tabulated-list-mark)
|
||
(govc-tabulated-list-unmark)))))
|
||
|
||
(defun govc-metric-info ()
|
||
"Wrapper for govc metric.info."
|
||
(govc-table-info "metric.info" (list govc-args (car govc-filter))))
|
||
|
||
(defvar govc-metric-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map (kbd "RET") 'govc-metric-sample)
|
||
(define-key map (kbd "P") 'govc-metric-sample-plot)
|
||
(define-key map (kbd "s") 'govc-metric-select)
|
||
map)
|
||
"Keymap for `govc-metric-mode'.")
|
||
|
||
(defun govc-metric ()
|
||
"Metrics info."
|
||
(interactive)
|
||
(let ((session (govc-current-session))
|
||
(filter (or (govc-selection) (list govc-session-path)))
|
||
(buffer (get-buffer-create "*govc-metric*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-metric-mode)
|
||
(govc-session-clone session)
|
||
(if current-prefix-arg (setq govc-args '("-i" "300")))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-metric-mode govc-tabulated-list-mode "Metric"
|
||
"Major mode for handling a govc metric."
|
||
(setq tabulated-list-format [("Name" 35 t)
|
||
("Group" 15 t)
|
||
("Unit" 4 t)
|
||
("Level" 5 t)
|
||
("Summary" 50)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-metric-info)
|
||
(tabulated-list-init-header))
|
||
|
||
|
||
;;; govc host mode
|
||
(defun govc-ls-host ()
|
||
"List hosts."
|
||
(govc "ls" "-t" "HostSystem" "host/*"))
|
||
|
||
(defun govc-esxcli-netstat-info ()
|
||
"Wrapper for govc host.esxcli network ip connection list."
|
||
(govc-table-info "host.esxcli"
|
||
(append govc-args '("-hints=false" "--" "network" "ip" "connection" "list"))))
|
||
|
||
(defun govc-esxcli-netstat (host)
|
||
"Tabulated `govc-esxcli-netstat-info' HOST."
|
||
(interactive (list (govc-object-prompt "Host: " 'govc-ls-host)))
|
||
(let ((session (govc-current-session))
|
||
(buffer (get-buffer-create "*govc-esxcli*")))
|
||
(pop-to-buffer buffer)
|
||
(tabulated-list-mode)
|
||
(setq govc-args (list "-host" host))
|
||
(govc-session-clone session)
|
||
(setq tabulated-list-format [("CCAlgo" 10 t)
|
||
("ForeignAddress" 20 t)
|
||
("LocalAddress" 20 t)
|
||
("Proto" 5 t)
|
||
("RecvQ" 5 t)
|
||
("SendQ" 5 t)
|
||
("State" 15 t)
|
||
("WorldID" 7 t)
|
||
("WorldName" 10 t)]
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-esxcli-netstat-info)
|
||
(tabulated-list-init-header)
|
||
(tabulated-list-print)))
|
||
|
||
(defun govc-host-esxcli-netstat ()
|
||
"Netstat via `govc-esxcli-netstat-info' with current host id."
|
||
(interactive)
|
||
(govc-esxcli-netstat (tabulated-list-get-id)))
|
||
|
||
(defun govc-host-info ()
|
||
"Wrapper for govc host.info."
|
||
(govc-table-info "host.info" (or govc-filter "*")))
|
||
|
||
(defun govc-host-json-info ()
|
||
"JSON via govc host.info -json on current selection."
|
||
(interactive)
|
||
(govc-json-info-selection "host.info"))
|
||
|
||
(defvar govc-host-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "E" 'govc-events)
|
||
(define-key map "L" 'govc-logs)
|
||
(define-key map "J" 'govc-host-json-info)
|
||
(define-key map "M" 'govc-metric)
|
||
(define-key map "N" 'govc-host-esxcli-netstat)
|
||
(define-key map "O" 'govc-object-info)
|
||
(define-key map "c" 'govc-mode-new-session)
|
||
(define-key map "p" 'govc-pool-with-session)
|
||
(define-key map "s" 'govc-datastore-with-session)
|
||
(define-key map "v" 'govc-vm-with-session)
|
||
(define-key map "?" 'govc-host-popup)
|
||
map)
|
||
"Keymap for `govc-host-mode'.")
|
||
|
||
(defun govc-host (&optional filter session)
|
||
"Host info via govc.
|
||
Optionally filter by FILTER and inherit SESSION."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-host*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-host-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-host-mode govc-tabulated-list-mode "Host"
|
||
"Major mode for handling a list of govc hosts."
|
||
(setq tabulated-list-format [("Name" 30 t)
|
||
("Logical CPUs" 20 t)
|
||
("CPU usage" 25 t)
|
||
("Memory" 10 t)
|
||
("Memory usage" 25 t)
|
||
("Manufacturer" 13 t)
|
||
("Boot time" 15 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-host-info)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-host-popup
|
||
"Host popup."
|
||
:actions (govc-keymap-popup govc-host-mode-map))
|
||
|
||
(easy-menu-define govc-host-mode-menu govc-host-mode-map
|
||
"Host menu."
|
||
(cons "Host" (govc-keymap-menu govc-host-mode-map)))
|
||
|
||
|
||
;;; govc pool mode
|
||
(defun govc-pool-destroy (name)
|
||
"Destroy pool with given NAME."
|
||
(interactive (list (completing-read "Destroy pool: " (govc "ls" "-t" "ResourcePool" "host/*"))))
|
||
(govc "pool.destroy" name))
|
||
|
||
(defun govc-pool-destroy-selection ()
|
||
"Destroy via `govc-pool-destroy' on the pool selection."
|
||
(interactive)
|
||
(govc-do-selection 'govc-pool-destroy "Delete")
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-pool-info ()
|
||
"Wrapper for govc pool.info."
|
||
(govc-table-info "pool.info" (list "-a" (or govc-filter (setq govc-filter "*")))))
|
||
|
||
(defun govc-pool-json-info ()
|
||
"JSON via govc pool.info -json on current selection."
|
||
(interactive)
|
||
(govc-json-info-selection "pool.info"))
|
||
|
||
(defvar govc-pool-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "D" 'govc-pool-destroy-selection)
|
||
(define-key map "E" 'govc-events)
|
||
(define-key map "J" 'govc-pool-json-info)
|
||
(define-key map "M" 'govc-metric)
|
||
(define-key map "O" 'govc-object-info)
|
||
(define-key map "c" 'govc-mode-new-session)
|
||
(define-key map "h" 'govc-host-with-session)
|
||
(define-key map "s" 'govc-datastore-with-session)
|
||
(define-key map "v" 'govc-vm-with-session)
|
||
(define-key map "?" 'govc-pool-popup)
|
||
map)
|
||
"Keymap for `govc-pool-mode'.")
|
||
|
||
(defun govc-pool (&optional filter session)
|
||
"Pool info via govc.
|
||
Optionally filter by FILTER and inherit SESSION."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-pool*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-pool-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-pool-mode govc-tabulated-list-mode "Pool"
|
||
"Major mode for handling a list of govc pools."
|
||
(setq tabulated-list-format [("Name" 30 t)
|
||
("CPU Usage" 25 t)
|
||
("CPU Shares" 25 t)
|
||
("CPU Reservation" 25 t)
|
||
("CPU Limit" 10 t)
|
||
("Mem Usage" 25 t)
|
||
("Mem Shares" 25 t)
|
||
("Mem Reservation" 25 t)
|
||
("Mem Limit" 10 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-pool-info)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-pool-popup
|
||
"Pool popup."
|
||
:actions (govc-keymap-popup govc-pool-mode-map))
|
||
|
||
(easy-menu-define govc-host-mode-menu govc-pool-mode-map
|
||
"Pool menu."
|
||
(cons "Pool" (govc-keymap-menu govc-pool-mode-map)))
|
||
|
||
|
||
;;; govc datastore mode
|
||
(defun govc-ls-datastore ()
|
||
"List datastores."
|
||
(govc "ls" "datastore"))
|
||
|
||
(defun govc-datastore-ls-entries ()
|
||
"Wrapper for govc datastore.ls."
|
||
(let* ((data (govc-json "datastore.ls" "-l" "-p" govc-filter))
|
||
(file (plist-get (elt data 0) :File)))
|
||
(-map (lambda (ent)
|
||
(let ((name (plist-get ent :Path))
|
||
(size (plist-get ent :FileSize))
|
||
(time (plist-get ent :Modification))
|
||
(user (plist-get ent :Owner)))
|
||
(list (concat govc-filter name)
|
||
(vector (file-size-human-readable size)
|
||
(current-time-string (date-to-time time))
|
||
name)))) file)))
|
||
|
||
(defun govc-datastore-ls-parent ()
|
||
"Up to parent folder."
|
||
(interactive)
|
||
(if (s-blank? govc-filter)
|
||
(let ((session (govc-current-session)))
|
||
(govc-datastore-mode)
|
||
(govc-session-clone session))
|
||
(setq govc-filter (file-name-directory (directory-file-name govc-filter))))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-datastore-ls-child ()
|
||
"Open datastore folder or file."
|
||
(interactive)
|
||
(let ((id (tabulated-list-get-id)))
|
||
(if current-prefix-arg
|
||
(govc-shell-command (govc-format-command "datastore.ls" "-l" "-p" "-R" id))
|
||
(if (s-ends-with? "/" id)
|
||
(progn (setq govc-filter id)
|
||
(tabulated-list-revert))
|
||
(govc-datastore-open)))))
|
||
|
||
(defun govc-datastore-open ()
|
||
"Open datastore file."
|
||
(lexical-let* ((srcfile (tabulated-list-get-id))
|
||
(srcpath (format "[%s] %s" (file-name-nondirectory govc-session-datastore) (s-chop-prefix "/" srcfile)))
|
||
(suffix (file-name-extension srcfile t))
|
||
(tmpfile (make-temp-file "govc-ds" nil suffix))
|
||
(session (govc-current-session)))
|
||
(when (yes-or-no-p (concat "Open " srcpath "?"))
|
||
(govc "datastore.download" srcfile tmpfile)
|
||
(with-current-buffer (pop-to-buffer (find-file-noselect tmpfile))
|
||
(govc-session-clone session)
|
||
(add-hook 'kill-buffer-hook (lambda ()
|
||
(with-demoted-errors
|
||
(delete-file tmpfile))) t t)
|
||
(add-hook 'after-save-hook (lambda ()
|
||
(if (yes-or-no-p (concat "Upload changes to " srcpath "?"))
|
||
(with-demoted-errors
|
||
(govc "datastore.upload" tmpfile srcfile)))) t t)))))
|
||
|
||
(defun govc-datastore-tail ()
|
||
"Tail datastore file."
|
||
(interactive)
|
||
(govc-shell-command
|
||
(govc-format-command "datastore.tail"
|
||
(list "-n" govc-max-events (if current-prefix-arg "-f")) (govc-selection))))
|
||
|
||
(defun govc-datastore-disk-info ()
|
||
"Info datastore disk."
|
||
(interactive)
|
||
(delete-other-windows)
|
||
(govc-shell-command
|
||
(govc-format-command "datastore.disk.info" (if current-prefix-arg "-c") (govc-selection))))
|
||
|
||
(defun govc-datastore-ls-json ()
|
||
"JSON via govc datastore.ls -json on current selection."
|
||
(interactive)
|
||
(let ((govc-args '("-l" "-p")))
|
||
(govc-json-info-selection "datastore.ls")))
|
||
|
||
(defun govc-datastore-ls-r-json ()
|
||
"Search via govc datastore.ls -json -R on current selection."
|
||
(interactive)
|
||
(let ((govc-args '("-l" "-p" "-R")))
|
||
(govc-json-info-selection "datastore.ls")))
|
||
|
||
(defun govc-datastore-mkdir (name)
|
||
"Mkdir via govc datastore.mkdir with given NAME."
|
||
(interactive (list (read-from-minibuffer "Create directory: " govc-filter)))
|
||
(govc "datastore.mkdir" name)
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-datastore-rm (paths)
|
||
"Delete datastore PATHS."
|
||
(--each paths (govc "datastore.rm" (if current-prefix-arg "-f") it)))
|
||
|
||
(defun govc-datastore-rm-selection ()
|
||
"Delete selected datastore paths."
|
||
(interactive)
|
||
(govc-do-selection 'govc-datastore-rm "Delete")
|
||
(tabulated-list-revert))
|
||
|
||
(defvar govc-datastore-ls-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "I" 'govc-datastore-disk-info)
|
||
(define-key map "J" 'govc-datastore-ls-json)
|
||
(define-key map "S" 'govc-datastore-ls-r-json)
|
||
(define-key map "D" 'govc-datastore-rm-selection)
|
||
(define-key map "T" 'govc-datastore-tail)
|
||
(define-key map "+" 'govc-datastore-mkdir)
|
||
(define-key map (kbd "DEL") 'govc-datastore-ls-parent)
|
||
(define-key map (kbd "RET") 'govc-datastore-ls-child)
|
||
(define-key map "?" 'govc-datastore-ls-popup)
|
||
map)
|
||
"Keymap for `govc-datastore-ls-mode'.")
|
||
|
||
(defun govc-datastore-ls (&optional datastore session filter)
|
||
"List govc datastore. Optionally specify DATASTORE, SESSION and FILTER."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-datastore*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-datastore-ls-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-session-datastore (or datastore (govc-object-prompt "govc datastore: " 'govc-ls-datastore)))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-datastore-ls-mode govc-tabulated-list-mode "Datastore"
|
||
"Major mode govc datastore.ls."
|
||
(setq tabulated-list-format [("Size" 10 t)
|
||
("Modification time" 25 t)
|
||
("Name" 40 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-datastore-ls-entries)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-datastore-ls-popup
|
||
"Datastore ls popup."
|
||
:actions (govc-keymap-popup govc-datastore-ls-mode-map))
|
||
|
||
(easy-menu-define govc-datastore-ls-mode-menu govc-datastore-ls-mode-map
|
||
"Datastore ls menu."
|
||
(cons "Datastore" (govc-keymap-menu govc-datastore-ls-mode-map)))
|
||
|
||
(defvar govc-datastore-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "J" 'govc-datastore-json-info)
|
||
(define-key map "M" 'govc-metric)
|
||
(define-key map "O" 'govc-object-info)
|
||
(define-key map (kbd "RET") 'govc-datastore-ls-selection)
|
||
(define-key map "c" 'govc-mode-new-session)
|
||
(define-key map "h" 'govc-host-with-session)
|
||
(define-key map "p" 'govc-pool-with-session)
|
||
(define-key map "v" 'govc-vm-with-session)
|
||
(define-key map "?" 'govc-datastore-popup)
|
||
map)
|
||
"Keymap for `govc-datastore-mode'.")
|
||
|
||
(defun govc-datastore-json-info ()
|
||
"JSON via govc datastore.info -json on current selection."
|
||
(interactive)
|
||
(govc-json-info-selection "datastore.info"))
|
||
|
||
(defun govc-datastore-info ()
|
||
"Wrapper for govc datastore.info."
|
||
(govc-table-info "datastore.info" (or govc-filter "*")))
|
||
|
||
(defun govc-datastore-ls-selection ()
|
||
"Browse datastore."
|
||
(interactive)
|
||
(govc-datastore-ls (tabulated-list-get-id) (govc-current-session)))
|
||
|
||
(defun govc-datastore (&optional filter session)
|
||
"Datastore info via govc.
|
||
Optionally filter by FILTER and inherit SESSION."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-datastore*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-datastore-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)
|
||
(if (and govc-session-datastore (search-forward govc-session-datastore nil t))
|
||
(beginning-of-line))))
|
||
|
||
(define-derived-mode govc-datastore-mode tabulated-list-mode "Datastore"
|
||
"Major mode for govc datastore.info."
|
||
(setq tabulated-list-format [("Name" 15 t)
|
||
("Type" 10 t)
|
||
("Capacity" 10 t)
|
||
("Free" 10 t)
|
||
("Remote" 30 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-datastore-info)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-datastore-popup
|
||
"Datastore popup."
|
||
:actions (govc-keymap-popup govc-datastore-mode-map))
|
||
|
||
(easy-menu-define govc-datastore-mode-menu govc-datastore-mode-map
|
||
"Datastore menu."
|
||
(cons "Datastore" (govc-keymap-menu govc-datastore-mode-map)))
|
||
|
||
|
||
;;; govc vm mode
|
||
(defun govc-vm-prompt (prompt)
|
||
"PROMPT for a vm name."
|
||
(completing-read prompt (govc "ls" "vm")))
|
||
|
||
(defun govc-vm-start (name)
|
||
"Start vm with given NAME."
|
||
(interactive (list (govc-vm-prompt "Start vm: ")))
|
||
(govc "vm.power" "-on" name))
|
||
|
||
(defun govc-vm-shutdown (name)
|
||
"Shutdown vm with given NAME."
|
||
(interactive (list (govc-vm-prompt "Shutdown vm: ")))
|
||
(govc "vm.power" "-s" "-force" name))
|
||
|
||
(defun govc-vm-reboot (name)
|
||
"Reboot vm with given NAME."
|
||
(interactive (list (govc-vm-prompt "Reboot vm: ")))
|
||
(govc "vm.power" "-r" "-force" name))
|
||
|
||
(defun govc-vm-suspend (name)
|
||
"Suspend vm with given NAME."
|
||
(interactive (list (govc-vm-prompt "Suspend vm: ")))
|
||
(govc "vm.power" "-suspend" name))
|
||
|
||
(defun govc-vm-destroy (name)
|
||
"Destroy vm with given NAME."
|
||
(interactive (list (govc-vm-prompt "Destroy vm: ")))
|
||
(govc "vm.destroy" name))
|
||
|
||
(defun govc-vm-vnc-enable (name)
|
||
"Enable vnc on vm with given NAME."
|
||
(--map (last (split-string it))
|
||
(govc "vm.vnc" "-enable"
|
||
"-port" "-1"
|
||
"-password" (format "%08x" (random (expt 16 8))) name)))
|
||
|
||
(defun govc-vm-vnc (name &optional arg)
|
||
"VNC for vm with given NAME.
|
||
By default, enable and open VNC for the given vm NAME.
|
||
With prefix \\[negative-argument] ARG, VNC will be disabled.
|
||
With prefix \\[universal-argument] ARG, VNC will be enabled but not opened."
|
||
(interactive (list (govc-vm-prompt "VNC vm: ")
|
||
current-prefix-arg))
|
||
(if (equal arg '-)
|
||
(govc "vm.vnc" "-disable" name)
|
||
(let ((urls (govc-vm-vnc-enable name)))
|
||
(unless arg
|
||
(-each (-flatten urls) 'browse-url)))))
|
||
|
||
(defun govc-vm-screen (name &optional arg)
|
||
"Console screenshot of vm NAME console.
|
||
Open via `eww' by default, via `browse-url' if ARG is non-nil."
|
||
(interactive (list (govc-vm-prompt "Console screenshot vm: ")
|
||
current-prefix-arg))
|
||
(let* ((data (govc-json "vm.info" name))
|
||
(vms (plist-get data :VirtualMachines))
|
||
(url (govc-url-parse govc-session-url)))
|
||
(mapc
|
||
(lambda (vm)
|
||
(let* ((moid (plist-get (plist-get vm :Self) :Value))
|
||
(on (string= "poweredOn" (plist-get (plist-get vm :Runtime) :PowerState)))
|
||
(host (format "%s:%d" (url-host url) (or (url-port url) 443)))
|
||
(path (concat "/screen?id=" moid))
|
||
(auth (concat (url-user url) ":" (url-password url))))
|
||
(if current-prefix-arg
|
||
(browse-url (concat "https://" auth "@" host path))
|
||
(let ((creds `((,host ("VMware HTTP server" . ,(base64-encode-string auth)))))
|
||
(url-basic-auth-storage 'creds)
|
||
(u (concat "https://" host path)))
|
||
(require 'eww)
|
||
(if on
|
||
(url-retrieve u 'eww-render (list u))
|
||
(kill-new (message u)))))))
|
||
vms)))
|
||
|
||
(defun govc-vm-start-selection ()
|
||
"Start via `govc-vm-start' on the current selection."
|
||
(interactive)
|
||
(govc-vm-start (govc-selection))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-vm-shutdown-selection ()
|
||
"Shutdown via `govc-vm-shutdown' on the current selection."
|
||
(interactive)
|
||
(govc-vm-shutdown (govc-selection))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-vm-reboot-selection ()
|
||
"Reboot via `govc-vm-reboot' on the current selection."
|
||
(interactive)
|
||
(govc-vm-reboot (govc-selection))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-vm-suspend-selection ()
|
||
"Suspend via `govc-vm-suspend' on the current selection."
|
||
(interactive)
|
||
(govc-vm-suspend (govc-selection))
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-vm-destroy-selection ()
|
||
"Destroy via `govc-vm-destroy' on the current selection."
|
||
(interactive)
|
||
(govc-do-selection 'govc-vm-destroy "Destroy")
|
||
(tabulated-list-revert))
|
||
|
||
(defun govc-vm-vnc-selection ()
|
||
"VNC via `govc-vm-vnc' on the current selection."
|
||
(interactive)
|
||
(govc-vm-vnc (govc-selection) current-prefix-arg))
|
||
|
||
(defun govc-vm-screen-selection ()
|
||
"Console screenshot via `govc-vm-screen' on the current selection."
|
||
(interactive)
|
||
(govc-vm-screen (govc-selection) current-prefix-arg))
|
||
|
||
(defun govc-vm-info ()
|
||
"Wrapper for govc vm.info."
|
||
(govc-table-info "vm.info" (list "-r" (or govc-filter (setq govc-filter "*")))))
|
||
|
||
(defun govc-vm-host ()
|
||
"Host info via `govc-host' with host(s) of current selection."
|
||
(interactive)
|
||
(govc-host (concat "*/" (govc-table-column-value "Host"))
|
||
(govc-current-session)))
|
||
|
||
(defun govc-vm-datastore ()
|
||
"Datastore via `govc-datastore-ls' with datastore of current selection."
|
||
(interactive)
|
||
(if current-prefix-arg
|
||
(govc-datastore (s-split ", " (govc-table-column-value "Storage") t)
|
||
(govc-current-session))
|
||
(let* ((data (govc-json "vm.info" (tabulated-list-get-id)))
|
||
(vm (elt (plist-get data :VirtualMachines) 0))
|
||
(dir (plist-get (plist-get (plist-get vm :Config) :Files) :LogDirectory))
|
||
(args (s-split "\\[\\|\\]" dir t)))
|
||
(govc-datastore-ls (first args) (govc-current-session) (concat (s-trim (second args)) "/")))))
|
||
|
||
(defun govc-vm-ping ()
|
||
"Ping VM."
|
||
(interactive)
|
||
(let ((ping-program-options '("-c" "20")))
|
||
(ping (govc-table-column-value "IP address"))))
|
||
|
||
(defun govc-vm-device-ls ()
|
||
"Devices via `govc-device' on the current selection."
|
||
(interactive)
|
||
(govc-device (tabulated-list-get-id)
|
||
(govc-current-session)))
|
||
|
||
(defun govc-vm-extra-config ()
|
||
"Populate table with govc vm.info -e output."
|
||
(let* ((data (govc-json "vm.info" govc-args))
|
||
(vms (plist-get data :VirtualMachines))
|
||
(info))
|
||
(mapc
|
||
(lambda (vm)
|
||
(let* ((config (plist-get vm :Config))
|
||
(name (plist-get config :Name)))
|
||
(mapc (lambda (x)
|
||
(let ((key (plist-get x :Key))
|
||
(val (plist-get x :Value)))
|
||
(push (list key (vector key val)) info)))
|
||
(plist-get config :ExtraConfig))
|
||
(if (> (length vms) 1)
|
||
(push (list name (vector "vm.name" name)) info))))
|
||
vms)
|
||
info))
|
||
|
||
(defun govc-vm-extra-config-table ()
|
||
"ExtraConfig via `govc-vm-extra-config' on the current selection."
|
||
(interactive)
|
||
(govc-map-info-table #'govc-vm-extra-config))
|
||
|
||
(defun govc-vm-json-info ()
|
||
"JSON via govc vm.info -json on current selection."
|
||
(interactive)
|
||
(govc-json-info-selection "vm.info"))
|
||
|
||
(defvar govc-vm-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map "E" 'govc-events)
|
||
(define-key map "J" 'govc-vm-json-info)
|
||
(define-key map "O" 'govc-object-info)
|
||
(define-key map "X" 'govc-vm-extra-config-table)
|
||
(define-key map (kbd "RET") 'govc-vm-device-ls)
|
||
(define-key map "C" 'govc-vm-screen-selection)
|
||
(define-key map "V" 'govc-vm-vnc-selection)
|
||
(define-key map "D" 'govc-vm-destroy-selection)
|
||
(define-key map "^" 'govc-vm-start-selection)
|
||
(define-key map "!" 'govc-vm-shutdown-selection)
|
||
(define-key map "@" 'govc-vm-reboot-selection)
|
||
(define-key map "&" 'govc-vm-suspend-selection)
|
||
(define-key map "H" 'govc-vm-host)
|
||
(define-key map "M" 'govc-metric)
|
||
(define-key map "P" 'govc-vm-ping)
|
||
(define-key map "S" 'govc-vm-datastore)
|
||
(define-key map "c" 'govc-mode-new-session)
|
||
(define-key map "h" 'govc-host-with-session)
|
||
(define-key map "p" 'govc-pool-with-session)
|
||
(define-key map "s" 'govc-datastore-with-session)
|
||
(define-key map "?" 'govc-vm-popup)
|
||
map)
|
||
"Keymap for `govc-vm-mode'.")
|
||
|
||
(defun govc-vm (&optional filter session)
|
||
"VM info via govc.
|
||
Optionally filter by FILTER and inherit SESSION."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-vm*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-vm-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-filter filter)
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-vm-mode govc-tabulated-list-mode "VM"
|
||
"Major mode for handling a list of govc vms."
|
||
(setq tabulated-list-format [("Name" 40 t)
|
||
("Power state" 12 t)
|
||
("Boot time" 13 t)
|
||
("IP address" 15 t)
|
||
("Guest name" 20 t)
|
||
("Host" 20 t)
|
||
("CPU usage" 15 t)
|
||
("Host memory usage" 18 t)
|
||
("Guest memory usage" 19 t)
|
||
("Storage committed" 18 t)
|
||
("Storage" 10 t)
|
||
("Network" 10 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-vm-info)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-vm-popup
|
||
"VM popup."
|
||
:actions (govc-keymap-popup govc-vm-mode-map))
|
||
|
||
(easy-menu-define govc-vm-mode-menu govc-vm-mode-map
|
||
"VM menu."
|
||
(cons "VM" (govc-keymap-menu govc-vm-mode-map)))
|
||
|
||
|
||
;;; govc device mode
|
||
(defun govc-device-ls ()
|
||
"Wrapper for govc device.ls -vm VM."
|
||
(govc-type-list-entries "device.ls"))
|
||
|
||
(defun govc-device-info ()
|
||
"Populate table with govc device.info output."
|
||
(govc-map-info "device.info" govc-args))
|
||
|
||
(defun govc-device-info-table ()
|
||
"Tabulated govc device.info."
|
||
(interactive)
|
||
(govc-map-info-table #'govc-device-info))
|
||
|
||
(defun govc-device-json-info ()
|
||
"JSON via govc device.info -json on current selection."
|
||
(interactive)
|
||
(govc-json-info-selection "device.info"))
|
||
|
||
(defvar govc-device-mode-map
|
||
(let ((map (make-sparse-keymap)))
|
||
(define-key map (kbd "J") 'govc-device-json-info)
|
||
(define-key map (kbd "RET") 'govc-device-info-table)
|
||
map)
|
||
"Keymap for `govc-device-mode'.")
|
||
|
||
(defun govc-device (&optional vm session)
|
||
"List govc devices for VM. Optionally inherit SESSION."
|
||
(interactive)
|
||
(let ((buffer (get-buffer-create "*govc-device*")))
|
||
(pop-to-buffer buffer)
|
||
(govc-device-mode)
|
||
(if session
|
||
(govc-session-clone session)
|
||
(call-interactively 'govc-session))
|
||
(setq govc-args (list "-vm" (or vm (govc-vm-prompt "vm: "))))
|
||
(tabulated-list-print)))
|
||
|
||
(define-derived-mode govc-device-mode govc-tabulated-list-mode "Device"
|
||
"Major mode for handling a govc device."
|
||
(setq tabulated-list-format [("Name" 15 t)
|
||
("Type" 30 t)
|
||
("Summary" 40 t)]
|
||
tabulated-list-sort-key (cons "Name" nil)
|
||
tabulated-list-padding 2
|
||
tabulated-list-entries #'govc-device-ls)
|
||
(tabulated-list-init-header))
|
||
|
||
(magit-define-popup govc-popup
|
||
"govc popup."
|
||
:actions (govc-keymap-list govc-command-map))
|
||
|
||
(easy-menu-change
|
||
'("Tools") "govc"
|
||
(govc-keymap-menu govc-command-map)
|
||
"Search Files (Grep)...")
|
||
|
||
(provide 'govc)
|
||
|
||
;;; govc.el ends here
|