her.esy.fun/project.el

301 lines
12 KiB
EmacsLisp
Raw Normal View History

2019-08-23 20:03:53 +00:00
;; sign it with
;; gpg --local-user yann@esposito.host --output project.el.sig --detach-sign project.el
2019-08-31 12:21:03 +00:00
(defvar domainname "https://her.esy.fun")
2019-10-20 19:56:27 +00:00
(defvar root-dir (projectile-project-root))
(defvar base-dir (concat root-dir "src"))
(defvar publish-dir (concat root-dir "_site"))
2019-08-31 12:21:03 +00:00
(defvar assets-dir (concat base-dir "/"))
(defvar publish-assets-dir (concat publish-dir "/"))
(defvar posts-dir (concat base-dir "/posts"))
(defvar rss-title "Subscribe to articles")
(defvar posts-descr "Articles")
2019-11-22 03:23:04 +00:00
(defvar css-path "/css/mk.css")
2019-08-31 12:21:03 +00:00
(defvar author-name "Yann Esposito")
(defvar author-email "yann@esposito.host")
2019-08-20 13:35:54 +00:00
(require 'org)
(require 'ox-publish)
(require 'ox-html)
(require 'org-element)
2019-08-31 22:04:23 +00:00
;; (setq org-link-file-path-type 'relative)
2019-08-20 13:35:54 +00:00
(setq org-publish-timestamp-directory
(concat (projectile-project-root) "_cache/"))
(defvar org-blog-head
(concat
2019-12-05 13:55:51 +00:00
"<link rel=\"stylesheet\" href=\"" css-path "\"/>"
"<link rel=\"alternate\" type=\"application/rss+xml\" href=\"/rss.xml\" />"
"<link rel=\"icon\" href=\"/favicon.ico\">"))
2019-08-20 13:35:54 +00:00
(defun menu (lst)
"Blog menu"
(concat
2019-11-19 03:56:12 +00:00
"<nav>"
2019-08-20 13:35:54 +00:00
(mapconcat 'identity
(append
'("<a href=\"/index.html\">Home</a>"
2019-09-01 06:53:11 +00:00
"<a href=\"/archive.html\">Posts</a>"
2019-08-23 20:03:53 +00:00
"<a href=\"/slides.html\">Slides</a>"
2019-08-20 13:35:54 +00:00
"<a href=\"/about-me.html\">About</a>")
lst)
" | ")
2019-11-19 03:56:12 +00:00
"</nav>"))
2019-08-20 13:35:54 +00:00
(defun get-from-info (info k)
(let ((i (car (plist-get info k))))
(when (and i (stringp i))
i)))
(defun org-blog-preamble (info)
"Pre-amble for whole blog."
(concat
"<div class=\"content\">"
(when-let ((date (plist-get info :date)))
(format "<span class=\"article-date\">%s</span>"
(format-time-string "%Y-%m-%d"
(org-timestamp-to-time
(car date)))))
2019-11-10 21:15:00 +00:00
"<h1>"
(format "%s" (car (plist-get info :title)))
"</h1>"
2019-08-20 13:35:54 +00:00
(when-let ((subtitle (car (plist-get info :subtitle))))
(format "<h2>%s</h2>" subtitle))
"</div>"))
(defun rand-obfs (c)
(let ((r (% (random) 20)))
(cond ;; ((eq 0 r) (format "%c" c))
((<= 0 r 10) (format "&#%d;" c))
(t (format "&#x%X;" c)))))
(defun obfuscate-html (txt)
(apply 'concat
(mapcar 'rand-obfs txt)))
(defun org-blog-postamble (info)
"Post-amble for whole blog."
(concat
"<div class=\"content\">"
;; TODO install a comment system
;; (let ((url (format "%s%s" domainname (replace-regexp-in-string base-dir "" (plist-get info :input-file)))))
;; (format "<a href=\"https://comments.esy.fun/slug/%s\">comment</a>"
;; (url-hexify-string url)))
"<footer>"
(when-let ((author (get-from-info info :author)))
(if-let ((email (plist-get info :email)))
(let* ((obfs-email (obfuscate-html email))
(obfs-author (obfuscate-html author))
(obfs-title (obfuscate-html (get-from-info info :title)))
(full-email (format "%s &lt;%s&gt;" obfs-author obfs-email)))
(format "<div class=\"author\">Author: <a href=\"%s%s%s%s\">%s</a></div>"
(obfuscate-html "mailto:")
full-email
(obfuscate-html "?subject=yblog: ")
obfs-title
full-email))
(format "<div class=\"author\">Author: %s</div>" author)))
(when-let ((date (get-from-info info :date)))
(format "<div class=\"date\">Created: %s (%s)</div>" date (y-date date)))
(when-let ((keywords (plist-get info :keywords)))
(format "<div class=\"keywords\">Keywords: <code>%s</code></div>" keywords))
2019-12-05 13:55:51 +00:00
"<div class=\"rss\"><a rel=\"alternate\" type=\"application/rss+xml\" href=\"/rss.xml\">RSS</a>: <a href=\"https://validator.w3.org/feed/check.cgi?url=https%3A%2F%2Fher.esy.fun%2Frss.xml\">Valid RSS</a></div>"
2019-08-20 13:35:54 +00:00
(format "<div class=\"date\">Generated: %s</div>"
(format-time-string "%Y-%m-%d %H:%M:%S"))
2019-11-12 23:36:17 +00:00
"<div class=\"web-file-size\">Size: XXK (HTML: XXK, CSS: XXK, IMG: XXK)</div>"
2019-08-20 13:35:54 +00:00
(format (concat "<div class=\"creator\"> Generated with "
"<a href=\"https://www.gnu.org/software/emacs/\" target=\"_blank\" rel=\"noopener noreferrer\">Emacs %s</a>, "
"<a href=\"http://spacemacs.org\" target=\"_blank\" rel=\"noopener noreferrer\">Spacemacs %s</a>, "
"<a href=\"http://orgmode.org\" target=\"_blank\" rel=\"noopener noreferrer\">Org Mode %s</a>"
"</div>")
emacs-version spacemacs-version org-version)
"</footer>"
(menu '("<a href=\"#preamble\">↑ Top ↑</a>"))
"</div>"))
2019-09-12 12:41:53 +00:00
(defun y/org-get-keywords ()
(org-element-map (org-element-parse-buffer 'element) 'keyword
(lambda (keyword) (cons (org-element-property :key keyword)
(org-element-property :value keyword)))))
(defun y/org-get-meta (keyword)
(cdr (assoc keyword (y/org-get-keywords))))
(defun y/get-meta (file meta-name)
"Return the value of the meta of an org-mode file.
(y/get-meta file \"DESCRIPTION\")
"
(org-babel-with-temp-filebuffer file (y/org-get-meta meta-name)))
2019-08-31 22:04:23 +00:00
(defun date-format-entry (entry _style project)
"Return string for each ENTRY in PROJECT."
2019-09-01 06:53:11 +00:00
(when (string-match "posts/.*" entry)
(let* ((file (org-publish--expand-file-name entry project))
(title (org-publish-find-title entry project))
2019-09-12 12:41:53 +00:00
(date (format-time-string "%Y-%m-%d" (org-publish-find-date entry project)))
2019-09-16 09:42:49 +00:00
(keywords (split-string (y/get-meta file "KEYWORDS") ",\s*"))
2019-09-12 12:41:53 +00:00
(description (y/get-meta file "DESCRIPTION")))
2019-09-16 09:42:49 +00:00
(concat
"- "
(format " [%s]" date)
(format " *[[file:%s][%s]]*" file title)
(format " @@html:<div class=\"keywords\">@@%s@@html:</div>@@"
(mapconcat (lambda (k) (format "@@html:<span class=\"keyword\">@@#%s@@html:</span>@@" k))
(cl-sort keywords 'string-lessp :key 'downcase)
" "))
(format "@@html:<div class=\"description\">@@%s@@html:</div>@@" description)))))
2019-08-20 13:35:54 +00:00
2019-08-31 12:21:03 +00:00
(defun org-blog-sitemap-fn-descr (descr title list)
2019-08-20 13:35:54 +00:00
"Return sitemap using TITLE and LIST returned by `org-blog-sitemap-format-entry'."
(concat "#+TITLE: " title "\n"
"#+AUTHOR: " author-name "\n"
"#+EMAIL: " author-email "\n"
2019-08-31 12:21:03 +00:00
"#+DESCRIPTION: " descr "\n"
2019-11-27 18:06:42 +00:00
""
"@@html:<nav>"
2019-11-27 16:07:25 +00:00
"<a href=\"/index.html\">Home</a> | "
"<a href=\"/archive.html\">Posts</a> | "
"<a href=\"/slides.html\">Slides</a> | "
"<a href=\"/about-me.html\">About</a>"
2019-11-27 18:06:42 +00:00
"</nav>@@"
"\n\n"
2019-09-01 06:53:11 +00:00
(mapconcat (lambda (li) (format "%s" (car li)))
2019-08-20 13:35:54 +00:00
(seq-filter #'car (cdr list))
2019-08-31 12:21:03 +00:00
"\n")))
2019-08-20 13:35:54 +00:00
2019-09-16 09:42:49 +00:00
(defun org-blog-prepare (project-plist)
"With help from `https://github.com/howardabrams/dot-files'.
Touch `archive.org' to rebuilt it.
Argument `PROJECT-PLIST' contains information about the current project."
(let* ((base-directory (plist-get project-plist :base-directory))
(buffer (find-file-noselect (expand-file-name "archive.org" base-directory) t)))
(with-current-buffer buffer
(set-buffer-modified-p t)
(save-buffer 0))
(kill-buffer buffer)))
2019-08-20 13:35:54 +00:00
(defun org-blog-publish-to-html (plist filename pub-dir)
"Same as `org-html-publish-to-html' but modifies html before finishing."
2019-11-22 03:23:04 +00:00
(let* ((file-path (org-html-publish-to-html plist filename pub-dir))
(mk-path (format "./%s.html" (replace-regexp-in-string ".*/\\([^/]*\\)\\.org$" "\\1" filename)))
2019-11-22 17:02:42 +00:00
(min-path (format "./%s-min.html" (replace-regexp-in-string ".*/\\([^/]*\\)\\.org$" "\\1" filename)))
2019-11-22 03:23:04 +00:00
(sci-path (format "./%s-sci.html" (replace-regexp-in-string ".*/\\([^/]*\\)\\.org$" "\\1" filename)))
(modern-path (format "./%s-modern.html" (replace-regexp-in-string ".*/\\([^/]*\\)\\.org$" "\\1" filename))))
2019-08-20 13:35:54 +00:00
(with-current-buffer (find-file-noselect file-path)
(goto-char (point-min))
(search-forward "<body>")
(insert (mapconcat 'identity
2019-11-22 03:23:04 +00:00
`("<input name=\"t\" type=\"radio\" id=\"l\">"
2019-11-19 15:18:41 +00:00
"<input name=\"t\" type=\"radio\" id=\"d\">"
2019-08-24 22:32:42 +00:00
2019-08-20 13:35:54 +00:00
"<div id=\"labels\">"
2019-09-01 21:27:15 +00:00
"<div class=\"content\">"
2019-11-27 16:07:25 +00:00
"<a id=\"h\" href=\"/index.html\">her.esy.fun</a> - "
2019-11-19 15:18:41 +00:00
"<label for=\"l\">light</label>"
2019-09-01 21:27:15 +00:00
" / "
2019-11-19 15:18:41 +00:00
"<label for=\"d\">dark</label>"
2019-11-28 16:56:57 +00:00
" - "
2019-11-22 03:23:04 +00:00
,(format "<a href=\"%s\">mk</a>" mk-path)
2019-11-22 17:02:42 +00:00
,(format "<a href=\"%s\">min</a>" min-path)
2019-11-22 03:23:04 +00:00
,(format "<a href=\"%s\">sci</a>" sci-path)
,(format "<a href=\"%s\">modern</a>" modern-path)
2019-09-01 21:27:15 +00:00
"</div>"
"</div>"
"<div class=\"main\">")
2019-11-22 03:23:04 +00:00
"\n"))
2019-09-01 21:27:15 +00:00
(goto-char (point-max))
(search-backward "</body>")
(insert "\n</div>\n")
2019-08-20 13:35:54 +00:00
(save-buffer)
(kill-buffer))
file-path))
(defun org-blog-publish-attachment (plist filename pub-dir)
"Publish a file with no transformation of any kind.
FILENAME is the filename of the Org file to be published. PLIST
is the property list for the given project. PUB-DIR is the
publishing directory.
Take care of minimizing the pictures using imagemagick.
Return output file name."
(unless (file-directory-p pub-dir)
(make-directory pub-dir t))
(or (equal (expand-file-name (file-name-directory filename))
(file-name-as-directory (expand-file-name pub-dir)))
(let ((dst-file (expand-file-name (file-name-nondirectory filename) pub-dir)))
2019-10-20 19:56:27 +00:00
(cond ((string-match-p ".*\\.\\(png\\|jpg\\|gif\\)$" filename)
2019-11-22 00:42:48 +00:00
(shell-command (format "~/.nix-profile/bin/convert %s -resize 400x400\\> -colorspace Gray -ordered-dither o8x8,8 %s"
2019-10-20 19:56:27 +00:00
filename
dst-file)))
((string-match-p ".*\\.css$" filename)
(shell-command (format "%s/compresscss.sh %s %s" root-dir filename dst-file)))
(t (copy-file filename dst-file t))))))
2019-08-20 13:35:54 +00:00
2019-08-31 12:21:03 +00:00
(defalias 'org-blog-posts-sitemap-fn
(apply-partially 'org-blog-sitemap-fn-descr posts-descr))
2020-01-04 00:15:07 +00:00
(defun org-blog-optimize (_)
(let ((default-directory root-dir))
(message "executing ./pre-deploy.sh")
(shell-command "./pre-deploy.sh")))
2019-11-04 08:03:38 +00:00
(setq org-html-htmlize-output-type 'css)
(setq org-html-htmlize-font-prefix "org-")
2019-08-20 13:35:54 +00:00
(setq org-publish-project-alist
`(("orgfiles"
:base-directory ,base-dir
2019-09-30 13:10:39 +00:00
:exclude ".*drafts/.*"
2019-08-20 13:35:54 +00:00
:base-extension "org"
:publishing-directory ,publish-dir
:recursive t
2019-09-16 09:42:49 +00:00
:preparation-function org-blog-prepare
2019-08-20 13:35:54 +00:00
:publishing-function org-blog-publish-to-html
2020-01-04 00:15:07 +00:00
:completion-function org-blog-optimize
2019-08-20 13:35:54 +00:00
:with-toc nil
:with-title nil
:with-date t
:section-numbers nil
:html-doctype "html5"
:html-html5-fancy t
:html-head-include-default-style nil
:html-head-include-scripts nil
:htmlized-source t
:html-head-extra ,org-blog-head
:html-preamble org-blog-preamble
:html-postamble org-blog-postamble
2019-08-31 13:45:32 +00:00
:auto-sitemap t
2019-09-01 06:53:11 +00:00
:sitemap-filename "archive.org"
2019-08-31 13:45:32 +00:00
:sitemap-title "Articles"
:sitemap-style list
2019-08-31 22:04:23 +00:00
:sitemap-sort-files anti-chronologically
:sitemap-format-entry date-format-entry
:sitemap-function org-blog-posts-sitemap-fn)
2019-08-31 13:45:32 +00:00
2019-08-20 13:35:54 +00:00
("assets"
:base-directory ,assets-dir
:base-extension ".*"
:exclude ".*\.org$"
:publishing-directory ,publish-assets-dir
:publishing-function org-blog-publish-attachment
:recursive t)
2020-01-04 00:15:07 +00:00
("blog" :components ("assets" "orgfiles"))))
2019-08-20 13:35:54 +00:00
;; add target=_blank and rel="noopener noreferrer" to all links by default
(defun my-org-export-add-target-blank-to-http-links (text backend info)
"Add target=\"_blank\" to external links."
(when (and
(org-export-derived-backend-p backend 'html)
(string-match "href=\"http[^\"]+" text)
(not (string-match "target=\"" text))
(not (string-match (concat "href=\"" domainname "[^\"]*") text)))
(string-match "<a " text)
(replace-match "<a target=\"_blank\" rel=\"noopener noreferrer\" " nil nil text)))
(add-to-list 'org-export-filter-link-functions
'my-org-export-add-target-blank-to-http-links)
(provide 'her-esy-fun-publish)