Skip to content

Commit afaf6a7

Browse files
authored
Merge pull request #31 from avo-hq/feature/render-using-commonmarker
feature: use commonmarker gem
2 parents 8989415 + 8c5407c commit afaf6a7

File tree

12 files changed

+220
-31
lines changed

12 files changed

+220
-31
lines changed

Gemfile

+5
Original file line numberDiff line numberDiff line change
@@ -27,3 +27,8 @@ end
2727
# Access an interactive console on exception pages or by calling 'console' anywhere in the code.
2828
gem "listen", ">= 3.5.1"
2929
# gem "avo", ">= 3.2.1"
30+
31+
gem "commonmarker"
32+
gem "redcarpet"
33+
34+
gem "minitest-difftastic", "~> 0.2.1", :group => :test

Gemfile.lock

+27-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ PATH
33
specs:
44
marksmith (0.1.2)
55
activesupport
6-
redcarpet
76

87
GEM
98
remote: https://rubygems.org/
@@ -85,10 +84,26 @@ GEM
8584
bigdecimal (3.1.9)
8685
bindex (0.8.1)
8786
builder (3.3.0)
87+
commonmarker (2.0.4)
88+
rb_sys (~> 0.9)
89+
commonmarker (2.0.4-aarch64-linux)
90+
commonmarker (2.0.4-arm64-darwin)
91+
commonmarker (2.0.4-x86_64-darwin)
92+
commonmarker (2.0.4-x86_64-linux)
8893
concurrent-ruby (1.3.5)
8994
connection_pool (2.5.0)
9095
crass (1.0.6)
9196
date (3.4.1)
97+
difftastic (0.6.0)
98+
pretty_please
99+
difftastic (0.6.0-arm64-darwin)
100+
pretty_please
101+
difftastic (0.6.0-x86_64-darwin)
102+
pretty_please
103+
difftastic (0.6.0-x86_64-linux)
104+
pretty_please
105+
dispersion (0.2.0)
106+
prism
92107
drb (2.2.1)
93108
dry-cli (1.2.0)
94109
erubi (1.13.1)
@@ -126,6 +141,8 @@ GEM
126141
marcel (1.0.4)
127142
mini_mime (1.1.5)
128143
minitest (5.25.4)
144+
minitest-difftastic (0.2.1)
145+
difftastic (~> 0.6)
129146
mutex_m (0.3.0)
130147
net-imap (0.5.5)
131148
date
@@ -159,7 +176,10 @@ GEM
159176
racc
160177
pp (0.6.2)
161178
prettyprint
179+
pretty_please (0.2.0)
180+
dispersion (~> 0.2)
162181
prettyprint (0.2.0)
182+
prism (1.3.0)
163183
propshaft (1.1.0)
164184
actionpack (>= 7.0.0)
165185
activesupport (>= 7.0.0)
@@ -212,9 +232,12 @@ GEM
212232
zeitwerk (~> 2.6)
213233
rainbow (3.1.1)
214234
rake (13.2.1)
235+
rake-compiler-dock (1.9.1)
215236
rb-fsevent (0.11.2)
216237
rb-inotify (0.11.1)
217238
ffi (~> 1.0)
239+
rb_sys (0.9.110)
240+
rake-compiler-dock (= 1.9.1)
218241
rdoc (6.12.0)
219242
psych (>= 4.0.0)
220243
redcarpet (3.6.0)
@@ -305,11 +328,14 @@ PLATFORMS
305328
x86_64-linux-musl
306329

307330
DEPENDENCIES
331+
commonmarker
308332
listen (>= 3.5.1)
309333
marksmith!
334+
minitest-difftastic (~> 0.2.1)
310335
propshaft
311336
puma
312337
rails (>= 7.0.0)
338+
redcarpet
313339
rubocop-rails-omakase
314340
sqlite3
315341
stimulus-rails

README.md

+21-21
Original file line numberDiff line numberDiff line change
@@ -154,8 +154,8 @@ end
154154

155155
## Built-in preview renderer
156156

