mirror of
https://github.com/org-roam/org-roam
synced 2025-08-01 12:17:21 -05:00
48a5d01726
was wrong, and the closure wasn't the cause. The source of
the issue stems from a missing compile-time dependency on subr-x.
Emacs 28 doesn't seem to require it for the needed functions anymore,
while older Emacsen still do. What even worse, the byte compiler doesn't
seem to complain about such thing on older Emacsen and the error message
produced at the run-time isn't helpful to identify the cause.
Fixes #1814.
319 lines
12 KiB
EmacsLisp
319 lines
12 KiB
EmacsLisp
;;; org-roam.el --- A database abstraction layer for Org-mode -*- coding: utf-8; lexical-binding: t; -*-
|
|
|
|
;; Copyright © 2020-2021 Jethro Kuan <jethrokuan95@gmail.com>
|
|
|
|
;; Author: Jethro Kuan <jethrokuan95@gmail.com>
|
|
;; URL: https://github.com/org-roam/org-roam
|
|
;; Keywords: org-mode, roam, convenience
|
|
;; Version: 2.1.0
|
|
;; Package-Requires: ((emacs "26.1") (dash "2.13") (f "0.17.2") (org "9.4") (emacsql "3.0.0") (emacsql-sqlite "1.0.0") (magit-section "3.0.0"))
|
|
|
|
;; 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:
|
|
;;
|
|
;; Org-roam is a Roam Research inspired Emacs package and is an addition to
|
|
;; Org-mode to have a way to quickly process complex SQL-like queries over a
|
|
;; large set of plain text Org-mode files. To achieve this Org-roam provides a
|
|
;; database abstraction layer, the capabilities of which include, but are not
|
|
;; limited to:
|
|
;;
|
|
;; - Link graph traversal and visualization.
|
|
;; - Instantaneous SQL-like queries on headlines
|
|
;; - What are my TODOs, scheduled for X, or due by Y?
|
|
;; - Accessing the properties of a node, such as its tags, refs, TODO state or
|
|
;; priority.
|
|
;;
|
|
;; All of these functionality is powered by this layer. Hence, at its core
|
|
;; Org-roam's primary goal is to provide a resilient dual representation of
|
|
;; what's already available in plain text, while cached in a binary database,
|
|
;; that is cheap to maintain, easy to understand, and is as up-to-date as it
|
|
;; possibly can. For users who would like to perform arbitrary programmatic
|
|
;; queries on their Org files Org-roam also exposes an API to this database
|
|
;; abstraction layer.
|
|
;;
|
|
;; -----------------------------------------------------------------------------
|
|
;;
|
|
;; In order for the package to correctly work through your interactive session
|
|
;; it's mandatory to add somewhere to your configuration the next form:
|
|
;;
|
|
;; (org-roam-db-autosync-mode)
|
|
;;
|
|
;; The form can be called both, before or after loading the package, which is up
|
|
;; to your preferences. If you call this before the package is loaded, then it
|
|
;; will automatically load the package.
|
|
;;
|
|
;; -----------------------------------------------------------------------------
|
|
;;
|
|
;; This package also comes with a set of officially supported extensions that
|
|
;; provide extra features. You can find them in the "extensions/" subdirectory.
|
|
;; These extensions are not automatically loaded with `org-roam`, but they still
|
|
;; will be lazy-loaded through their own `autoload's.
|
|
;;
|
|
;; Org-roam also has other extensions that don't come together with this package.
|
|
;; Such extensions are distributed as their own packages, while also
|
|
;; authored and maintained by different people on distinct repositories. The
|
|
;; majority of them can be found at https://github.com/org-roam and MELPA.
|
|
;;
|
|
;;; Code:
|
|
(require 'f)
|
|
(require 'dash)
|
|
|
|
(require 'rx)
|
|
(require 'seq)
|
|
(require 'cl-lib)
|
|
|
|
(require 'magit-section)
|
|
|
|
(require 'emacsql)
|
|
(require 'emacsql-sqlite)
|
|
|
|
(require 'org)
|
|
(require 'org-id)
|
|
(require 'ol)
|
|
(require 'org-element)
|
|
(require 'org-capture)
|
|
|
|
(require 'ansi-color) ; to strip ANSI color codes in `org-roam--list-files'
|
|
|
|
(eval-when-compile
|
|
(require 'subr-x))
|
|
|
|
;;; Options
|
|
(defgroup org-roam nil
|
|
"A database abstraction layer for Org-mode."
|
|
:group 'org
|
|
:prefix "org-roam-"
|
|
:link '(url-link :tag "Github" "https://github.com/org-roam/org-roam")
|
|
:link '(url-link :tag "Online Manual" "https://www.orgroam.com/manual.html"))
|
|
|
|
(defgroup org-roam-faces nil
|
|
"Faces used by Org-roam."
|
|
:group 'org-roam
|
|
:group 'faces)
|
|
|
|
(defcustom org-roam-verbose t
|
|
"Echo messages that are not errors."
|
|
:type 'boolean
|
|
:group 'org-roam)
|
|
|
|
(defcustom org-roam-directory (expand-file-name "~/org-roam/")
|
|
"Default path to Org-roam files.
|
|
All Org files, at any level of nesting, are considered part of the Org-roam."
|
|
:type 'directory
|
|
:group 'org-roam)
|
|
|
|
(defcustom org-roam-find-file-hook nil
|
|
"Hook run when an Org-roam file is visited."
|
|
:group 'org-roam
|
|
:type 'hook)
|
|
|
|
(defcustom org-roam-file-extensions '("org")
|
|
"List of file extensions to be included by Org-Roam.
|
|
While a file extension different from \".org\" may be used, the
|
|
file still needs to be an `org-mode' file, and it is the user's
|
|
responsibility to ensure that."
|
|
:type '(repeat string)
|
|
:group 'org-roam)
|
|
|
|
(defcustom org-roam-file-exclude-regexp nil
|
|
"Files matching this regular expression are excluded from the Org-roam."
|
|
:type '(choice
|
|
(string :tag "Regular expression matching files to ignore")
|
|
(const :tag "Include everything" nil))
|
|
:group 'org-roam)
|
|
|
|
(defcustom org-roam-list-files-commands
|
|
(if (member system-type '(windows-nt ms-dos cygwin))
|
|
nil
|
|
'(find fd fdfind rg))
|
|
"Commands that will be used to find Org-roam files.
|
|
|
|
It should be a list of symbols or cons cells representing any of the following
|
|
supported file search methods.
|
|
|
|
The commands will be tried in order until an executable for a command is found.
|
|
The Elisp implementation is used if no command in the list is found.
|
|
|
|
`find'
|
|
Use find as the file search method.
|
|
Example command:
|
|
find /path/to/dir -type f \( -name \"*.org\" -o -name \"*.org.gpg\" \)
|
|
|
|
`fd'
|
|
Use fd as the file search method.
|
|
Example command: fd /path/to/dir/ --type file -e \".org\" -e \".org.gpg\"
|
|
|
|
`fdfind'
|
|
Same as `fd'. It's an alias that used in some OSes (e.g. Debian, Ubuntu)
|
|
|
|
`rg'
|
|
Use ripgrep as the file search method.
|
|
Example command: rg /path/to/dir/ --files -g \"*.org\" -g \"*.org.gpg\"
|
|
|
|
By default, `executable-find' will be used to look up the path to the
|
|
executable. If a custom path is required, it can be specified together with the
|
|
method symbol as a cons cell. For example: '(find (rg . \"/path/to/rg\"))."
|
|
:type '(set (const :tag "find" find)
|
|
(const :tag "fd" fd)
|
|
(const :tag "fdfind" fdfind)
|
|
(const :tag "rg" rg)
|
|
(const :tag "elisp" nil)))
|
|
|
|
;;; Library
|
|
(defun org-roam-file-p (&optional file)
|
|
"Return t if FILE is an Org-roam file, nil otherwise.
|
|
If FILE is not specified, use the current buffer's file-path.
|
|
|
|
FILE is an Org-roam file if:
|
|
- It's located somewhere under `org-roam-directory'
|
|
- It has a matching file extension (`org-roam-file-extensions')
|
|
- It doesn't match excluded regexp (`org-roam-file-exclude-regexp')"
|
|
(let* ((path (or file (buffer-file-name (buffer-base-buffer))))
|
|
(ext (when path (org-roam--file-name-extension path)))
|
|
(ext (if (string= ext "gpg")
|
|
(org-roam--file-name-extension (file-name-sans-extension path))
|
|
ext)))
|
|
(save-match-data
|
|
(and
|
|
path
|
|
(member ext org-roam-file-extensions)
|
|
(not (and org-roam-file-exclude-regexp
|
|
(string-match-p org-roam-file-exclude-regexp path)))
|
|
(f-descendant-of-p path (expand-file-name org-roam-directory))))))
|
|
|
|
(defun org-roam-list-files ()
|
|
"Return a list of all Org-roam files under `org-roam-directory'.
|
|
See `org-roam-file-p' for how each file is determined to be as
|
|
part of Org-Roam."
|
|
(org-roam--list-files (expand-file-name org-roam-directory)))
|
|
|
|
(defun org-roam-buffer-p (&optional buffer)
|
|
"Return t if BUFFER is for an Org-roam file.
|
|
If BUFFER is not specified, use the current buffer."
|
|
(let ((buffer (or buffer (current-buffer)))
|
|
path)
|
|
(with-current-buffer buffer
|
|
(and (derived-mode-p 'org-mode)
|
|
(setq path (buffer-file-name (buffer-base-buffer)))
|
|
(org-roam-file-p path)))))
|
|
|
|
(defun org-roam-buffer-list ()
|
|
"Return a list of buffers that are Org-roam files."
|
|
(--filter (org-roam-buffer-p it)
|
|
(buffer-list)))
|
|
|
|
(defun org-roam--file-name-extension (filename)
|
|
"Return file name extension for FILENAME.
|
|
Like `file-name-extension', but does not strip version number."
|
|
(save-match-data
|
|
(let ((file (file-name-nondirectory filename)))
|
|
(if (and (string-match "\\.[^.]*\\'" file)
|
|
(not (eq 0 (match-beginning 0))))
|
|
(substring file (+ (match-beginning 0) 1))))))
|
|
|
|
(defun org-roam--list-files (dir)
|
|
"Return all Org-roam files located recursively within DIR.
|
|
Use external shell commands if defined in `org-roam-list-files-commands'."
|
|
(let (path exe)
|
|
(cl-dolist (cmd org-roam-list-files-commands)
|
|
(pcase cmd
|
|
(`(,e . ,path)
|
|
(setq path (executable-find path)
|
|
exe (symbol-name e)))
|
|
((pred symbolp)
|
|
(setq path (executable-find (symbol-name cmd))
|
|
exe (symbol-name cmd)))
|
|
(wrong-type
|
|
(signal 'wrong-type-argument
|
|
`((consp symbolp)
|
|
,wrong-type))))
|
|
(when path (cl-return)))
|
|
(if-let* ((files (when path
|
|
(let ((fn (intern (concat "org-roam--list-files-" exe))))
|
|
(unless (fboundp fn) (user-error "%s is not an implemented search method" fn))
|
|
(funcall fn path (format "\"%s\"" dir)))))
|
|
(files (seq-filter #'org-roam-file-p files))
|
|
(files (mapcar #'expand-file-name files))) ; canonicalize names
|
|
files
|
|
(org-roam--list-files-elisp dir))))
|
|
|
|
(defun org-roam--shell-command-files (cmd)
|
|
"Run CMD in the shell and return a list of files.
|
|
If no files are found, an empty list is returned."
|
|
(--> cmd
|
|
(shell-command-to-string it)
|
|
(ansi-color-filter-apply it)
|
|
(split-string it "\n")
|
|
(seq-filter #'s-present? it)))
|
|
|
|
(defun org-roam--list-files-search-globs (exts)
|
|
"Given EXTS, return a list of search globs.
|
|
E.g. (\".org\") => (\"*.org\" \"*.org.gpg\")"
|
|
(cl-loop for e in exts
|
|
append (list (format "\"*.%s\"" e)
|
|
(format "\"*.%s.gpg\"" e))))
|
|
|
|
(defun org-roam--list-files-find (executable dir)
|
|
"Return all Org-roam files under DIR, using \"find\", provided as EXECUTABLE."
|
|
(let* ((globs (org-roam--list-files-search-globs org-roam-file-extensions))
|
|
(names (s-join " -o " (mapcar (lambda (glob) (concat "-name " glob)) globs)))
|
|
(command (s-join " " `(,executable "-L" ,dir "-type f \\(" ,names "\\)"))))
|
|
(org-roam--shell-command-files command)))
|
|
|
|
(defun org-roam--list-files-fd (executable dir)
|
|
"Return all Org-roam files under DIR, using \"fd\", provided as EXECUTABLE."
|
|
(let* ((globs (org-roam--list-files-search-globs org-roam-file-extensions))
|
|
(extensions (s-join " -e " (mapcar (lambda (glob) (substring glob 2 -1)) globs)))
|
|
(command (s-join " " `(,executable "-L" ,dir "--type file" ,extensions))))
|
|
(org-roam--shell-command-files command)))
|
|
|
|
(defalias 'org-roam--list-files-fdfind #'org-roam--list-files-fd)
|
|
|
|
(defun org-roam--list-files-rg (executable dir)
|
|
"Return all Org-roam files under DIR, using \"rg\", provided as EXECUTABLE."
|
|
(let* ((globs (org-roam--list-files-search-globs org-roam-file-extensions))
|
|
(command (s-join " " `(,executable "-L" ,dir "--files"
|
|
,@(mapcar (lambda (glob) (concat "-g " glob)) globs)))))
|
|
(org-roam--shell-command-files command)))
|
|
|
|
(defun org-roam--list-files-elisp (dir)
|
|
"Return all Org-roam files under DIR, using Elisp based implementation."
|
|
(let ((regex (concat "\\.\\(?:"(mapconcat
|
|
#'regexp-quote org-roam-file-extensions
|
|
"\\|" )"\\)\\(?:\\.gpg\\)?\\'"))
|
|
result)
|
|
(dolist (file (org-roam--directory-files-recursively dir regex nil nil t) result)
|
|
(when (and (file-readable-p file)
|
|
(org-roam-file-p file))
|
|
(push file result)))))
|
|
|
|
;;; Package bootstrap
|
|
(provide 'org-roam)
|
|
|
|
(cl-eval-when (load eval)
|
|
(require 'org-roam-compat)
|
|
(require 'org-roam-utils)
|
|
(require 'org-roam-db)
|
|
(require 'org-roam-node)
|
|
(require 'org-roam-capture)
|
|
(require 'org-roam-mode)
|
|
(require 'org-roam-migrate))
|
|
|
|
;;; org-roam.el ends here
|