summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKyle Meyer <kyle@kyleam.com>2021-06-05 17:13:48 -0400
committerKyle Meyer <kyle@kyleam.com>2021-06-07 00:12:06 -0400
commit941e347c49caa46c71ba9d9842e4a47270cac452 (patch)
tree4e67796bbf05c8eb359c6e30fb23972e152e2e0b
parent7b51ed76fdbea12fe98b03e9ebaacf18fa16e8be (diff)
downloadpiem-941e347c49caa46c71ba9d9842e4a47270cac452.tar.gz
lei: Add command and mode for displaying overview of search results
The output is intended to resemble search in public-inbox's web interface: an entry for each matching message. This is different from notmuch-search's output in that results are not grouped in their thread. I like notmuch's interface, although I'm not sure that trying to reshape lei-q's JSON output into something like that is worth the code complication or computation cost. The plan is to eventually wire this up to a transient to allow the caller to specify arguments (e.g., --only to restrict the search results to a particular inbox). Message-Id: <20210605211402.20304-5-kyle@kyleam.com>
-rw-r--r--piem-lei.el68
1 files changed, 68 insertions, 0 deletions
diff --git a/piem-lei.el b/piem-lei.el
index 291964f..ed153c2 100644
--- a/piem-lei.el
+++ b/piem-lei.el
@@ -21,6 +21,7 @@
;;; Code:
+(require 'json)
(require 'message)
(require 'piem)
@@ -140,5 +141,72 @@ unless DISPLAY is non-nil."
(setq font-lock-defaults (list piem-lei-show-mode-font-lock-keywords t))
(setq-local line-move-visual t))
+
+;;;; Searching
+
+(defun piem-lei-query--read-json-item ()
+ (let ((json-object-type 'alist)
+ (json-array-type 'list)
+ ;; Using symbols for lei-q's output should be fine, though
+ ;; it's a little odd for the "t:" field.
+ (json-key-type 'symbol)
+ (json-false nil)
+ (json-null nil))
+ (json-read)))
+
+(defvar piem-lei-query--date-re
+ (rx string-start
+ (group (= 4 digit) "-" (= 2 digit) "-" (= 2 digit))
+ "T" (group (= 2 digit) ":" (= 2 digit)) ":" (= 2 digit) "Z"
+ string-end))
+
+(defun piem-lei-query--format-date (data)
+ (let ((date (cdr (assq 'dt data))))
+ (if (string-match piem-lei-query--date-re date)
+ (concat (match-string 1 date) " " (match-string 2 date))
+ (error "Date did not match expected format: %S" date))))
+
+;;;###autoload
+(defun piem-lei-query (query)
+ "Call `lei q' with QUERY.
+QUERY is split according to `split-string-and-unquote'."
+ (interactive
+ (list (split-string-and-unquote
+ (read-string "Query: " "d:20.days.ago.. " 'piem-lei-query-history))))
+ (with-current-buffer (get-buffer-create "*lei-query*")
+ (let ((inhibit-read-only t))
+ (erase-buffer)
+ (apply #'call-process "lei" nil '(t nil) nil
+ "q" "--format=ldjson" query)
+ (goto-char (point-min))
+ (while (not (eobp))
+ (let ((data (piem-lei-query--read-json-item)))
+ (delete-region (line-beginning-position) (point))
+ (insert
+ (format "%s %3s %-20.20s %s"
+ (piem-lei-query--format-date data)
+ (if-let ((pct (cdr (assq 'pct data))))
+ (concat (number-to-string (cdr (assq 'pct data)))
+ "%")
+ "")
+ (let ((from (car (cdr (assq 'f data)))))
+ (or (car from) (cadr from)))
+ (cdr (assq 's data))))
+ (add-text-properties (line-beginning-position) (line-end-position)
+ (list 'piem-lei-query-result data)))
+ (forward-line))
+ (insert "End of lei-q results"))
+ (goto-char (point-min))
+ (piem-lei-query-mode)
+ (pop-to-buffer-same-window (current-buffer))))
+
+(define-derived-mode piem-lei-query-mode special-mode "lei-query"
+ "Major mode for displaying overview of `lei q' results."
+ :group 'piem-lei
+ (buffer-disable-undo)
+ (setq truncate-lines t)
+ (setq buffer-read-only t)
+ (setq-local line-move-visual t))
+
;;; piem-lei.el ends here
(provide 'piem-lei)