All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
3 # NeoMutt helper script to create multipart/* emails with Markdown → HTML
4 # alternative conversion, and handling of inline images, using NeoMutt's
5 # ability to manually craft MIME trees, but automating this process.
8 # neomuttrc (needs to be a single line):
9 # set my_mdwn_extensions="extra,admonition,codehilite,sane_lists,smarty"
11 # <enter-command> source '$my_confdir/buildmimetree.py \
12 # --tempdir $tempdir --extensions $my_mdwn_extensions|'<enter>\
13 # <enter-command> sourc e \$my_mdwn_postprocess_cmd_file<enter>\
14 # " "Convert message into a modern MIME tree with inline images"
16 # (Yes, we need to call source twice, as mutt only starts to process output
17 # from a source command when the command exits, and since we need to react
18 # to the output, we need to be invoked again, using a $my_ variable to pass
26 # - Pynliner, provides --css-file and thus inline styling of HTML output
27 # - Pygments, then syntax highlighting for fenced code is enabled
30 # https://git.madduck.net/etc/neomutt.git/blob_plain/HEAD:/.config/neomutt/buildmimetree.py
32 # Copyright © 2023 martin f. krafft <madduck@madduck.net>
33 # Released under the GPL-2+ licence, just like Mutt itself.
43 from collections import namedtuple, OrderedDict
44 from markdown.extensions import Extension
45 from markdown.inlinepatterns import ImageInlineProcessor, IMAGE_LINK_RE
46 from email.utils import make_msgid
47 from urllib import request
50 def parse_cli_args(*args, **kwargs):
51 parser = argparse.ArgumentParser(
53 "NeoMutt helper to turn text/markdown email parts "
54 "into full-fledged MIME trees"
58 "Copyright © 2023 martin f. krafft <madduck@madduck.net>.\n"
59 "Released under the MIT licence"
66 help="Markdown extension to use (comma-separated list)",
74 help="CSS file to merge with the final HTML",
77 parser.set_defaults(css_file=None)
82 help="Only build, don't send the message",
88 help="Specify temporary directory to use for attachments",
94 help="Turn on debug logging of commands generated to stderr",
97 subp = parser.add_subparsers(help="Sub-command parsers", dest="mode")
98 massage_p = subp.add_parser(
99 "massage", help="Massaging phase (internal use)"
102 massage_p.add_argument(
103 "--write-commands-to",
108 help="Temporary file path to write commands to",
111 massage_p.add_argument(
114 help="Turn on debugging to stderr of the MIME tree walk",
117 massage_p.add_argument(
120 help="If provided, the script is invoked as editor on the mail draft",
123 return parser.parse_args(*args, **kwargs)
126 # [ MARKDOWN WRAPPING ] #######################################################
129 InlineImageInfo = namedtuple(
130 "InlineImageInfo", ["cid", "desc"], defaults=[None]
134 class InlineImageExtension(Extension):
135 class RelatedImageInlineProcessor(ImageInlineProcessor):
136 def __init__(self, re, md, ext):
137 super().__init__(re, md)
140 def handleMatch(self, m, data):
141 el, start, end = super().handleMatch(m, data)
142 if "src" in el.attrib:
143 src = el.attrib["src"]
144 if "://" not in src or src.startswith("file://"):
145 # We only inline local content
146 cid = self._ext.get_cid_for_image(el.attrib)
147 el.attrib["src"] = f"cid:{cid}"
148 return el, start, end
152 self._images = OrderedDict()
154 def extendMarkdown(self, md):
155 md.registerExtension(self)
156 inline_image_proc = self.RelatedImageInlineProcessor(
157 IMAGE_LINK_RE, md, self
159 md.inlinePatterns.register(inline_image_proc, "image_link", 150)
161 def get_cid_for_image(self, attrib):
162 msgid = make_msgid()[1:-1]
164 if path.startswith("/"):
165 path = f"file://{path}"
166 self._images[path] = InlineImageInfo(
167 msgid, attrib.get("title", attrib.get("alt"))
171 def get_images(self):
175 def markdown_with_inline_image_support(
176 text, *, extensions=None, extension_configs=None
178 inline_image_handler = InlineImageExtension()
179 extensions = extensions or []
180 extensions.append(inline_image_handler)
181 mdwn = markdown.Markdown(
182 extensions=extensions, extension_configs=extension_configs
184 htmltext = mdwn.convert(text)
186 images = inline_image_handler.get_images()
188 def replace_image_with_cid(matchobj):
189 for m in (matchobj.group(1), f"file://{matchobj.group(1)}"):
191 return f"(cid:{images[m].cid}"
192 return matchobj.group(0)
194 text = re.sub(r"\(([^)\s]+)", replace_image_with_cid, text)
195 return text, htmltext, images
198 # [ CSS STYLING ] #############################################################
209 from pygments.formatters import get_formatter_by_name
211 _CODEHILITE_CLASS = "codehilite"
213 _PYGMENTS_CSS = get_formatter_by_name(
214 "html", style="default"
215 ).get_style_defs(f".{_CODEHILITE_CLASS}")
221 def apply_styling(html, css):
225 .with_cssString("\n".join(s for s in [_PYGMENTS_CSS, css] if s))
230 # [ PARTS GENERATION ] ########################################################
236 ["type", "subtype", "path", "desc", "cid", "orig"],
237 defaults=[None, None, False],
241 ret = f"<{self.type}/{self.subtype}>"
243 ret = f"{ret} cid:{self.cid}"
245 ret = f"{ret} ORIGINAL"
250 namedtuple("Multipart", ["subtype", "children", "desc"], defaults=[None])
253 return f"<multipart/{self.subtype}> children={len(self.children)}"
256 return hash(str(self.subtype) + "".join(str(self.children)))
259 def filewriter_fn(path, content, mode="w", **kwargs):
260 with open(path, mode, **kwargs) as out_f:
264 def collect_inline_images(
265 images, *, tempdir=None, filewriter_fn=filewriter_fn
268 for path, info in images.items():
269 if path.startswith("cid:"):
272 data = request.urlopen(path)
274 mimetype = data.headers["Content-Type"]
275 ext = mimetypes.guess_extension(mimetype)
276 tempfilename = tempfile.mkstemp(prefix="img", suffix=ext, dir=tempdir)
277 path = pathlib.Path(tempfilename[1])
279 filewriter_fn(path, data.read(), "w+b")
283 *mimetype.split("/"),
286 desc=f"Image: {info.desc}",
293 def convert_markdown_to_html(
298 filewriter_fn=filewriter_fn,
301 extension_configs=None,
303 # TODO extension_configs need to be handled differently
304 extension_configs = extension_configs or {}
305 extension_configs.setdefault("pymdownx.highlight", {})
306 extension_configs["pymdownx.highlight"]["css_class"] = _CODEHILITE_CLASS
308 origtext, htmltext, images = markdown_with_inline_image_support(
309 origtext, extensions=extensions, extension_configs=extension_configs
312 filewriter_fn(draftpath, origtext, encoding="utf-8")
314 "text", "plain", draftpath, "Plain-text version", orig=True
317 htmltext = apply_styling(htmltext, cssfile)
319 htmlpath = draftpath.with_suffix(".html")
321 htmlpath, htmltext, encoding="utf-8", errors="xmlcharrefreplace"
323 htmlpart = Part("text", "html", htmlpath, "HTML version")
326 "alternative", [textpart, htmlpart], "Group of alternative content"
329 imgparts = collect_inline_images(
330 images, tempdir=tempdir, filewriter_fn=filewriter_fn
334 "relative", [altpart] + imgparts, "Group of related content"
340 class MIMETreeDFWalker:
341 def __init__(self, *, visitor_fn=None, debug=False):
342 self._visitor_fn = visitor_fn or self._echovisit
345 def _echovisit(self, node, ancestry, debugprint):
346 debugprint(f"node={node} ancestry={ancestry}")
348 def walk(self, root, *, visitor_fn=None):
350 Recursive function to implement a depth-dirst walk of the MIME-tree
354 if isinstance(root, list):
355 root = Multipart("mixed", children=root)
360 visitor_fn=visitor_fn or self._visitor_fn,
363 def _walk(self, node, *, stack, visitor_fn):
364 # Let's start by enumerating the parts at the current level. At the
365 # root level, stack will be the empty list, and we expect a multipart/*
366 # container at this level. Later, e.g. within a mutlipart/alternative
367 # container, the subtree will just be the alternative parts, while the
368 # top of the stack will be the multipart/alternative container, which
369 # we will process after the following loop.
371 lead = f"{'| '*len(stack)}|-"
372 if isinstance(node, Multipart):
374 f"{lead}{node} parents={[s.subtype for s in stack]}"
377 # Depth-first, so push the current container onto the stack,
380 self.debugprint("| " * (len(stack) + 1))
381 for child in node.children:
385 visitor_fn=visitor_fn,
387 self.debugprint("| " * len(stack))
388 assert stack.pop() == node
391 self.debugprint(f"{lead}{node}")
394 visitor_fn(node, stack, debugprint=self.debugprint)
396 def debugprint(self, s, **kwargs):
398 print(s, file=sys.stderr, **kwargs)
401 # [ RUN MODES ] ###############################################################
406 Stupid class to interface writing out Mutt commands. This is quite a hack
407 to deal with the fact that Mutt runs "push" commands in reverse order, so
408 all of a sudden, things become very complicated when mixing with "real"
411 Hence we keep two sets of commands, and one set of pushes. Commands are
412 added to the first until a push is added, after which commands are added to
413 the second set of commands.
415 On flush(), the first set is printed, followed by the pushes in reverse,
416 and then the second set is printed. All 3 sets are then cleared.
419 def __init__(self, out_f=sys.stdout, *, debug=False):
420 self._cmd1, self._push, self._cmd2 = [], [], []
432 s = s.replace('"', '"')
435 self._push.insert(0, s)
439 "\n".join(self._cmd1 + self._push + self._cmd2), file=self._out_f
441 self._cmd1, self._push, self._cmd2 = [], [], []
443 def debugprint(self, s, **kwargs):
445 print(s, file=sys.stderr, **kwargs)
453 debug_commands=False,
455 temppath = temppath or pathlib.Path(
456 tempfile.mkstemp(prefix="muttmdwn-", dir=tempdir)[1]
458 cmds = MuttCommands(out_f, debug=debug_commands)
460 editor = f"{' '.join(sys.argv)} massage --write-commands-to {temppath}"
462 cmds.cmd('set my_editor="$editor"')
463 cmds.cmd('set my_edit_headers="$edit_headers"')
464 cmds.cmd(f'set editor="{editor}"')
465 cmds.cmd("unset edit_headers")
466 cmds.cmd(f"set my_mdwn_postprocess_cmd_file={temppath}")
467 cmds.push("<first-entry><edit-file>")
478 converter=convert_markdown_to_html,
481 debug_commands=False,
484 # Here's the big picture: we're being invoked as the editor on the email
485 # draft, and whatever commands we write to the file given as cmdpath will
486 # be run by the second source command in the macro definition.
488 # Let's start by cleaning up what the setup did (see above), i.e. we
489 # restore the $editor and $edit_headers variables, and also unset the
490 # variable used to identify the command file we're currently writing
492 cmds = MuttCommands(cmd_f, debug=debug_commands)
493 cmds.cmd('set editor="$my_editor"')
494 cmds.cmd('set edit_headers="$my_edit_headers"')
495 cmds.cmd("unset my_editor")
496 cmds.cmd("unset my_edit_headers")
498 # let's flush those commands, as there'll be a lot of pushes from now
499 # on, which need to be run in reverse order
502 extensions = extensions.split(",") if extensions else []
508 extensions=extensions,
511 mimetree = MIMETreeDFWalker(debug=debug_walk)
513 def visitor_fn(item, stack, *, debugprint=None):
515 Visitor function called for every node (part) of the MIME tree,
516 depth-first, and responsible for telling NeoMutt how to assemble
519 KILL_LINE = r"\Ca\Ck"
521 if isinstance(item, Part):
522 # We've hit a leaf-node, i.e. an alternative or a related part
523 # with actual content.
527 # The original source already exists in the NeoMutt tree, but
528 # the underlying file may have been modified, so we need to
529 # update the encoding, but that's it:
530 cmds.push("<update-encoding>")
532 # … whereas all other parts need to be added, and they're all
533 # considered to be temporary and inline:
534 cmds.push(f"<attach-file>{item.path}<enter>")
535 cmds.push("<toggle-unlink><toggle-disposition>")
537 # If the item (including the original) comes with additional
538 # information, then we might just as well update the NeoMutt
541 cmds.push(f"<edit-content-id>{KILL_LINE}{item.cid}<enter>")
543 elif isinstance(item, Multipart):
544 # This node has children, but we already visited them (see
545 # above), and so they have been tagged in NeoMutt's compose
546 # window. Now it's just a matter of telling NeoMutt to do the
547 # appropriate grouping:
548 if item.subtype == "alternative":
549 cmds.push("<group-alternatives>")
550 elif item.subtype in ("relative", "related"):
551 cmds.push("<group-related>")
552 elif item.subtype == "multilingual":
553 cmds.push("<group-multilingual>")
556 # We should never get here
557 assert not "is valid part"
559 # If the item has a description, we might just as well add it
561 cmds.push(f"<edit-description>{KILL_LINE}{item.desc}<enter>")
563 # Finally, if we're at non-root level, tag the new container,
564 # as it might itself be part of a container, to be processed
567 cmds.push("<tag-entry>")
572 # Let's walk the tree and visit every node with our fancy visitor
574 mimetree.walk(tree, visitor_fn=visitor_fn)
577 cmds.push("<send-message>")
579 # Finally, cleanup. Since we're responsible for removing the temporary
580 # file, how's this for a little hack?
582 filename = cmd_f.name
583 except AttributeError:
584 filename = "pytest_internal_file"
585 cmds.cmd(f"source 'rm -f {filename}|'")
586 cmds.cmd("unset my_mdwn_postprocess_cmd_file")
590 # [ CLI ENTRY ] ###############################################################
592 if __name__ == "__main__":
593 args = parse_cli_args()
595 if args.mode is None:
597 tempdir=args.tempdir,
598 debug_commands=args.debug_commands,
601 elif args.mode == "massage":
602 with open(args.MAILDRAFT, "r") as draft_f, open(
607 pathlib.Path(args.MAILDRAFT),
609 extensions=args.extensions,
610 cssfile=args.css_file,
611 only_build=args.only_build,
612 tempdir=args.tempdir,
613 debug_commands=args.debug_commands,
614 debug_walk=args.debug_walk,
618 # [ TESTS ] ###################################################################
622 from io import StringIO
627 return "CONSTANT STRING 1"
631 return "CONSTANT STRING 2"
633 # NOTE: tests using the capsys fixture must specify sys.stdout to the
634 # functions they call, else old stdout is used and not captured
636 def test_MuttCommands_cmd(self, const1, const2, capsys):
637 "Assert order of commands"
638 cmds = MuttCommands(out_f=sys.stdout)
642 captured = capsys.readouterr()
643 assert captured.out == "\n".join((const1, const2, ""))
645 def test_MuttCommands_push(self, const1, const2, capsys):
646 "Assert reverse order of pushes"
647 cmds = MuttCommands(out_f=sys.stdout)
651 captured = capsys.readouterr()
654 == ('"\npush "'.join(("", const2, const1, "")))[2:-6]
657 def test_MuttCommands_cmd_push_mixed(self, const1, const2, capsys):
658 "Assert reverse order of pushes"
659 cmds = MuttCommands(out_f=sys.stdout)
660 lines = ["000", "001", "010", "011", "100", "101", "110", "111"]
662 cmds.cmd(lines[4 * i + 0])
663 cmds.cmd(lines[4 * i + 1])
664 cmds.push(lines[4 * i + 2])
665 cmds.push(lines[4 * i + 3])
668 captured = capsys.readouterr()
669 lines_out = captured.out.splitlines()
670 assert lines[0] in lines_out[0]
671 assert lines[1] in lines_out[1]
672 assert lines[7] in lines_out[2]
673 assert lines[6] in lines_out[3]
674 assert lines[3] in lines_out[4]
675 assert lines[2] in lines_out[5]
676 assert lines[4] in lines_out[6]
677 assert lines[5] in lines_out[7]
680 def basic_mime_tree(self):
694 Part("text", "html", "part.html", desc="HTML"),
699 "text", "png", "logo.png", cid="logo.png", desc="Logo"
705 def test_MIMETreeDFWalker_depth_first_walk(self, basic_mime_tree):
706 mimetree = MIMETreeDFWalker()
710 def visitor_fn(item, stack, debugprint):
711 items.append((item, len(stack)))
713 mimetree.walk(basic_mime_tree, visitor_fn=visitor_fn)
714 assert len(items) == 5
715 assert items[0][0].subtype == "plain"
716 assert items[0][1] == 2
717 assert items[1][0].subtype == "html"
718 assert items[1][1] == 2
719 assert items[2][0].subtype == "alternative"
720 assert items[2][1] == 1
721 assert items[3][0].subtype == "png"
722 assert items[3][1] == 1
723 assert items[4][0].subtype == "relative"
724 assert items[4][1] == 0
726 def test_MIMETreeDFWalker_list_to_mixed(self, basic_mime_tree):
727 mimetree = MIMETreeDFWalker()
730 def visitor_fn(item, stack, debugprint):
733 mimetree.walk([basic_mime_tree], visitor_fn=visitor_fn)
734 assert items[-1].subtype == "mixed"
736 def test_MIMETreeDFWalker_visitor_in_constructor(
737 self, basic_mime_tree
741 def visitor_fn(item, stack, debugprint):
744 mimetree = MIMETreeDFWalker(visitor_fn=visitor_fn)
745 mimetree.walk(basic_mime_tree)
746 assert len(items) == 5
749 def string_io(self, const1, text=None):
750 return StringIO(text or const1)
752 def test_do_massage_basic(self, const1, string_io, capsys):
753 def converter(drafttext, draftpath, cssfile, extensions, tempdir):
754 return Part("text", "plain", draftpath, orig=True)
763 captured = capsys.readouterr()
764 lines = captured.out.splitlines()
765 assert '="$my_editor"' in lines.pop(0)
766 assert '="$my_edit_headers"' in lines.pop(0)
767 assert "unset my_editor" == lines.pop(0)
768 assert "unset my_edit_headers" == lines.pop(0)
769 assert "send-message" in lines.pop(0)
770 assert "update-encoding" in lines.pop(0)
771 assert "source 'rm -f " in lines.pop(0)
772 assert "unset my_mdwn_postprocess_cmd_file" == lines.pop(0)
774 def test_do_massage_fulltree(
775 self, string_io, const1, basic_mime_tree, capsys
777 def converter(drafttext, draftpath, cssfile, extensions, tempdir):
778 return basic_mime_tree
787 captured = capsys.readouterr()
788 lines = captured.out.splitlines()[4:]
789 assert "send-message" in lines.pop(0)
790 assert "Related" in lines.pop(0)
791 assert "group-related" in lines.pop(0)
792 assert "tag-entry" in lines.pop(0)
793 assert "Logo" in lines.pop(0)
794 assert "content-id" in lines.pop(0)
795 assert "toggle-unlink" in lines.pop(0)
796 assert "logo.png" in lines.pop(0)
797 assert "tag-entry" in lines.pop(0)
798 assert "Alternative" in lines.pop(0)
799 assert "group-alternatives" in lines.pop(0)
800 assert "tag-entry" in lines.pop(0)
801 assert "HTML" in lines.pop(0)
802 assert "toggle-unlink" in lines.pop(0)
803 assert "part.html" in lines.pop(0)
804 assert "tag-entry" in lines.pop(0)
805 assert "Plain" in lines.pop(0)
806 assert "update-encoding" in lines.pop(0)
807 assert len(lines) == 2
810 def fake_filewriter(self):
815 def __call__(self, path, content, mode="w", **kwargs):
816 self._writes.append((path, content))
818 def pop(self, index=-1):
819 return self._writes.pop(index)
824 def markdown_non_converter(self, const1, const2):
825 return lambda s, text: f"{const1}{text}{const2}"
827 def test_converter_tree_basic(
828 self, const1, const2, fake_filewriter, markdown_non_converter
830 path = pathlib.Path(const2)
831 tree = convert_markdown_to_html(
832 const1, path, filewriter_fn=fake_filewriter
835 assert tree.subtype == "alternative"
836 assert len(tree.children) == 2
837 assert tree.children[0].subtype == "plain"
838 assert tree.children[0].path == path
839 assert tree.children[0].orig
840 assert tree.children[1].subtype == "html"
841 assert tree.children[1].path == path.with_suffix(".html")
843 def test_converter_writes(
849 markdown_non_converter,
851 path = pathlib.Path(const2)
853 with monkeypatch.context() as m:
854 m.setattr(markdown.Markdown, "convert", markdown_non_converter)
855 convert_markdown_to_html(
856 const1, path, filewriter_fn=fake_filewriter
859 assert (path, const1) == fake_filewriter.pop(0)
861 path.with_suffix(".html"),
862 markdown_non_converter(None, const1),
863 ) == fake_filewriter.pop(0)
865 def test_markdown_inline_image_processor(self):
866 imgpath1 = "file:/path/to/image.png"
867 imgpath2 = "file:///path/to/image.png?url=params"
868 imgpath3 = "/path/to/image.png"
869 text = f"""![inline local image]({imgpath1})
871 with newline]({imgpath2})
872 ![image local path]({imgpath3})"""
873 text, html, images = markdown_with_inline_image_support(text)
875 # local paths have been normalised to URLs:
876 imgpath3 = f"file://{imgpath3}"
878 assert 'src="cid:' in html
879 assert "](cid:" in text
880 assert len(images) == 3
881 assert imgpath1 in images
882 assert imgpath2 in images
883 assert imgpath3 in images
884 assert images[imgpath1].cid != images[imgpath2].cid
885 assert images[imgpath1].cid != images[imgpath3].cid
886 assert images[imgpath2].cid != images[imgpath3].cid
888 def test_markdown_inline_image_processor_title_to_desc(self, const1):
889 imgpath = "file:///path/to/image.png"
890 text = f'![inline local image]({imgpath} "{const1}")'
891 text, html, images = markdown_with_inline_image_support(text)
892 assert images[imgpath].desc == const1
894 def test_markdown_inline_image_processor_alt_to_desc(self, const1):
895 imgpath = "file:///path/to/image.png"
896 text = f"![{const1}]({imgpath})"
897 text, html, images = markdown_with_inline_image_support(text)
898 assert images[imgpath].desc == const1
900 def test_markdown_inline_image_processor_title_over_alt_desc(
903 imgpath = "file:///path/to/image.png"
904 text = f'![{const1}]({imgpath} "{const2}")'
905 text, html, images = markdown_with_inline_image_support(text)
906 assert images[imgpath].desc == const2
908 def test_markdown_inline_image_not_external(self):
909 imgpath = "https://path/to/image.png"
910 text = f"![inline image]({imgpath})"
911 text, html, images = markdown_with_inline_image_support(text)
913 assert 'src="cid:' not in html
914 assert "](cid:" not in text
915 assert len(images) == 0
917 def test_markdown_inline_image_local_file(self):
918 imgpath = "/path/to/image.png"
919 text = f"![inline image]({imgpath})"
920 text, html, images = markdown_with_inline_image_support(text)
922 for k, v in images.items():
923 assert k == f"file://{imgpath}"
929 ""
930 "AAAABCAAAAAA6fptVAAAACklEQVQI12P4DwABAQEAG7buVgAA"
933 def test_markdown_inline_image_processor_base64(self, test_png):
934 text = f"![1px white inlined]({test_png})"
935 text, html, images = markdown_with_inline_image_support(text)
937 assert 'src="cid:' in html
938 assert "](cid:" in text
939 assert len(images) == 1
940 assert test_png in images
942 def test_converter_tree_inline_image_base64(
943 self, test_png, const1, fake_filewriter
945 text = f"![inline base64 image]({test_png})"
946 path = pathlib.Path(const1)
947 tree = convert_markdown_to_html(
948 text, path, filewriter_fn=fake_filewriter
951 assert tree.subtype == "relative"
952 assert tree.children[1].subtype == "png"
953 written = fake_filewriter.pop()
954 assert tree.children[1].path == written[0]
955 assert written[1] == request.urlopen(test_png).read()
957 def test_converter_tree_inline_image_cid(
958 self, const1, fake_filewriter
960 text = f"![inline base64 image](cid:{const1})"
961 path = pathlib.Path(const1)
962 tree = convert_markdown_to_html(
965 filewriter_fn=fake_filewriter,
966 related_to_html_only=False,
968 assert len(tree.children) == 2
969 assert tree.children[0].cid != const1
970 assert tree.children[0].type != "image"
971 assert tree.children[1].cid != const1
972 assert tree.children[1].type != "image"
974 def test_inline_image_collection(
975 self, test_png, const1, const2, fake_filewriter
977 test_images = {test_png: InlineImageInfo(cid=const1, desc=const2)}
978 relparts = collect_inline_images(
979 test_images, filewriter_fn=fake_filewriter
982 written = fake_filewriter.pop()
983 assert b"PNG" in written[1]
985 assert relparts[0].subtype == "png"
986 assert relparts[0].path == written[0]
987 assert relparts[0].cid == const1
988 assert relparts[0].desc.endswith(const2)
990 def test_apply_stylesheet(self):
992 html = "<p>Hello, world!</p>"
993 css = "p { color:red }"
994 out = apply_styling(html, css)
995 assert 'p style="color' in out
997 def test_apply_stylesheet_pygments(self):
1000 f'<div class="{_CODEHILITE_CLASS}">'
1001 "<pre>def foo():\n return</pre></div>"
1003 out = apply_styling(html, _PYGMENTS_CSS)
1004 assert f'{_CODEHILITE_CLASS}" style="' in out