💾 Archived View for isoraqathedh.pollux.casa › dirs.org captured on 2021-12-05 at 23:47:19.
View Raw
More Information
⬅️ Previous capture (2021-12-03)
➡️ Next capture (2021-12-17)
-=-=-=-=-=-=-
#+Title: Directory and feed generator
#+Date: 2021-08-25
This file is used to generate a number of files
in the absence of server-side scripting.
I intend for this file to also be usable to others,
as long as you have prerequisites.
Granted, the prerequisites are fairly steep,
as they require org-mode, which in turn needs Emacs,
as well as ~zsh~ and a number of other utilities
that are /normally/ installed with most Linux computers but not always.
This file and all source code within it is licensed under GPLv3 or later.
#+CALL: generate-feed()
#+RESULTS:
[[file:atom.xml]]
#+CALL: make-directory-file()
#+RESULTS:
[[file:dirs.gmi]]
This lists all the gemtext files inside the capsule.
First, we define the global UUID.
The ID is for use in the Atom feed,
but can also be considered to be the ID of the capsule itself.
#+NAME: main-uuid
: 43730995-ccfc-4202-afef-f4ba2b2a42c2
Then, we get all the gemtext files in and below the directory this file is in,
and ask for its path and the creation date.
A version 3 UUID is also generated for the Atom feed.
#+NAME: file-list
#+begin_src shell :var mainuuid=main-uuid filedir=(format "%s" default-directory) :hlines yes
# <<<NOPROMPT>>>
for file in "$filedir"/**/*.gmi; do
echo -n "${file#$filedir/} "
date -r "$file" +"%Y-%m-%dT%H:%M:%S%z" | tr '\n' ' '
uuidgen -m -n "$mainuuid" -N "${file#$filedir}"
done
#+end_src
#+RESULTS: file-list
| aesthetic.gmi | 2021-11-05T17:15:19+0800 | 555d9cea-8ef7-3816-8090-ffdbad45d185 |
| article-conventions.gmi | 2021-09-04T22:50:28+0800 | 3e3e28f1-d4d8-3b7a-a6fc-58e98a90a216 |
| detritus.gmi | 2021-11-05T17:15:56+0800 | a5ee1f9e-e989-3d43-8788-4e38010b5fad |
| digits-10-and-11.gmi | 2021-08-28T20:30:50+0800 | 604fba54-299d-3853-ac3f-c5fa5150488f |
| dirs.gmi | 2021-11-05T17:16:14+0800 | 4ffc8e5c-e98d-3ac4-9682-182df8f66c03 |
| energy-competition.gmi | 2021-10-23T15:57:38+0800 | 3d2215b0-f25f-3424-8df2-7ee2956cd68c |
| gemlog-2021-09b.gmi | 2021-09-29T22:23:23+0800 | 3c0b1833-41a8-39e2-b65c-ca4481325367 |
| gemlog-2021-10.gmi | 2021-10-31T00:46:22+0800 | 13acd661-06d7-3b31-9ee3-c959e6735972 |
| gem-org-odt.gmi | 2021-11-22T11:04:42+0800 | 602080ab-b3ec-382c-86cc-8b71fb758d42 |
| index.gmi | 2021-09-20T14:45:35+0800 | 618744bb-4dff-3bbe-bbe5-70c36d516100 |
| language-list.gmi | 2021-08-28T11:48:03+0800 | a5a5cf51-08b5-3601-b3a1-90169f88e9e5 |
| masquerades.gmi | 2021-08-11T22:27:49+0800 | db47f8aa-749c-3cdb-ada8-b5f859d4406a |
| monetising-conlangs.gmi | 2021-09-17T23:38:55+0800 | 058e3984-2473-3c0c-9213-2b915a37445f |
| notation.gmi | 2021-10-03T22:09:18+0800 | afa2bc53-f0a6-3de3-b2fa-a5e054f22e88 |
| programming.gmi | 2021-08-05T19:33:57+0800 | 99a8c7e9-fc8e-3445-a63c-2d4c989d07e4 |
| re/power-of-git.gmi | 2021-09-21T13:50:15+0800 | dc7bc220-0b7e-38fc-8591-b3ff34f7c00b |
| topic-symbols.gmi | 2021-09-26T22:17:19+0800 | b4027394-01f5-37da-9009-76354306fb76 |
| twitch-stats.gmi | 2021-10-09T22:42:17+0800 | 7210075b-8dd9-31d5-b846-d5f45a75020c |
| ujmj.gmi | 2021-11-11T20:33:34+0800 | 736106f0-4bcc-3823-9255-674e25565d7b |
| worldbuilding.gmi | 2021-08-22T12:58:08+0800 | 18850017-bbc2-3999-9df8-2448003d90ba |
The UUID is version 3 so it can be reliably recovered
using the global UUID and the path
(which works as the namespace and the name respectively);
and since we do not need to concern ourselves with security here,
the MD5 hash is used.
Based on the table above we can make the directory file.
We add a title, a summary and some explanatory text,
as well as a link to this page as the source code.
#+NAME: make-directory-file
#+begin_src shell :var file_list=file-list :results file :file dirs.gmi
gentime=$(date +"%Y-%m-%d %H:%M:%S %Z")
cat <<EOF
# Directory listing
This is a list of Gemini files in this capsule. This includes pages that don't otherwise appear in the full directory listing, and they might not be complete, so be aware.
Each link also carries the title of the document (i.e. the first top-level heading of the file).
It is last generated at $gentime.
=> dirs.org Source code for file generation
EOF
echo $file_list |
while read file date _; do
echo "=> $file ${file%.gmi}: $(grep "^# " $file | head -n 1 | cut -c3-)"
done | sort
#+end_src
#+RESULTS: make-directory-file
[[file:dirs.gmi]]
- Atom feed for creative files
In this section, we generate an Atom feed for the following files:
#+NAME: selected-files
- language-list.gmi
- ujmj.gmi
- digits-10-and-11.gmi
- article-conventions.gmi
- monetising-conlangs.gmi
- re/power-of-git.gmi
- notation.gmi
- twitch-stats.gmi
- energy-competition.gmi
- aesthetic.gmi
- gem-org-odt.gmi
- scheduler.gmi
(For use later)
- masquerades.gmi
- topic-symbols.gmi
The files here are selected as, for lack of a better word, "content files":
they are why the capsule is put up here in the first place.
This is handled using Emacs Lisp's ~xmlgen~
and some fairly involved text-manipulating commands that Emacs Lisp has
(hence the change in language).
The text-manipulation commands are actually just there
to the resulting output human-readable and also visually appealing,
though some actions are done in shell.
The feed uses my name as the author for all files,
and [[ref:file-list]] for the remaining data.
(It's actually why we pre-computed the UUIDs there,
to save on another shell-out).
The summary is the first paragraph
(non-blank line that isn't a heading, quote or link),
and the content is left empty.
#+Name: generate-feed
#+begin_src emacs-lisp :var uuid=main-uuid files=file-list selected=selected-files[,0] :results file :file atom.xml
(with-temp-buffer
;; Write XML
(let ((gemini-host "gemini://isoraqathedh.pollux.casa/")
(https-host "https://isoraqathedh.pollux.casa/")
(title-program "grep \"^#\" \"%s\" | sed '1s/^# //' | head -n 1 | tr -d '\n'")
(summary-program "grep -Pv \"^($|>|#|=>)\" \"%s\" | sed '1s/^# //' | head -n 1 | fold -s -w 70 | sed 's/ $//'"))
(flet ((file->entry (file)
`(entry (link :href ,(concat gemini-host (first file)))
(link :rel "alternate" :href ,(concat https-host (first file)))
(updated ,(second file))
(id "urn:uuid:" ,(third file))
(title ,(shell-command-to-string
(format title-program (first file))))
(summary "\n"
,(shell-command-to-string
(format summary-program (first file))))
(author
(name "isoraqathedh")))))
(insert
"<?xml version=\"1.0\" encoding=\"utf-8\"?>"
(xmlgen `(feed :xmlns "http://www.w3.org/2005/Atom"
(link :href ,gemini-host)
(link :rel "self"
:type "application/atom+xml"
:href ,(concat gemini-host "atom.xml"))
(id "urn:uuid:" ,(string-trim uuid))
(updated ,(format-time-string "%Y-%m-%dT%H:%M:%S%z"))
(generator "dirs.org")
(title "kapsyël")
,@(mapcar #'file->entry
(sort (cl-remove-if-not #'(lambda (x)
(cl-find (first x) selected
:test #'string=))
files)
#'(lambda (x y) (string> (second x) (second y))))))))
;; Format nicely
(nxml-mode)
(goto-char (1+ (point-min)))
(while (re-search-forward "<\\([^/]\\)" nil t nil)
(replace-match "\n<\\1"))
(goto-char (point-min))
(while (re-search-forward "></" nil t nil)
(replace-match ">\n</"))
(indent-region (point-min) (point-max))
(buffer-substring-no-properties (point-min) (point-max)))))
#+end_src
#+RESULTS: generate-feed
[[file:atom.xml]]
#+Name: upload-to-capsule
#+begin_src shell :results silent
find . -mtime -2 -type f |
grep -Pv "(?<=^\./).*/" |
sed "s:^\./:put :" |
sftp pollux:/capsule
#+end_src
#+RESULTS: upload-to-capsule
# LocalWords: gemtext