Last active
September 18, 2024 06:45
-
-
Save zot/ddf1a89a567fea73bc3c8a209d48f527 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | |
;; | |
;; BACK UP YOUR LOGSEQ DIR BEFORE RUNNING THIS! | |
;; | |
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | |
;; Copyright (C) Aug 4 2022, William R. Burdick Jr. | |
;; | |
;; LICENSE | |
;; This code is dual-licensed with MIT and GPL licenses. | |
;; Take your pick and abide by whichever license appeals to you. | |
;; | |
;; logseq compatibility | |
;; put ids and titles at the tops of non-journal files | |
;; change fuzzy links from [[PAGE]] to [[id:2324234234][PAGE]] | |
;; also change file links to id links, provided that the links | |
;; expand to file names that have ids in the roam database. | |
;; | |
;; NOTE: this currently only converts fuzzy links. | |
;; If you have the setting :org-mode/insert-file-link? true in your Logseq config, | |
;; it won't convert the resulting links. | |
;; | |
;; Your logseq directory should be inside your org-roam directory, | |
;; put the directory you use here | |
(defvar bill/logseq-folder (f-expand (f-join org-roam-directory "zettel"))) | |
;; You probably don't need to change these values | |
(defvar bill/logseq-pages (f-expand (f-join bill/logseq-folder "pages"))) | |
(defvar bill/logseq-journals (f-expand (f-join bill/logseq-folder "journals"))) | |
;;(defvar bill/rich-text-types [bold italic subscript link strike-through superscript underline inline-src-block footnote-reference inline-babel-call entity]) | |
(defvar bill/rich-text-types '(bold italic subscript link strike-through superscript underline inline-src-block)) | |
;; ignore files matching bill/logseq-exclude-pattern | |
;; example: (defvar bill/logseq-exclude-pattern (string "^" bill/logseq-folder "/bak/.*$")) | |
(defvar bill/logseq-exclude-pattern "^$") | |
(defun bill/textify (headline) | |
(save-excursion | |
(apply 'concat (flatten-list | |
(bill/textify-all (org-element-property :title headline)))))) | |
(defun bill/textify-all (nodes) (mapcar 'bill/subtextify nodes)) | |
(defun bill/with-length (str) (cons (length str) str)) | |
(defun bill/subtextify (node) | |
(cond ((not node) "") | |
((stringp node) (substring-no-properties node)) | |
((member (org-element-type node) bill/rich-text-types) | |
(list (bill/textify-all (cddr node)) | |
(if (> (org-element-property :post-blank node)) | |
(make-string (org-element-property :post-blank node) ?\s) | |
""))) | |
(t ""))) | |
(defun bill/logseq-journal-p (file) (string-match-p (concat "^" bill/logseq-journals) file)) | |
(defun bill/ensure-file-id (file) | |
"Visit an existing file, ensure it has an id, return whether the a new buffer was created" | |
(setq file (f-expand file)) | |
(if (bill/logseq-journal-p file) | |
`(nil . nil) | |
(let* ((buf (get-file-buffer file)) | |
(was-modified (buffer-modified-p buf)) | |
(new-buf nil) | |
has-data | |
org | |
changed | |
sec-end) | |
(when (not buf) | |
(setq buf (find-file-noselect file)) | |
(setq new-buf t)) | |
(set-buffer buf) | |
(setq org (org-element-parse-buffer)) | |
(setq has-data (cddr org)) | |
(goto-char 1) | |
(when (not (and (eq 'section (org-element-type (nth 2 org))) (org-roam-id-at-point))) | |
;; this file has no file id | |
(setq changed t) | |
(when (eq 'headline (org-element-type (nth 2 org))) | |
;; if there's no section before the first headline, add one | |
(insert "\n") | |
(goto-char 1)) | |
(org-id-get-create) | |
(setq org (org-element-parse-buffer))) | |
(when (nth 3 org) | |
(when (not (org-collect-keywords ["title"])) | |
;; no title -- ensure there's a blank line at the section end | |
(setq changed t) | |
(setq sec-end (org-element-property :end (nth 2 org))) | |
(goto-char (1- sec-end)) | |
(when (and (not (equal "\n\n" (buffer-substring-no-properties (- sec-end 2) sec-end)))) | |
(insert "\n") | |
(goto-char (1- (point))) | |
(setq org (org-element-parse-buffer))) | |
;; copy the first headline to the title | |
(insert (format "#+title: %s" (string-trim (bill/textify (nth 3 org))))))) | |
;; ensure org-roam knows about the new id and/or title | |
(when changed (save-buffer)) | |
(cons new-buf buf)))) | |
(defun bill/convert-logseq-file (buf) | |
"convert fuzzy and file:../pages logseq links in the file to id links" | |
(save-excursion | |
(let* (changed | |
link) | |
(set-buffer buf) | |
(goto-char 1) | |
(while (search-forward "[[" nil t) | |
(setq link (org-element-context)) | |
(setq newlink (bill/reformat-link link)) | |
(when newlink | |
(setq changed t) | |
(goto-char (org-element-property :begin link)) | |
(delete-region (org-element-property :begin link) (org-element-property :end link)) | |
;; note, this format string is reall =[[%s][%s]]= but =%= is a markup char so one's hidden | |
(insert newlink))) | |
;; ensure org-roam knows about the changed links | |
(when changed (save-buffer))))) | |
(defun bill/reformat-link (link) | |
(let (filename | |
id | |
linktext | |
newlink) | |
(when (eq 'link (org-element-type link)) | |
(when (equal "fuzzy" (org-element-property :type link)) | |
(setq filename (f-expand (f-join bill/logseq-pages | |
(concat (org-element-property :path link) ".org")))) | |
(setq linktext (org-element-property :raw-link link))) | |
(when (equal "file" (org-element-property :type link)) | |
(setq filename (f-expand (org-element-property :path link))) | |
(if (org-element-property :contents-begin link) | |
(setq linktext (buffer-substring-no-properties | |
(org-element-property :contents-begin link) | |
(org-element-property :contents-end link))) | |
(setq linktext (buffer-substring-no-properties | |
(+ (org-element-property :begin link) 2) | |
(- (org-element-property :end link) 2))))) | |
(when (and filename (f-exists-p filename)) | |
(setq id (caar (org-roam-db-query [:select id :from nodes :where (like file $s1)] | |
filename))) | |
(when id | |
(setq newlink (format "[[id:%s][%s]]%s" | |
id | |
linktext | |
(if (> (org-element-property :post-blank link)) | |
(make-string (org-element-property :post-blank link) ?\s) | |
""))) | |
(when (not (equal newlink | |
(buffer-substring-no-properties | |
(org-element-property :begin link) | |
(org-element-property :end link)))) | |
newlink)))))) | |
(defun bill/roam-file-modified-p (file-path) | |
(and (not (string-match-p bill/logseq-exclude-pattern file-path)) | |
(let ((content-hash (org-roam-db--file-hash file-path)) | |
(db-hash (caar (org-roam-db-query [:select hash :from files | |
:where (= file $s1)] file-path)))) | |
(not (string= content-hash db-hash))))) | |
(defun bill/modified-logseq-files () | |
(emacsql-with-transaction (org-roam-db) | |
(seq-filter 'bill/roam-file-modified-p | |
(org-roam--list-files bill/logseq-folder)))) | |
(defun bill/check-logseq () | |
(interactive) | |
(let (created | |
files | |
bufs | |
unmodified | |
cur | |
bad | |
buf) | |
(setq files (org-roam--list-files bill/logseq-folder)) | |
;; make sure all the files have file ids | |
(dolist (file-path files) | |
(setq file-path (f-expand file-path)) | |
(setq cur (bill/ensure-file-id file-path)) | |
(setq buf (cdr cur)) | |
(push buf bufs) | |
(when (and (not (bill/logseq-journal-p file-path)) (not buf)) | |
(push file-path bad)) | |
(when (not (buffer-modified-p buf)) | |
(push buf unmodified)) | |
(when (car cur) | |
(push buf created))) | |
;; patch fuzzy links | |
(mapc 'bill/convert-logseq-file (seq-filter 'identity bufs)) | |
(dolist (buf unmodified) | |
(when (buffer-modified-p buf) | |
(save-buffer unmodified))) | |
(mapc 'kill-buffer created) | |
(when bad | |
(message "Bad items: %s" bad)) | |
nil)) |
hmm, maybe this would work better?
(defvar bill/logseq-exclude-pattern "/bak/")
The pattern you used seems to match only files with names like ./bak/.
, i.e. ones whose names start with a single character, then have "/bak/" and then end with only one more character.
Looks like I needed string-join
instead of string
in that example:
I think this should be the exact value for you:
(defvar bill/logseq-exclude-pattern (string-join (list "^" bill/logseq-folder "/bak/.*$")))
Thanks Bill. I’ll test this.
I commented that I had it working with just a hardwired regexp but creating
one from a list is probably a safer approach.
On Thu, 17 Nov 2022 at 11:36 pm, Bill Burdick ***@***.***> wrote:
***@***.**** commented on this gist.
------------------------------
Looks like I needed string-join instead of string in that example:
I think this should be the exact value for you:
(defvar bill/logseq-exclude-pattern (string-join (list "^" bill/logseq-folder "/bak/.*$")))
—
Reply to this email directly, view it on GitHub
<https://gist.github.com/ddf1a89a567fea73bc3c8a209d48f527#gistcomment-4372579>
or unsubscribe
<https://github.com/notifications/unsubscribe-auth/ACFPFQUOZJQZ5EVFRJCK2YTWIYRF7BFKMF2HI4TJMJ2XIZLTSKBKK5TBNR2WLJDHNFZXJJDOMFWWLK3UNBZGKYLEL52HS4DFQKSXMYLMOVS2I5DSOVS2I3TBNVS3W5DIOJSWCZC7OBQXE5DJMNUXAYLOORPWCY3UNF3GS5DZVRZXKYTKMVRXIX3UPFYGLK2HNFZXIQ3PNVWWK3TUUZ2G64DJMNZZDAVEOR4XAZNEM5UXG5FFOZQWY5LFVEYTCNZVGM4TEMRZU52HE2LHM5SXFJTDOJSWC5DF>
.
You are receiving this email because you commented on a thread.
Triage notifications on the go with GitHub Mobile for iOS
<https://apps.apple.com/app/apple-store/id1477376905?ct=notification-email&mt=8&pt=524675>
or Android
<https://play.google.com/store/apps/details?id=com.github.android&referrer=utm_campaign%3Dnotification-email%26utm_medium%3Demail%26utm_source%3Dgithub>
.
--
regards
David Maslen
Thanks for the amazing work here @zot! I needed this as an emacs package and modified it slightly to fit my needs. You can find my version here in case you would like to apply some of the changes back (namely adding ability to work with page aliases). Thanks to the code here I also found a bug in Logseq which I reported to their repo.
Very cool! I'll check it out!
I am sorry that I never thanked you for this, and for your work in creating
and publishing the code. I just wanted to let you know that I — and others
I am sure — appreciate your work. It means I have an iPhone client, which I
use every day, that I can use with org-roam, which I also rely on. It is
thanks to the efforts of people like you that Emacs remains a powerful and
adaptable tool, especially because people like me use Emacs heavily despite
having a minimal understanding of elisp.
Sincerely,
Bill Day
…On Fri, Nov 11, 2022 at 9:30 AM Bill Burdick ***@***.***> wrote:
***@***.**** commented on this gist.
------------------------------
I made the change -- I think this should work. Please let me know if there
are problems. I don't have a ton of free time to devote to this, but I am
reading the comments here...
As the example in the comment says, put a regex pattern in
bill/logseq-exclude-pattern that represents what you want to ignore.
Example:
(defvar bill/logseq-exclude-pattern (string "^" bill/logseq-folder "/bak/.*$"))
—
Reply to this email directly, view it on GitHub
<https://gist.github.com/ddf1a89a567fea73bc3c8a209d48f527#gistcomment-4366306>
or unsubscribe
<https://github.com/notifications/unsubscribe-auth/AAC3BP6QZBBTSBJ6VPKR4L3WHZKADBFKMF2HI4TJMJ2XIZLTSKBKK5TBNR2WLJDHNFZXJJDOMFWWLK3UNBZGKYLEL52HS4DFQKSXMYLMOVS2I5DSOVS2I3TBNVS3W5DIOJSWCZC7OBQXE5DJMNUXAYLOORPWCY3UNF3GS5DZVRZXKYTKMVRXIX3UPFYGLK2HNFZXIQ3PNVWWK3TUUZ2G64DJMNZZDAVEOR4XAZNEM5UXG5FFOZQWY5LFVEYTCNZVGM4TEMRZU52HE2LHM5SXFJTDOJSWC5DF>
.
You are receiving this email because you commented on a thread.
Triage notifications on the go with GitHub Mobile for iOS
<https://apps.apple.com/app/apple-store/id1477376905?ct=notification-email&mt=8&pt=524675>
or Android
<https://play.google.com/store/apps/details?id=com.github.android&referrer=utm_campaign%3Dnotification-email%26utm_medium%3Demail%26utm_source%3Dgithub>
.
--
Bill Day
***@***.***
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Changing to (defvar bill/logseq-exclude-pattern "^./bak/.$")
Seems to work.