mirror of
				https://github.com/livebook-dev/livebook.git
				synced 2025-10-25 12:56:13 +08:00 
			
		
		
		
	Keep inline Markdown unchanged when importing/exporting (#487)
* Merge: [WIP] Make Markdown formatter math-safe #447 * Keep inline Markdown unchanged when importing/exporting
This commit is contained in:
		
							parent
							
								
									0ac475d915
								
							
						
					
					
						commit
						f22bf2a21d
					
				
					 8 changed files with 169 additions and 34 deletions
				
			
		|  | @ -79,7 +79,7 @@ defmodule Livebook.LiveMarkdown.Export do | |||
|   end | ||||
| 
 | ||||
|   defp render_cell(%Cell.Elixir{} = cell, ctx) do | ||||
|     delimiter = code_block_delimiter(cell.source) | ||||
|     delimiter = MarkdownHelpers.code_block_delimiter(cell.source) | ||||
|     code = get_elixir_cell_code(cell) | ||||
|     outputs = if ctx.include_outputs?, do: render_outputs(cell), else: [] | ||||
| 
 | ||||
|  | @ -131,13 +131,13 @@ defmodule Livebook.LiveMarkdown.Export do | |||
| 
 | ||||
|   defp render_output(text) when is_binary(text) do | ||||
|     text = String.replace_suffix(text, "\n", "") | ||||
|     delimiter = code_block_delimiter(text) | ||||
|     delimiter = MarkdownHelpers.code_block_delimiter(text) | ||||
|     text = strip_ansi(text) | ||||
|     [delimiter, "output\n", text, "\n", delimiter] | ||||
|   end | ||||
| 
 | ||||
|   defp render_output({:text, text}) do | ||||
|     delimiter = code_block_delimiter(text) | ||||
|     delimiter = MarkdownHelpers.code_block_delimiter(text) | ||||
|     text = strip_ansi(text) | ||||
|     [delimiter, "output\n", text, "\n", delimiter] | ||||
|   end | ||||
|  | @ -163,7 +163,7 @@ defmodule Livebook.LiveMarkdown.Export do | |||
| 
 | ||||
|   defp format_markdown_source(markdown) do | ||||
|     markdown | ||||
|     |> EarmarkParser.as_ast() | ||||
|     |> MarkdownHelpers.markdown_to_block_ast() | ||||
|     |> elem(1) | ||||
|     |> rewrite_ast() | ||||
|     |> MarkdownHelpers.markdown_from_ast() | ||||
|  | @ -202,15 +202,6 @@ defmodule Livebook.LiveMarkdown.Export do | |||
|     end | ||||
|   end | ||||
| 
 | ||||
|   defp code_block_delimiter(code) do | ||||
|     max_streak = | ||||
|       Regex.scan(~r/`{3,}/, code) | ||||
|       |> Enum.map(fn [string] -> byte_size(string) end) | ||||
|       |> Enum.max(&>=/2, fn -> 2 end) | ||||
| 
 | ||||
|     String.duplicate("`", max_streak + 1) | ||||
|   end | ||||
| 
 | ||||
|   defp put_unless_implicit(map, entries) do | ||||
|     Enum.reduce(entries, map, fn {key, value}, map -> | ||||
|       if value in [false, %{}] do | ||||
|  |  | |||
|  | @ -10,7 +10,7 @@ defmodule Livebook.LiveMarkdown.Import do | |||
|   """ | ||||
|   @spec notebook_from_markdown(String.t()) :: {Notebook.t(), list(String.t())} | ||||
|   def notebook_from_markdown(markdown) do | ||||
|     {_, ast, earmark_messages} = EarmarkParser.as_ast(markdown) | ||||
|     {_, ast, earmark_messages} = MarkdownHelpers.markdown_to_block_ast(markdown) | ||||
|     earmark_messages = Enum.map(earmark_messages, &earmark_message_to_string/1) | ||||
| 
 | ||||
|     {ast, rewrite_messages} = rewrite_ast(ast) | ||||
|  | @ -221,7 +221,7 @@ defmodule Livebook.LiveMarkdown.Import do | |||
|   end | ||||
| 
 | ||||
|   defp build_notebook([{:section_name, content} | elems], cells, sections) do | ||||
|     name = MarkdownHelpers.text_from_ast(content) | ||||
|     name = text_from_markdown(content) | ||||
|     {metadata, elems} = grab_metadata(elems) | ||||
|     attrs = section_metadata_to_attrs(metadata) | ||||
|     section = %{Notebook.Section.new() | name: name, cells: cells} |> Map.merge(attrs) | ||||
|  | @ -242,7 +242,7 @@ defmodule Livebook.LiveMarkdown.Import do | |||
|   end | ||||
| 
 | ||||
|   defp build_notebook([{:notebook_name, content} | elems], [], sections) do | ||||
|     name = MarkdownHelpers.text_from_ast(content) | ||||
|     name = text_from_markdown(content) | ||||
|     {metadata, []} = grab_metadata(elems) | ||||
|     attrs = notebook_metadata_to_attrs(metadata) | ||||
|     %{Notebook.new() | name: name, sections: sections} |> Map.merge(attrs) | ||||
|  | @ -253,6 +253,13 @@ defmodule Livebook.LiveMarkdown.Import do | |||
|     %{Notebook.new() | sections: sections} | ||||
|   end | ||||
| 
 | ||||
|   defp text_from_markdown(markdown) do | ||||
|     markdown | ||||
|     |> MarkdownHelpers.markdown_to_ast() | ||||
|     |> elem(1) | ||||
|     |> MarkdownHelpers.text_from_ast() | ||||
|   end | ||||
| 
 | ||||
|   # Takes optional leading metadata JSON object and returns {metadata, rest}. | ||||
|   defp grab_metadata([{:metadata, metadata} | elems]) do | ||||
|     {metadata, elems} | ||||
|  |  | |||
|  | @ -1,13 +1,32 @@ | |||
| defmodule Livebook.LiveMarkdown.MarkdownHelpers do | ||||
|   @moduledoc false | ||||
| 
 | ||||
|   @doc """ | ||||
|   Wraps `EarmarkParser.as_ast/2`. | ||||
|   """ | ||||
|   @spec markdown_to_ast(String.t()) :: {:ok | :error, EarmarkParser.ast(), list()} | ||||
|   def markdown_to_ast(markdown) do | ||||
|     EarmarkParser.as_ast(markdown) | ||||
|   end | ||||
| 
 | ||||
|   @doc """ | ||||
|   Wraps `EarmarkParser.as_ast/2`. | ||||
| 
 | ||||
|   Markdown blocks are parsed into the AST, while inline | ||||
|   content is kept as is. | ||||
|   """ | ||||
|   @spec markdown_to_block_ast(String.t()) :: {:ok | :error, EarmarkParser.ast(), list()} | ||||
|   def markdown_to_block_ast(markdown) do | ||||
|     EarmarkParser.as_ast(markdown, parse_inline: false) | ||||
|   end | ||||
| 
 | ||||
|   @doc """ | ||||
|   Reformats the given markdown document. | ||||
|   """ | ||||
|   @spec reformat(String.t()) :: String.t() | ||||
|   def reformat(markdown) do | ||||
|     markdown | ||||
|     |> EarmarkParser.as_ast() | ||||
|     |> markdown_to_block_ast() | ||||
|     |> elem(1) | ||||
|     |> markdown_from_ast() | ||||
|   end | ||||
|  | @ -27,6 +46,20 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpers do | |||
|   def text_from_ast(ast) when is_binary(ast), do: ast | ||||
|   def text_from_ast({_, _, ast, _}), do: text_from_ast(ast) | ||||
| 
 | ||||
|   @doc """ | ||||
|   Determines suitable Markdown fence delimiter for the | ||||
|   given code. | ||||
|   """ | ||||
|   @spec code_block_delimiter(String.t()) :: String.t() | ||||
|   def code_block_delimiter(code) do | ||||
|     max_streak = | ||||
|       Regex.scan(~r/`{3,}/, code) | ||||
|       |> Enum.map(fn [string] -> byte_size(string) end) | ||||
|       |> Enum.max(&>=/2, fn -> 2 end) | ||||
| 
 | ||||
|     String.duplicate("`", max_streak + 1) | ||||
|   end | ||||
| 
 | ||||
|   @doc """ | ||||
|   Renders Markdown string from the given `EarmarkParser` AST. | ||||
|   """ | ||||
|  | @ -161,12 +194,12 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpers do | |||
|   @void_elements ~W(area base br col command embed hr img input keygen link meta param source track wbr) | ||||
| 
 | ||||
|   defp render_html(tag, attrs, []) when tag in @void_elements do | ||||
|     ["<", tag, " ", attrs_to_string(attrs), " />"] | ||||
|     ["<", tag, attrs_to_string(attrs), " />"] | ||||
|   end | ||||
| 
 | ||||
|   defp render_html(tag, attrs, lines) do | ||||
|     inner = Enum.intersperse(lines, "\n") | ||||
|     ["<", tag, " ", attrs_to_string(attrs), ">\n", inner, "\n</", tag, ">"] | ||||
|     ["<", tag, attrs_to_string(attrs), ">\n", inner, "\n</", tag, ">"] | ||||
|   end | ||||
| 
 | ||||
|   defp render_emphasis(content) do | ||||
|  | @ -243,8 +276,9 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpers do | |||
|   end | ||||
| 
 | ||||
|   defp render_code_block(content, attrs) do | ||||
|     delimiter = code_block_delimiter(content) | ||||
|     language = get_attr(attrs, "class", "") | ||||
|     ["```", language, "\n", content, "\n```"] | ||||
|     [delimiter, language, "\n", content, "\n", delimiter] | ||||
|   end | ||||
| 
 | ||||
|   defp render_blockquote(content) do | ||||
|  | @ -376,9 +410,7 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpers do | |||
|   end | ||||
| 
 | ||||
|   defp attrs_to_string(attrs) do | ||||
|     attrs | ||||
|     |> Enum.map(fn {key, value} -> ~s/#{key}="#{value}"/ end) | ||||
|     |> Enum.join(" ") | ||||
|     Enum.map(attrs, fn {key, value} -> ~s/ #{key}="#{value}"/ end) | ||||
|   end | ||||
| 
 | ||||
|   defp blank?(string), do: String.trim(string) == "" | ||||
|  |  | |||
							
								
								
									
										4
									
								
								mix.exs
									
										
									
									
									
								
							
							
						
						
									
										4
									
								
								mix.exs
									
										
									
									
									
								
							|  | @ -47,7 +47,9 @@ defmodule Livebook.MixProject do | |||
|       {:telemetry_poller, "~> 0.4"}, | ||||
|       {:jason, "~> 1.0"}, | ||||
|       {:plug_cowboy, "~> 2.0"}, | ||||
|       {:earmark_parser, "~> 1.4"}, | ||||
|       # {:earmark_parser, "~> 1.4"}, | ||||
|       {:earmark_parser, "~> 1.4", | ||||
|        github: "jonatanklosko/earmark_parser", branch: "jk-optional-inline"}, | ||||
|       {:bypass, "~> 2.1", only: :test}, | ||||
|       {:castore, "~> 0.1.0"} | ||||
|     ] | ||||
|  |  | |||
							
								
								
									
										2
									
								
								mix.lock
									
										
									
									
									
								
							
							
						
						
									
										2
									
								
								mix.lock
									
										
									
									
									
								
							|  | @ -4,7 +4,7 @@ | |||
|   "cowboy": {:hex, :cowboy, "2.9.0", "865dd8b6607e14cf03282e10e934023a1bd8be6f6bacf921a7e2a96d800cd452", [:make, :rebar3], [{:cowlib, "2.11.0", [hex: :cowlib, repo: "hexpm", optional: false]}, {:ranch, "1.8.0", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "2c729f934b4e1aa149aff882f57c6372c15399a20d54f65c8d67bef583021bde"}, | ||||
|   "cowboy_telemetry": {:hex, :cowboy_telemetry, "0.3.1", "ebd1a1d7aff97f27c66654e78ece187abdc646992714164380d8a041eda16754", [:rebar3], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "3a6efd3366130eab84ca372cbd4a7d3c3a97bdfcfb4911233b035d117063f0af"}, | ||||
|   "cowlib": {:hex, :cowlib, "2.11.0", "0b9ff9c346629256c42ebe1eeb769a83c6cb771a6ee5960bd110ab0b9b872063", [:make, :rebar3], [], "hexpm", "2b3e9da0b21c4565751a6d4901c20d1b4cc25cbb7fd50d91d2ab6dd287bc86a9"}, | ||||
|   "earmark_parser": {:hex, :earmark_parser, "1.4.13", "0c98163e7d04a15feb62000e1a891489feb29f3d10cb57d4f845c405852bbef8", [:mix], [], "hexpm", "d602c26af3a0af43d2f2645613f65841657ad6efc9f0e361c3b6c06b578214ba"}, | ||||
|   "earmark_parser": {:git, "https://github.com/jonatanklosko/earmark_parser.git", "288ed586d45825a28b824d1cfd42fcd0959cd628", [branch: "jk-optional-inline"]}, | ||||
|   "file_system": {:hex, :file_system, "0.2.10", "fb082005a9cd1711c05b5248710f8826b02d7d1784e7c3451f9c1231d4fc162d", [:mix], [], "hexpm", "41195edbfb562a593726eda3b3e8b103a309b733ad25f3d642ba49696bf715dc"}, | ||||
|   "floki": {:hex, :floki, "0.31.0", "f05ee8a8e6a3ced4e62beeb2c79a63bc8e12ab98fbaaf6e6a3d9b76b1278e23f", [:mix], [{:html_entities, "~> 0.5.0", [hex: :html_entities, repo: "hexpm", optional: false]}], "hexpm", "b05afa372f5c345a5bf240ac25ea1f0f3d5fcfd7490ac0beeb4a203f9444891e"}, | ||||
|   "html_entities": {:hex, :html_entities, "0.5.2", "9e47e70598da7de2a9ff6af8758399251db6dbb7eebe2b013f2bbd2515895c3c", [:mix], [], "hexpm", "c53ba390403485615623b9531e97696f076ed415e8d8058b1dbaa28181f4fdcc"}, | ||||
|  |  | |||
|  | @ -20,7 +20,9 @@ defmodule Livebook.LiveMarkdown.ExportTest do | |||
| 
 | ||||
|                     * Erlang | ||||
|                     * Elixir | ||||
|                     * PostgreSQL\ | ||||
|                     * PostgreSQL | ||||
| 
 | ||||
|                     $x_{i} + y_{i}$\ | ||||
|                     """ | ||||
|                 }, | ||||
|                 %{ | ||||
|  | @ -92,6 +94,8 @@ defmodule Livebook.LiveMarkdown.ExportTest do | |||
|     * Elixir | ||||
|     * PostgreSQL | ||||
| 
 | ||||
|     $x_{i} + y_{i}$ | ||||
| 
 | ||||
|     <!-- livebook:{"disable_formatting":true} --> | ||||
| 
 | ||||
|     ```elixir | ||||
|  |  | |||
|  | @ -17,6 +17,8 @@ defmodule Livebook.LiveMarkdown.ImportTest do | |||
|     * Elixir | ||||
|     * PostgreSQL | ||||
| 
 | ||||
|     $x_{i} + y_{i}$ | ||||
| 
 | ||||
|     <!-- livebook:{"disable_formatting": true} --> | ||||
| 
 | ||||
|     ```elixir | ||||
|  | @ -60,7 +62,9 @@ defmodule Livebook.LiveMarkdown.ImportTest do | |||
| 
 | ||||
|                      * Erlang | ||||
|                      * Elixir | ||||
|                      * PostgreSQL\ | ||||
|                      * PostgreSQL | ||||
| 
 | ||||
|                      $x_{i} + y_{i}$\ | ||||
|                      """ | ||||
|                    }, | ||||
|                    %Cell.Elixir{ | ||||
|  | @ -120,9 +124,6 @@ defmodule Livebook.LiveMarkdown.ImportTest do | |||
| 
 | ||||
|     ## Section 1 | ||||
| 
 | ||||
|     Line 1.\s\s | ||||
|     Line 2. | ||||
| 
 | ||||
|     |State|Abbrev|Capital| | ||||
|     | --: | :-: | --- | | ||||
|     | Texas | TX | Austin | | ||||
|  | @ -139,9 +140,6 @@ defmodule Livebook.LiveMarkdown.ImportTest do | |||
|                  cells: [ | ||||
|                    %Cell.Markdown{ | ||||
|                      source: """ | ||||
|                      Line 1.\\ | ||||
|                      Line 2. | ||||
| 
 | ||||
|                      | State | Abbrev | Capital | | ||||
|                      | ----: | :----: | ------- | | ||||
|                      | Texas | TX     | Austin  | | ||||
|  |  | |||
|  | @ -9,6 +9,16 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "other emphasis" do | ||||
|       markdown = "The Game, _Mrs Hudson_, is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "nested emphasis" do | ||||
|       markdown = "The *Game, _Mrs Hudson_, is* on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "bold" do | ||||
|       markdown = "The Game, **Mrs Hudson**, is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|  | @ -24,16 +34,50 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "inline code with a line break" do | ||||
|       markdown = "The Game, `Mrs\nHudson`, is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "inline code with extra spaces" do | ||||
|       markdown = "The Game, `Mrs Huds  on`, is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "combined" do | ||||
|       markdown = "The Game, ~~***`Mrs Hudson`***~~, is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "inline math" do | ||||
|       markdown = "The Game, $x_{i} + y_{i}$, is on!" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "link" do | ||||
|       markdown = "The Game, [Mrs Hudson](https://youtu.be/M-KqaO1oH2E), is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "link with IAL" do | ||||
|       markdown = "[link](url){: .classy}" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "link followed by escaped IAL" do | ||||
|       markdown = "[link](url)\\{: .classy}" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "autolink" do | ||||
|       markdown = "<https://elixir-lang.com>" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "basic image" do | ||||
|       markdown = "The Game, , is on!" | ||||
|       assert markdown == reformat(markdown) | ||||
|  | @ -49,6 +93,11 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "line break" do | ||||
|       markdown = "Line 1.\\\nLine 2." | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "oneline comment" do | ||||
|       markdown = "<!-- The Game, Mrs Hudson, is on! -->" | ||||
|       assert markdown == reformat(markdown) | ||||
|  | @ -108,6 +157,32 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "code block with fences inside it" do | ||||
|       markdown = """ | ||||
|       ````elixir | ||||
|       before | ||||
| 
 | ||||
|       ``` | ||||
|       _inside_ | ||||
|       ``` | ||||
| 
 | ||||
|       after | ||||
|       ````\ | ||||
|       """ | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "display math" do | ||||
|       markdown = """ | ||||
|       $$ | ||||
|       R_{ij}^{kl} = R_{ij} - \Gamma^k_{kl} | ||||
|       $$\ | ||||
|       """ | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "blockquote" do | ||||
|       markdown = """ | ||||
|       > The Game, Mrs Hudson, | ||||
|  | @ -243,6 +318,18 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "surprise ordered list" do | ||||
|       markdown = "1986\\. What a great season." | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "escaped Markdown" do | ||||
|       markdown = "not a \\[link\\]()" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "raw html" do | ||||
|       markdown = """ | ||||
|       <div class="box" aria-label="box"> | ||||
|  | @ -253,6 +340,18 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "raw html at the beginning of a line" do | ||||
|       markdown = "line\n\n<hr />" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "raw html not at the beginning of a line" do | ||||
|       markdown = "line\n  <hr />" | ||||
| 
 | ||||
|       assert markdown == reformat(markdown) | ||||
|     end | ||||
| 
 | ||||
|     test "separates blocks with a single line" do | ||||
|       markdown = """ | ||||
|       The first paragraph, | ||||
|  | @ -274,7 +373,9 @@ defmodule Livebook.LiveMarkdown.MarkdownHelpersTest do | |||
|     # By reformatting we can assert correct rendering | ||||
|     # by comparing against the original content. | ||||
|     defp reformat(markdown) do | ||||
|       {:ok, ast, []} = EarmarkParser.as_ast(markdown) | ||||
|       # Note: we don't parse inline content, so some of the tests | ||||
|       # above are not stricly necessary, but we keep them for completeness. | ||||
|       {:ok, ast, []} = MarkdownHelpers.markdown_to_block_ast(markdown) | ||||
|       MarkdownHelpers.markdown_from_ast(ast) | ||||
|     end | ||||
|   end | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue