;;; khoj.el --- Natural, Incremental Search via Emacs ;; Copyright (C) 2021-2022 Debanjum Singh Solanky ;; Author: Debanjum Singh Solanky ;; Version: 2.0 ;; Keywords: search, org-mode, outlines, markdown, image ;; URL: http://github.com/debanjum/khoj/interface/emacs ;; This file is NOT part of GNU Emacs. ;;; License: ;; 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 . ;;; Commentary: ;; This package provides a natural, incremental search interface to your ;; org-mode notes, markdown files, beancount transactions and images. ;; It is a wrapper that interfaces with transformer based ML models. ;; The models search capabilities are exposed via the Khoj HTTP API. ;;; Code: (require 'url) (require 'json) (defcustom khoj--server-url "http://localhost:8000" "Location of Khoj API server." :group 'khoj :type 'string) (defcustom khoj--image-width 156 "Width of rendered images returned by Khoj." :group 'khoj :type 'integer) (defcustom khoj--rerank-after-idle-time 1.0 "Idle time (in seconds) to trigger cross-encoder to rerank incremental search results." :group 'khoj :type 'float) (defcustom khoj--results-count 5 "Number of results to get from Khoj API for each query." :group 'khoj :type 'integer) (defvar khoj--rerank-timer nil "Idle timer to make cross-encoder re-rank incremental search results if user idle.") (defvar khoj--minibuffer-window nil "Minibuffer window being used by user to enter query.") (defconst khoj--query-prompt "🦅Khoj: " "Query prompt shown to user in the minibuffer.") (defconst khoj--buffer-name "*🦅Khoj*" "Name of buffer to show results from Khoj.") (defvar khoj--search-type "org" "The type of content to perform search on.") (defvar khoj--keybindings-info-message " Set Search Type ------------------------- C-x m | markdown C-x o | org-mode C-x l | ledger/beancount C-x i | images ") (defun khoj--search-markdown (interactive) (setq khoj--search-type "markdown")) (defun khoj--search-org (interactive) (setq khoj--search-type "org")) (defun khoj--search-ledger (interactive) (setq khoj--search-type "ledger")) (defun khoj--search-images (interactive) (setq khoj--search-type "image")) (defun khoj--make-search-keymap (&optional existing-keymap) "Setup keymap to configure Khoj search" (let ((kmap (or existing-keymap (make-sparse-keymap)))) (define-key kmap (kbd "C-x m") #'khoj--search-markdown) (define-key kmap (kbd "C-x o") #'khoj--search-org) (define-key kmap (kbd "C-x l") #'khoj--search-ledger) (define-key kmap (kbd "C-x i") #'khoj--search-images) kmap)) (defun khoj--display-keybinding-info () "Display information on keybindings to customize khoj search. Use `which-key` if available, else display simple message in echo area" (if (fboundp 'which-key--create-buffer-and-show) (which-key--create-buffer-and-show (kbd "C-x") (symbolp (khoj--make-search-keymap)) '(lambda (binding) (string-prefix-p "khoj--" (cdr binding))) "Khoj Bindings") (message "%s" khoj--keybindings-info-message))) (defun khoj--extract-entries-as-markdown (json-response query) "Convert json response from API to markdown entries" ;; remove leading (, ) or SPC from extracted entries string (replace-regexp-in-string "^[\(\) ]" "" ;; extract entries from response as single string and convert to entries (format "# %s\n%s" query (mapcar (lambda (args) (replace-regexp-in-string "^\#+" "##" (format "%s" (cdr (assoc 'entry args))))) json-response)))) (defun khoj--extract-entries-as-org (json-response query) "Convert json response from API to org-mode entries" ;; remove leading (, ) or SPC from extracted entries string (replace-regexp-in-string "^[\(\) ]" "" ;; extract entries from response as single string and convert to entries (format "#+STARTUP: showall hidestars inlineimages\n* %s\n%s" query (mapcar (lambda (args) (replace-regexp-in-string "^\*+" "**" (format "%s" (cdr (assoc 'entry args))))) json-response)))) (defun khoj--extract-entries-as-images (json-response query) "Convert json response from API to html with images" ;; remove leading (, ) or SPC from extracted entries string (replace-regexp-in-string "[\(\) ]$" "" ;; remove leading (, ) or SPC from extracted entries string (replace-regexp-in-string "^[\(\) ]" "" ;; extract entries from response as single string and convert to entries (format "\n\n

%s

%s\n\n\n" query (mapcar (lambda (args) (format "\n\n

Score: %s Meta: %s Image: %s

\n\n\n\n" (cdr (assoc 'score args)) (cdr (assoc 'metadata_score args)) (cdr (assoc 'image_score args)) khoj--server-url (cdr (assoc 'entry args)) khoj--server-url (cdr (assoc 'entry args)) (random 10000))) json-response))))) (defun khoj--extract-entries-as-ledger (json-response query) "Convert json response from API to ledger entries" ;; remove leading (, ) or SPC from extracted entries string (replace-regexp-in-string "[\(\) ]$" "" (replace-regexp-in-string "^[\(\) ]" "" ;; extract entries from response as single string and convert to entries (format ";; %s\n\n%s\n" query (mapcar (lambda (args) (format "%s\n\n" (cdr (assoc 'entry args)))) json-response))))) (defun khoj--buffer-name-to-search-type (buffer-name) (let ((file-extension (file-name-extension buffer-name))) (cond ((equal buffer-name "Music.org") "music") ((or (equal file-extension "bean") (equal file-extension "beancount")) "ledger") ((equal file-extension "org") "org") ((or (equal file-extension "markdown") (equal file-extension "md")) "markdown") (t "org")))) (defun khoj--construct-api-query (query search-type &optional rerank) (let ((rerank (or rerank "false")) (results-count (or khoj--results-count 5)) (encoded-query (url-hexify-string query))) (format "%s/search?q=%s&t=%s&r=%s&n=%s" khoj--server-url encoded-query search-type rerank results-count))) (defun khoj--query-api-and-render-results (query search-type query-url buffer-name) ;; get json response from api (with-current-buffer buffer-name (let ((inhibit-read-only t)) (erase-buffer) (url-insert-file-contents query-url))) ;; render json response into formatted entries (with-current-buffer buffer-name (let ((inhibit-read-only t) (json-response (json-parse-buffer :object-type 'alist))) (erase-buffer) (insert (cond ((or (equal search-type "org") (equal search-type "music")) (khoj--extract-entries-as-org json-response query)) ((equal search-type "markdown") (khoj--extract-entries-as-markdown json-response query)) ((equal search-type "ledger") (khoj--extract-entries-as-ledger json-response query)) ((equal search-type "image") (khoj--extract-entries-as-images json-response query)) (t (format "%s" json-response)))) (cond ((equal search-type "org") (org-mode)) ((equal search-type "markdown") (markdown-mode)) ((equal search-type "ledger") (beancount-mode)) ((equal search-type "music") (progn (org-mode) (org-music-mode))) ((equal search-type "image") (progn (shr-render-region (point-min) (point-max)) (goto-char (point-min)))) (t (fundamental-mode)))) (read-only-mode t))) ;; Incremental Search on Khoj (defun khoj--incremental-search (&optional rerank) (let* ((rerank-str (cond (rerank "true") (t "false"))) (khoj-buffer-name (get-buffer-create khoj--buffer-name)) (query (minibuffer-contents-no-properties)) (query-url (khoj--construct-api-query query khoj--search-type rerank-str))) ;; Query khoj API only when user in khoj minibuffer and non-empty query ;; Prevents querying if ;; 1. user hasn't started typing query ;; 2. during recursive edits ;; 3. with contents of other buffers user may jump to (when (and (not (equal query "")) (active-minibuffer-window) (equal (current-buffer) khoj--minibuffer-window)) (progn (when rerank (message "Khoj: Rerank Results")) (khoj--query-api-and-render-results query khoj--search-type query-url khoj-buffer-name))))) (defun delete-open-network-connections-to-khoj () "Delete all network connections to khoj server" (dolist (proc (process-list)) (let ((proc-buf (buffer-name (process-buffer proc))) (khoj-network-proc-buf (string-join (split-string khoj--server-url "://") " "))) (when (string-match (format "%s" khoj-network-proc-buf) proc-buf) (delete-process proc))))) (defun khoj--teardown-incremental-search () (message "[Khoj]: Teardown Incremental Search") ;; remove advice to rerank results on normal exit from minibuffer (advice-remove 'exit-minibuffer #'khoj--minibuffer-exit-advice) ;; unset khoj minibuffer window (setq khoj--minibuffer-window nil) ;; cancel rerank timer (when (timerp khoj--rerank-timer) (cancel-timer khoj--rerank-timer)) ;; delete open connections to khoj (delete-open-network-connections-to-khoj) ;; remove hooks for khoj incremental query and self (remove-hook 'post-command-hook #'khoj--incremental-search) (remove-hook 'minibuffer-exit-hook #'khoj--teardown-incremental-search)) (defun khoj--minibuffer-exit-advice (&rest _args) (khoj--incremental-search t)) ;;;###autoload (defun khoj () "Natural, Incremental Search for your personal notes, transactions and music using Khoj" (interactive) (let* ((khoj-buffer-name (get-buffer-create khoj--buffer-name))) ;; set khoj search type to last used or based on current buffer (setq khoj--search-type (or khoj--search-type (khoj--buffer-name-to-search-type (buffer-name)))) ;; setup rerank to improve results once user idle for KHOJ--RERANK-AFTER-IDLE-TIME seconds (setq khoj--rerank-timer (run-with-idle-timer khoj--rerank-after-idle-time t 'khoj--incremental-search t)) ;; switch to khoj results buffer (switch-to-buffer khoj-buffer-name) ;; open and setup minibuffer for incremental search (minibuffer-with-setup-hook (lambda () ;; Add khoj keybindings for configuring search to minibuffer keybindings (khoj--make-search-keymap minibuffer-local-map) ;; Display information on keybindings to customize khoj search (khoj--display-keybinding-info) ;; set current (mini-)buffer entered as khoj minibuffer ;; used to query khoj API only when user in khoj minibuffer (setq khoj--minibuffer-window (current-buffer)) ;; rerank results on normal exit from minibuffer (advice-add 'exit-minibuffer :before #'khoj--minibuffer-exit-advice) (add-hook 'post-command-hook #'khoj--incremental-search) ; do khoj incremental search after every user action (add-hook 'minibuffer-exit-hook #'khoj--teardown-incremental-search)) ; teardown khoj incremental search on minibuffer exit (read-string khoj--query-prompt)))) ;;;###autoload (defun khoj-simple (query) "Natural Search for QUERY in your personal notes, transactions, music and images using Khoj" (interactive "s🦅Khoj: ") (let* ((rerank "true") (default-type (khoj--buffer-name-to-search-type (buffer-name))) (search-type (completing-read "Type: " '("org" "markdown" "ledger" "music" "image") nil t default-type)) (query-url (khoj--construct-api-query query search-type rerank)) (buffer-name (get-buffer-create (format "*%s (q:%s t:%s)*" khoj--buffer-name query search-type)))) (khoj--query-api-and-render-results query search-type query-url buffer-name) (switch-to-buffer buffer-name))) (provide 'khoj) ;;; khoj.el ends here