chore(users/sterni/mblog): rename apple-note to mail-note
The type identifier Apple uses is com.apple.mail-note, so “Mail Note” is actually the best way to refer to this format. Not only doesn't it include a trademark, but it's also more accurate. The iOS and macOS Notes.app(s) allow authoring Notes to be saved in iCloud which seems to use a different API and/or storage format (at least these notes are no longer accessible via IMAP). In this sense they are “Apple Notes”, but not “Mail Notes”. Change-Id: I2fd3d3bd253ed39adf7965008290f7d1e622831d Reviewed-on: https://cl.tvl.fyi/c/depot/+/12815 Autosubmit: sterni <sternenseemann@systemli.org> Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
This commit is contained in:
parent
b8e4da856f
commit
0ead86ec89
6 changed files with 60 additions and 60 deletions
|
|
@ -1,118 +0,0 @@
|
|||
;; SPDX-License-Identifier: GPL-3.0-only
|
||||
;; SPDX-FileCopyrightText: Copyright (C) 2022-2023 by sterni
|
||||
|
||||
(in-package :note)
|
||||
(declaim (optimize (safety 3)))
|
||||
|
||||
;;; util
|
||||
|
||||
(defun html-escape-stream (in out)
|
||||
"Escape characters read from stream IN and write them to
|
||||
stream OUT escaped using WHO:ESCAPE-STRING-MINIMAL."
|
||||
(let ((buf (make-string config:*general-buffer-size*)))
|
||||
(loop for len = (read-sequence buf in)
|
||||
while (> len 0)
|
||||
do (write-string (who:escape-string-minimal (subseq buf 0 len)) out))))
|
||||
|
||||
(defun cid-header-value (cid)
|
||||
"Takes a Content-ID as present in Apple Notes' <object> tags and properly
|
||||
surrounds them with angle brackets for a MIME header"
|
||||
(concatenate 'string "<" cid ">"))
|
||||
|
||||
(defun find-mime-message-date (message)
|
||||
(when-let ((date-string (car (mime:mime-message-header-values "Date" message))))
|
||||
(date-time-parser:parse-date-time date-string)))
|
||||
|
||||
;;; main implementation
|
||||
|
||||
(defun apple-note-mime-subtype-p (x)
|
||||
(member x '("plain" "html") :test #'string-equal))
|
||||
|
||||
(deftype apple-note-mime-subtype ()
|
||||
'(satisfies apple-note-mime-subtype-p))
|
||||
|
||||
(defclass apple-note (mime:mime-message)
|
||||
((text-part
|
||||
:type mime:mime-text
|
||||
:initarg :text-part
|
||||
:reader apple-note-text-part)
|
||||
(subject
|
||||
:type string
|
||||
:initarg :subject
|
||||
:reader apple-note-subject)
|
||||
(uuid
|
||||
:type string
|
||||
:initarg :uuid
|
||||
:reader apple-note-uuid)
|
||||
(time
|
||||
:type integer
|
||||
:initarg :time
|
||||
:reader apple-note-time)
|
||||
(mime-subtype
|
||||
:type apple-note-mime-subtype
|
||||
:initarg :mime-subtype
|
||||
:reader apple-note-mime-subtype))
|
||||
(:documentation
|
||||
"Representation of a Note created using Apple's Notes via the IMAP backend"))
|
||||
|
||||
(defun apple-note-p (msg)
|
||||
"Checks X-Uniform-Type-Identifier of a MIME:MIME-MESSAGE
|
||||
to determine if a given mime message claims to be an Apple Note."
|
||||
(when-let (uniform-id (car (mime:mime-message-header-values
|
||||
"X-Uniform-Type-Identifier"
|
||||
msg)))
|
||||
(string-equal uniform-id "com.apple.mail-note")))
|
||||
|
||||
(defun make-apple-note (msg)
|
||||
(check-type msg mime-message)
|
||||
|
||||
(unless (apple-note-p msg)
|
||||
(error "Passed message is not an Apple Note according to headers"))
|
||||
|
||||
(let ((text-part (mime:find-mime-text-part msg))
|
||||
(subject (car (mime:mime-message-header-values "Subject" msg :decode t)))
|
||||
(uuid (when-let ((val (car (mime:mime-message-header-values
|
||||
"X-Universally-Unique-Identifier"
|
||||
msg))))
|
||||
(string-downcase val)))
|
||||
(time (find-mime-message-date msg)))
|
||||
;; The idea here is that we don't need to check a lot manually, instead
|
||||
;; the type annotation are going to do this for us (with sufficient safety?)
|
||||
(change-class msg 'apple-note
|
||||
:text-part text-part
|
||||
:subject subject
|
||||
:uuid uuid
|
||||
:time time
|
||||
:mime-subtype (mime:mime-subtype text-part))))
|
||||
|
||||
(defgeneric apple-note-html-fragment (note out)
|
||||
(:documentation
|
||||
"Takes an APPLE-NOTE and writes its text content as HTML to
|
||||
the OUT stream. The <object> tags are resolved to <img> which
|
||||
refer to the respective attachment's filename as a relative path,
|
||||
but extraction of the attachments must be done separately. The
|
||||
surrounding <html> and <body> tags are stripped and <head>
|
||||
discarded completely, so only a fragment which can be included
|
||||
in custom templates remains."))
|
||||
|
||||
(defmethod apple-note-html-fragment ((note apple-note) (out stream))
|
||||
(let ((text (apple-note-text-part note)))
|
||||
(cond
|
||||
;; notemap creates text/plain notes we need to handle properly.
|
||||
;; Additionally we *could* check X-Mailer which notemap sets
|
||||
((string-equal (apple-note-mime-subtype note) "plain")
|
||||
(html-escape-stream (mime:mime-body-stream text) out))
|
||||
;; Notes.app creates text/html parts
|
||||
((string-equal (apple-note-mime-subtype note) "html")
|
||||
(closure-html:parse
|
||||
(mime:mime-body-stream text)
|
||||
(make-instance
|
||||
'apple-note-transformer
|
||||
:cid-lookup
|
||||
(lambda (cid)
|
||||
(when-let* ((part (mime:find-mime-part-by-id note (cid-header-value cid)))
|
||||
(file (mime:mime-part-file-name part)))
|
||||
file))
|
||||
:next-handler
|
||||
(closure-html:make-character-stream-sink out))))
|
||||
(t (error "Internal error: unexpected MIME subtype")))))
|
||||
Loading…
Add table
Add a link
Reference in a new issue