157-
The renderer is powered by [`Redcarpet`](https://github.com/vmg/redcarpet).
158-
It supports basic styles for headings, `strong`, `italic` and others.
157+
The renderer is powered by [`Commonmarker`](https://github.com/gjtorikian/commonmarker) by default but it can be changed to [`Redcarpet`](https://github.com/vmg/redcarpet) in the configuration or add your own logic by customizing the `Marksmith::Renderer` model.
158+
It supports basic styles like headings, `strong`, `italic` and others.
159159

160160
In your `show.html.erb` view or the place where you want to render the compiled markup use the `marksmithed` helper and it will run the content through the renderer.
161161

@@ -172,31 +172,31 @@ In your `show.html.erb` view or the place where you want to render the compiled
172172
> sanitize(body, tags: %w(table th tr td span) + ActionView::Helpers::SanitizeHelper.sanitizer_vendor.safe_list_sanitizer.allowed_tags.to_a)
173173
> ```
174174
175-
### Customize the renderer
175+
## Customize the renderer
176176
177-
You can customize the renderer by overriding the `Marksmith::Renderer` model.
177+
Marksmith comes with a default renderer that uses `Commonmarker` by default but it can be changed to `Redcarpet` in the configuration.
178178
179179
```ruby
180-
# app/models/marksmith/renderer.rb
181-
require "redcarpet"
180+
# config/initializers/marksmith.rb
181+
Marksmith.configure do |config|
182+
config.parser = "redcarpet"
183+
end
184+
```
185+
186+
### Add your own renderer
182187

188+
You can completely customize the renderer by overriding the `Marksmith::Renderer` model.
189+
190+
```ruby
191+
# app/models/marksmith/renderer.rb
183192
module Marksmith
184193
class Renderer
185-
def renderer
186-
::Redcarpet::Markdown.new(
187-
::Redcarpet::Render::HTML,
188-
tables: true,
189-
lax_spacing: true,
190-
fenced_code_blocks: true,
191-
space_after_headers: true,
192-
hard_wrap: true,
193-
autolink: true,
194-
strikethrough: true,
195-
underline: true,
196-
highlight: true,
197-
quote: true,
198-
with_toc_data: true
199-
)
194+
def initialize(body:)
195+
@body = body
196+
end
197+
198+
def render
199+
# Your custom renderer logic here
200200
end
201201
end
202202
end
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
<%= field_wrapper **field_wrapper_args, full_width: true do %>
2-
<%= render partial: "marksmith/shared/rendered_body", locals: { body: Marksmith::Renderer.new.renderer.render(@field.value) } %>
2+
<%= render partial: "marksmith/shared/rendered_body", locals: { body: Marksmith::Renderer.new(body: @field.value).render } %>
33
<% end %>

app/controllers/marksmith/markdown_previews_controller.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
module Marksmith
22
class MarkdownPreviewsController < ApplicationController
33
def create
4-
@body = Marksmith::Renderer.new.renderer.render(params[:body])
4+
@body = Marksmith::Renderer.new(body: params[:body]).render
55
end
66
end
77
end

app/models/marksmith/renderer.rb

+20-4
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,24 @@
1-
require "redcarpet"
2-
31
module Marksmith
42
class Renderer
5-
def renderer
3+
def initialize(body:)
4+
@body = body
5+
end
6+
7+
def render
8+
if Marksmith.configuration.parser == "commonmarker"
9+
render_commonmarker
10+
else
11+
render_redcarpet
12+
end
13+
end
14+
15+
def render_commonmarker
16+
# commonmarker expects an utf-8 encoded string
17+
body = @body.to_s.dup.force_encoding('utf-8')
18+
Commonmarker.to_html(body)
19+
end
20+
21+
def render_redcarpet
622
::Redcarpet::Markdown.new(
723
::Redcarpet::Render::HTML,
824
tables: true,
@@ -16,7 +32,7 @@ def renderer
1632
highlight: true,
1733
quote: true,
1834
with_toc_data: true
19-
)
35+
).render(@body)
2036
end
2137
end
2238
end

lib/marksmith/configuration.rb

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ class Configuration
44

55
config_accessor(:automatically_mount_engine) { true }
66
config_accessor(:mount_path) { "/marksmith" }
7+
config_accessor(:parser) { "commonmarker" }
78
end
89

910
def self.configuration

lib/marksmith/helper.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
module Marksmith
22
module Helper
33
def marksmithed(body)
4-
Marksmith::Renderer.new.renderer.render(body)
4+
Marksmith::Renderer.new(body:).render
55
end
66

77
def marksmith_tag(name, **kwargs, &block)

marksmith.gemspec

-1
Original file line numberDiff line numberDiff line change
@@ -19,5 +19,4 @@ Gem::Specification.new do |spec|
1919
end
2020

2121
spec.add_dependency "activesupport"
22-
spec.add_dependency "redcarpet"
2322
end
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
# Marksmith.configure do |config|
22
# config.automatically_mount_engine = true
33
# config.mount_path = "/marksmith"
4+
# config.parser = "commonmarker"
45
# end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,137 @@
1+
require "test_helper"
2+
require "marksmith/helper"
3+
4+
class CommonmarkerHelperTest < ActiveSupport::TestCase
5+
include Marksmith::Helper
6+
7+
def setup
8+
Marksmith.configuration.parser = "commonmarker"
9+
end
10+
11+
test "marksmithed#renders simple markdown" do
12+
body = "# Hello World\n\nThis is a test."
13+
expected = "<h1><a href=\"#hello-world\" aria-hidden=\"true\" class=\"anchor\" id=\"hello-world\"></a>Hello World</h1>\n<p>This is a test.</p>\n"
14+
15+
assert_equal expected, marksmithed(body)
16+
end
17+
18+
test "marksmithed#when rendering a header" do
19+
body = "# Header"
20+
expected = "<h1><a href=\"#header\" aria-hidden=\"true\" class=\"anchor\" id=\"header\"></a>Header</h1>\n"
21+
22+
assert_equal expected, marksmithed(body)
23+
end
24+
25+
# Test rendering a list
26+
test "marksmithed#when rendering a list" do
27+
body = "- item1\n- item2"
28+
expected = "<ul>\n<li>item1</li>\n<li>item2</li>\n</ul>\n"
29+
30+
assert_equal expected, marksmithed(body)
31+
end
32+
33+
# Test rendering a code block
34+
test "marksmithed#when rendering a code block" do
35+
body = "```\ndef hello\n puts 'hello'\nend\n```"
36+
expected = "<pre style=\"background-color:#2b303b;\"><code><span style=\"color:#c0c5ce;\">def hello\n</span><span style=\"color:#c0c5ce;\"> puts &#39;hello&#39;\n</span><span style=\"color:#c0c5ce;\">end\n</span></code></pre>\n"
37+
38+
assert_equal expected, marksmithed(body)
39+
end
40+
41+
# Test rendering a table
42+
test "marksmithed#when rendering a table" do
43+
body = "| a | b |\n|---|---|\n| c | d |"
44+
expected = "<table>\n<thead>\n<tr>\n<th>a</th>\n<th>b</th>\n</tr>\n</thead>\n<tbody>\n<tr>\n<td>c</td>\n<td>d</td>\n</tr>\n</tbody>\n</table>\n"
45+
46+
assert_equal expected, marksmithed(body)
47+
end
48+
49+
# Test rendering strikethrough text
50+
test "marksmithed#when rendering strikethrough text" do
51+
body = "This is ~~strikethrough~~ text."
52+
expected = "<p>This is <del>strikethrough</del> text.</p>\n"
53+
54+
assert_equal expected, marksmithed(body)
55+
end
56+
57+
# Test rendering underline text using single underscores (emphasis)
58+
test "marksmithed#when rendering underline (emphasis) text" do
59+
body = "This is _underline_ text."
60+
expected = "<p>This is <em>underline</em> text.</p>\n"
61+
62+
assert_equal expected, marksmithed(body)
63+
end
64+
65+
# Test rendering underline text using double underscores (underline)
66+
test "marksmithed#when rendering strong text using double underscores" do
67+
body = "This is __underline__ text."
68+
expected = "<p>This is <strong>underline</strong> text.</p>\n"
69+
70+
assert_equal expected, marksmithed(body)
71+
end
72+
73+
# Test rendering a blockquote
74+
test "marksmithed#when rendering a blockquote" do
75+
body = "> This is a blockquote."
76+
expected = "<blockquote>\n<p>This is a blockquote.</p>\n</blockquote>\n"
77+
78+
assert_equal expected, marksmithed(body)
79+
end
80+
81+
# Test rendering autolinks
82+
test "marksmithed#when rendering autolinks" do
83+
body = "Visit https://example.com"
84+
expected = "<p>Visit <a href=\"https://example.com\">https://example.com</a></p>\n"
85+
86+
assert_equal expected, marksmithed(body)
87+
end
88+
89+
# Test rendering fenced code blocks with language specification
90+
test "marksmithed#when rendering fenced code blocks with language" do
91+
body = "```ruby\ndef hello\n puts 'hello'\nend\n```"
92+
expected = "<pre lang=\"ruby\" style=\"background-color:#2b303b;\"><code><span style=\"color:#b48ead;\">def </span><span style=\"color:#8fa1b3;\">hello\n</span><span style=\"color:#c0c5ce;\"> </span><span style=\"color:#96b5b4;\">puts </span><span style=\"color:#c0c5ce;\">&#39;</span><span style=\"color:#a3be8c;\">hello</span><span style=\"color:#c0c5ce;\">&#39;\n</span><span style=\"color:#b48ead;\">end\n</span></code></pre>\n"
93+
94+
assert_equal expected, marksmithed(body)
95+
end
96+
97+
# Test rendering hard line breaks
98+
test "marksmithed#when rendering with hard line breaks" do
99+
body = "Line1\nLine2"
100+
expected = "<p>Line1<br />\nLine2</p>\n"
101+
102+
assert_equal expected, marksmithed(body)
103+
end
104+
105+
# Test rendering with lax spacing (multiple paragraphs without explicit breaks)
106+
test "marksmithed#when rendering with lax spacing" do
107+
body = "Paragraph one
108+
Paragraph two"
109+
expected = "<p>Paragraph one<br />\nParagraph two</p>\n"
110+
111+
assert_equal expected, marksmithed(body)
112+
end
113+
114+
# Test rendering emphasis and strong emphasis
115+
test "marksmithed#when rendering emphasis and strong emphasis" do
116+
body = "This is *emphasized* and this is **strongly emphasized**."
117+
expected = "<p>This is <em>emphasized</em> and this is <strong>strongly emphasized</strong>.</p>\n"
118+
119+
assert_equal expected, marksmithed(body)
120+
end
121+
122+
# Test rendering an empty string
123+
test "marksmithed#when rendering an empty string" do
124+
body = ""
125+
expected = ""
126+
127+
assert_equal expected, marksmithed(body)
128+
end
129+
130+
# Test handling of nil input
131+
test "marksmithed#when handling nil input" do
132+
body = nil
133+
expected = ""
134+
135+
assert_equal expected, marksmithed(body.to_s)
136+
end
137+
end

test/lib/marksmith/helper_test.rb test/lib/marksmith/redcarpet_helper_test.rb

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,13 @@
11
require "test_helper"
22
require "marksmith/helper"
33

4-
class HelperTest < ActiveSupport::TestCase
4+
class RedcarpetHelperTest < ActiveSupport::TestCase
55
include Marksmith::Helper
66

7+
def setup
8+
Marksmith.configuration.parser = "redcarpet"
9+
end
10+
711
test "marksmithed#renders simple markdown" do
812
body = "# Hello World\n\nThis is a test."
913
expected = "<h1>Hello World</h1>\n\n<p>This is a test.</p>\n"

0 commit comments

Comments
 (0)