commonmarker 0.23.10 → 2.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (97) hide show
  1. checksums.yaml +4 -4
  2. data/Cargo.lock +1156 -0
  3. data/Cargo.toml +7 -0
  4. data/README.md +237 -172
  5. data/ext/commonmarker/Cargo.toml +20 -0
  6. data/ext/commonmarker/extconf.rb +3 -6
  7. data/ext/commonmarker/src/lib.rs +103 -0
  8. data/ext/commonmarker/src/node.rs +1221 -0
  9. data/ext/commonmarker/src/options.rs +220 -0
  10. data/ext/commonmarker/src/plugins/syntax_highlighting.rs +166 -0
  11. data/ext/commonmarker/src/plugins.rs +6 -0
  12. data/ext/commonmarker/src/utils.rs +8 -0
  13. data/lib/commonmarker/config.rb +92 -40
  14. data/lib/commonmarker/constants.rb +7 -0
  15. data/lib/commonmarker/extension.rb +14 -0
  16. data/lib/commonmarker/node/ast.rb +8 -0
  17. data/lib/commonmarker/node/inspect.rb +14 -4
  18. data/lib/commonmarker/node.rb +29 -47
  19. data/lib/commonmarker/renderer.rb +1 -127
  20. data/lib/commonmarker/utils.rb +22 -0
  21. data/lib/commonmarker/version.rb +2 -2
  22. data/lib/commonmarker.rb +27 -25
  23. metadata +38 -191
  24. data/Rakefile +0 -109
  25. data/bin/commonmarker +0 -118
  26. data/commonmarker.gemspec +0 -38
  27. data/ext/commonmarker/arena.c +0 -104
  28. data/ext/commonmarker/autolink.c +0 -508
  29. data/ext/commonmarker/autolink.h +0 -8
  30. data/ext/commonmarker/blocks.c +0 -1622
  31. data/ext/commonmarker/buffer.c +0 -278
  32. data/ext/commonmarker/buffer.h +0 -116
  33. data/ext/commonmarker/case_fold_switch.inc +0 -4327
  34. data/ext/commonmarker/chunk.h +0 -135
  35. data/ext/commonmarker/cmark-gfm-core-extensions.h +0 -54
  36. data/ext/commonmarker/cmark-gfm-extension_api.h +0 -737
  37. data/ext/commonmarker/cmark-gfm-extensions_export.h +0 -42
  38. data/ext/commonmarker/cmark-gfm.h +0 -833
  39. data/ext/commonmarker/cmark-gfm_export.h +0 -42
  40. data/ext/commonmarker/cmark-gfm_version.h +0 -7
  41. data/ext/commonmarker/cmark.c +0 -55
  42. data/ext/commonmarker/cmark_ctype.c +0 -44
  43. data/ext/commonmarker/cmark_ctype.h +0 -33
  44. data/ext/commonmarker/commonmark.c +0 -514
  45. data/ext/commonmarker/commonmarker.c +0 -1308
  46. data/ext/commonmarker/commonmarker.h +0 -16
  47. data/ext/commonmarker/config.h +0 -76
  48. data/ext/commonmarker/core-extensions.c +0 -27
  49. data/ext/commonmarker/entities.inc +0 -2138
  50. data/ext/commonmarker/ext_scanners.c +0 -879
  51. data/ext/commonmarker/ext_scanners.h +0 -24
  52. data/ext/commonmarker/footnotes.c +0 -63
  53. data/ext/commonmarker/footnotes.h +0 -27
  54. data/ext/commonmarker/houdini.h +0 -57
  55. data/ext/commonmarker/houdini_href_e.c +0 -100
  56. data/ext/commonmarker/houdini_html_e.c +0 -66
  57. data/ext/commonmarker/houdini_html_u.c +0 -149
  58. data/ext/commonmarker/html.c +0 -502
  59. data/ext/commonmarker/html.h +0 -27
  60. data/ext/commonmarker/inlines.c +0 -1788
  61. data/ext/commonmarker/inlines.h +0 -29
  62. data/ext/commonmarker/iterator.c +0 -159
  63. data/ext/commonmarker/iterator.h +0 -26
  64. data/ext/commonmarker/latex.c +0 -468
  65. data/ext/commonmarker/linked_list.c +0 -37
  66. data/ext/commonmarker/man.c +0 -274
  67. data/ext/commonmarker/map.c +0 -129
  68. data/ext/commonmarker/map.h +0 -44
  69. data/ext/commonmarker/node.c +0 -1045
  70. data/ext/commonmarker/node.h +0 -167
  71. data/ext/commonmarker/parser.h +0 -59
  72. data/ext/commonmarker/plaintext.c +0 -218
  73. data/ext/commonmarker/plugin.c +0 -36
  74. data/ext/commonmarker/plugin.h +0 -34
  75. data/ext/commonmarker/references.c +0 -43
  76. data/ext/commonmarker/references.h +0 -26
  77. data/ext/commonmarker/registry.c +0 -63
  78. data/ext/commonmarker/registry.h +0 -24
  79. data/ext/commonmarker/render.c +0 -213
  80. data/ext/commonmarker/render.h +0 -62
  81. data/ext/commonmarker/scanners.c +0 -14056
  82. data/ext/commonmarker/scanners.h +0 -70
  83. data/ext/commonmarker/scanners.re +0 -341
  84. data/ext/commonmarker/strikethrough.c +0 -167
  85. data/ext/commonmarker/strikethrough.h +0 -9
  86. data/ext/commonmarker/syntax_extension.c +0 -149
  87. data/ext/commonmarker/syntax_extension.h +0 -34
  88. data/ext/commonmarker/table.c +0 -917
  89. data/ext/commonmarker/table.h +0 -12
  90. data/ext/commonmarker/tagfilter.c +0 -60
  91. data/ext/commonmarker/tagfilter.h +0 -8
  92. data/ext/commonmarker/tasklist.c +0 -156
  93. data/ext/commonmarker/tasklist.h +0 -8
  94. data/ext/commonmarker/utf8.c +0 -317
  95. data/ext/commonmarker/utf8.h +0 -35
  96. data/ext/commonmarker/xml.c +0 -182
  97. data/lib/commonmarker/renderer/html_renderer.rb +0 -256
@@ -0,0 +1,1221 @@
1
+ use comrak::arena_tree::Node as ComrakNode;
2
+ use comrak::nodes::{
3
+ AlertType, Ast as ComrakAst, AstNode as ComrakAstNode, ListDelimType, ListType, NodeAlert,
4
+ NodeCode, NodeCodeBlock, NodeDescriptionItem, NodeFootnoteDefinition, NodeFootnoteReference,
5
+ NodeHeading, NodeHtmlBlock, NodeLink, NodeList, NodeMath, NodeMultilineBlockQuote,
6
+ NodeShortCode, NodeTable, NodeValue as ComrakNodeValue, NodeWikiLink, TableAlignment,
7
+ };
8
+ use magnus::RArray;
9
+ use magnus::{function, method, scan_args, Module, Object, RHash, RModule, Symbol, Value};
10
+ use rctree::Node;
11
+ use typed_arena::Arena;
12
+
13
+ use std::cell::RefCell;
14
+
15
+ use crate::format_options;
16
+
17
+ use crate::plugins::syntax_highlighting::construct_syntax_highlighter_from_plugin;
18
+
19
+ #[derive(Debug, Clone)]
20
+ #[magnus::wrap(class = "Commonmarker::Node::Ast", size, mark)]
21
+ pub struct CommonmarkerAst {
22
+ data: ComrakAst,
23
+ }
24
+
25
+ #[derive(Debug, Clone)]
26
+ #[magnus::wrap(class = "Commonmarker::Node", size, mark)]
27
+ pub struct CommonmarkerNode {
28
+ inner: Node<CommonmarkerAst>,
29
+ }
30
+
31
+ /// SAFETY: This is safe because we only access this data when the GVL is held.
32
+ unsafe impl Send for CommonmarkerNode {}
33
+
34
+ impl CommonmarkerNode {
35
+ pub fn new(args: &[Value]) -> Result<Self, magnus::Error> {
36
+ let args = scan_args::scan_args::<_, (), (), (), _, ()>(args)?;
37
+ let (node_type,): (Symbol,) = args.required;
38
+
39
+ let node = match node_type.to_string().as_str() {
40
+ "document" => ComrakNodeValue::Document,
41
+ "block_quote" => ComrakNodeValue::BlockQuote,
42
+ "footnote_definition" => {
43
+ let kwargs = scan_args::get_kwargs::<_, (String,), (Option<u32>,), ()>(
44
+ args.keywords,
45
+ &["name"],
46
+ &["total_references"],
47
+ )?;
48
+ let (name,) = kwargs.required;
49
+ let (total_reference,) = kwargs.optional;
50
+
51
+ ComrakNodeValue::FootnoteDefinition(NodeFootnoteDefinition {
52
+ // The name of the footnote.
53
+ name,
54
+ // Total number of references to this footnote
55
+ total_references: total_reference.unwrap_or(1),
56
+ })
57
+ }
58
+ "list" => {
59
+ let kwargs = scan_args::get_kwargs::<
60
+ _,
61
+ (Symbol,),
62
+ (
63
+ Option<usize>,
64
+ Option<usize>,
65
+ Option<usize>,
66
+ Option<String>,
67
+ Option<u8>,
68
+ Option<bool>,
69
+ Option<bool>,
70
+ ),
71
+ (),
72
+ >(
73
+ args.keywords,
74
+ &["type"],
75
+ &[
76
+ "marker_offset",
77
+ "padding",
78
+ "start",
79
+ "delimiter",
80
+ "bullet_char",
81
+ "tight",
82
+ "task_list",
83
+ ],
84
+ )?;
85
+
86
+ let (list_type,) = kwargs.required;
87
+ let (marker_offset, padding, start, delimiter, bullet_char, tight, task_list) =
88
+ kwargs.optional;
89
+
90
+ let commonmark_list_type = list_type.to_string();
91
+
92
+ if commonmark_list_type != "bullet" && commonmark_list_type != "ordered" {
93
+ return Err(magnus::Error::new(
94
+ magnus::exception::arg_error(),
95
+ "list type must be `bullet` or `ordered`",
96
+ ));
97
+ }
98
+
99
+ let comrak_list_type = if commonmark_list_type == "ordered" {
100
+ ListType::Ordered
101
+ } else {
102
+ ListType::Bullet
103
+ };
104
+
105
+ let comrak_delimiter = match delimiter.unwrap_or("".to_string()).as_str() {
106
+ ")" => ListDelimType::Paren,
107
+ _ => ListDelimType::Period,
108
+ };
109
+
110
+ ComrakNodeValue::List(NodeList {
111
+ // The kind of list (bullet (unordered) or ordered).
112
+ list_type: comrak_list_type,
113
+ // Number of spaces before the list marker.
114
+ marker_offset: marker_offset.unwrap_or(0),
115
+ // Number of characters between the start of the list marker and the item text (including the list marker(s)).
116
+ padding: padding.unwrap_or(0),
117
+ // For ordered lists, the ordinal the list starts at.
118
+ start: start.unwrap_or(0),
119
+ // For ordered lists, the delimiter after each number.
120
+ delimiter: comrak_delimiter,
121
+ // For bullet lists, the character used for each bullet.
122
+ bullet_char: bullet_char.unwrap_or(0),
123
+ // Whether the list is [tight](https://github.github.com/gfm/#tight), i.e. whether the
124
+ // paragraphs are wrapped in `<p>` tags when formatted as HTML.
125
+ tight: tight.unwrap_or(false),
126
+ is_task_list: task_list.unwrap_or(false),
127
+ })
128
+ }
129
+ "description_list" => ComrakNodeValue::DescriptionList,
130
+ "description_item" => {
131
+ let kwargs = scan_args::get_kwargs::<
132
+ _,
133
+ (),
134
+ (Option<usize>, Option<usize>, Option<bool>),
135
+ (),
136
+ >(
137
+ args.keywords, &[], &["marker_offset", "padding", "tight"]
138
+ )?;
139
+
140
+ let (marker_offset, padding, tight) = kwargs.optional;
141
+
142
+ ComrakNodeValue::DescriptionItem(NodeDescriptionItem {
143
+ // Number of spaces before the list marker.
144
+ marker_offset: marker_offset.unwrap_or(0),
145
+ // Number of characters between the start of the list marker and the item text (including the list marker(s)).
146
+ padding: padding.unwrap_or(0),
147
+ // Whether the list is [tight](https://github.github.com/gfm/#tight), i.e. whether the
148
+ // paragraphs are wrapped in `<p>` tags when formatted as HTML.
149
+ tight: tight.unwrap_or(false),
150
+ })
151
+ }
152
+ "description_term" => ComrakNodeValue::DescriptionTerm,
153
+ "description_details" => ComrakNodeValue::DescriptionDetails,
154
+ "code_block" => {
155
+ let kwargs = scan_args::get_kwargs::<
156
+ _,
157
+ (bool,),
158
+ (
159
+ Option<u8>,
160
+ Option<usize>,
161
+ Option<usize>,
162
+ Option<String>,
163
+ Option<String>,
164
+ ),
165
+ (),
166
+ >(
167
+ args.keywords,
168
+ &["fenced"],
169
+ &[
170
+ "fence_char",
171
+ "fence_length",
172
+ "fence_offset",
173
+ "info",
174
+ "literal",
175
+ ],
176
+ )?;
177
+ let (fenced,) = kwargs.required;
178
+ let (fence_char, fence_length, fence_offset, info, literal) = kwargs.optional;
179
+
180
+ ComrakNodeValue::CodeBlock(NodeCodeBlock {
181
+ // Whether the code block is fenced.
182
+ fenced,
183
+ // For fenced code blocks, the fence character itself (`` ` `` or `~`).
184
+ fence_char: fence_char.unwrap_or(b'`'),
185
+ // For fenced code blocks, the length of the fence.
186
+ fence_length: fence_length.unwrap_or(0),
187
+ // For fenced code blocks, the indentation level of the code within the block.
188
+ fence_offset: fence_offset.unwrap_or(0),
189
+
190
+ // For fenced code blocks, the [info string](https://github.github.com/gfm/#info-string) after
191
+ // the opening fence, if any.
192
+ info: info.unwrap_or(String::with_capacity(10)),
193
+
194
+ // The literal contents of the code block. As the contents are not interpreted as Markdown at
195
+ // all, they are contained within this structure, rather than inserted into a child inline of
196
+ // any kind.
197
+ literal: literal.unwrap_or(String::new()),
198
+ })
199
+ }
200
+ "html_block" => {
201
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<u8>, Option<String>), ()>(
202
+ args.keywords,
203
+ &[],
204
+ &["block_type", "literal"],
205
+ )?;
206
+
207
+ let (block_type, literal) = kwargs.optional;
208
+
209
+ ComrakNodeValue::HtmlBlock(NodeHtmlBlock {
210
+ // Number of spaces before the list marker.
211
+ block_type: block_type.unwrap_or(0),
212
+ // Number of characters between the start of the list marker and the item text (including the list marker(s)).
213
+ literal: literal.unwrap_or(String::new()),
214
+ })
215
+ }
216
+ "paragraph" => ComrakNodeValue::Paragraph,
217
+ "heading" => {
218
+ let kwargs = scan_args::get_kwargs::<_, (u8,), (Option<bool>,), ()>(
219
+ args.keywords,
220
+ &["level"],
221
+ &["setext"],
222
+ )?;
223
+
224
+ let (level,) = kwargs.required;
225
+ let (setext,) = kwargs.optional;
226
+
227
+ ComrakNodeValue::Heading(NodeHeading {
228
+ // Number of spaces before the list marker.
229
+ level,
230
+ // Number of characters between the start of the list marker and the item text (including the list marker(s)).
231
+ setext: setext.unwrap_or(false),
232
+ })
233
+ }
234
+ "thematic_break" => ComrakNodeValue::ThematicBreak,
235
+ "table" => {
236
+ let kwargs = scan_args::get_kwargs::<_, (RArray, usize, usize, usize), (), ()>(
237
+ args.keywords,
238
+ &[
239
+ "alignments",
240
+ "num_columns",
241
+ "num_rows",
242
+ "num_nonempty_cells",
243
+ ],
244
+ &[],
245
+ )?;
246
+
247
+ let (alignments, num_columns, num_rows, num_nonempty_cells) = kwargs.required;
248
+
249
+ let mut comrak_alignments = vec![];
250
+ alignments
251
+ .into_iter()
252
+ .for_each(|alignment| match alignment.to_string().as_str() {
253
+ "left" => {
254
+ comrak_alignments.push(TableAlignment::Left);
255
+ }
256
+ "right" => {
257
+ comrak_alignments.push(TableAlignment::Right);
258
+ }
259
+ "center" => {
260
+ comrak_alignments.push(TableAlignment::Center);
261
+ }
262
+ _ => {
263
+ comrak_alignments.push(TableAlignment::None);
264
+ }
265
+ });
266
+ ComrakNodeValue::Table(NodeTable {
267
+ // The table alignments
268
+ alignments: comrak_alignments,
269
+
270
+ // Number of columns of the table
271
+ num_columns,
272
+
273
+ // Number of rows of the table
274
+ num_rows,
275
+
276
+ // Number of non-empty, non-autocompleted cells
277
+ num_nonempty_cells,
278
+ })
279
+ }
280
+ "table_row" => {
281
+ let kwargs =
282
+ scan_args::get_kwargs::<_, (bool,), (), ()>(args.keywords, &["header"], &[])?;
283
+
284
+ let (header,) = kwargs.required;
285
+
286
+ ComrakNodeValue::TableRow(header)
287
+ }
288
+ "table_cell" => ComrakNodeValue::TableCell,
289
+ "text" => {
290
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<String>,), ()>(
291
+ args.keywords,
292
+ &[],
293
+ &["content"],
294
+ )?;
295
+
296
+ let (content,) = kwargs.optional;
297
+
298
+ ComrakNodeValue::Text(content.unwrap_or("".to_string()))
299
+ }
300
+ "taskitem" => {
301
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<char>,), ()>(
302
+ args.keywords,
303
+ &[],
304
+ &["mark"],
305
+ )?;
306
+
307
+ let (mark,) = kwargs.optional;
308
+
309
+ ComrakNodeValue::TaskItem(mark)
310
+ }
311
+ "softbreak" => ComrakNodeValue::SoftBreak,
312
+ "linebreak" => ComrakNodeValue::LineBreak,
313
+ "code" => {
314
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<usize>, Option<String>), ()>(
315
+ args.keywords,
316
+ &[],
317
+ &["num_backticks", "literal"],
318
+ )?;
319
+
320
+ let (num_backticks, literal) = kwargs.optional;
321
+
322
+ ComrakNodeValue::Code(NodeCode {
323
+ // The number of backticks
324
+ num_backticks: num_backticks.unwrap_or(1),
325
+ // The content of the inline code span.
326
+ // As the contents are not interpreted as Markdown at all,
327
+ // they are contained within this structure,
328
+ // rather than inserted into a child inline of any kind
329
+ literal: literal.unwrap_or_default(),
330
+ })
331
+ }
332
+ "html_inline" => {
333
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<String>,), ()>(
334
+ args.keywords,
335
+ &[],
336
+ &["content"],
337
+ )?;
338
+
339
+ let (content,) = kwargs.optional;
340
+
341
+ ComrakNodeValue::HtmlInline(content.unwrap_or_default())
342
+ }
343
+ "emph" => ComrakNodeValue::Emph,
344
+ "strong" => ComrakNodeValue::Strong,
345
+ "strikethrough" => ComrakNodeValue::Strikethrough,
346
+ "superscript" => ComrakNodeValue::Superscript,
347
+ "subscript" => ComrakNodeValue::Subscript,
348
+ "link" => {
349
+ let kwargs = scan_args::get_kwargs::<_, (String,), (Option<String>,), ()>(
350
+ args.keywords,
351
+ &["url"],
352
+ &["title"],
353
+ )?;
354
+
355
+ let (url,) = kwargs.required;
356
+ let (title,) = kwargs.optional;
357
+
358
+ ComrakNodeValue::Link(NodeLink {
359
+ // The URL for the link destination or image source.
360
+ url,
361
+ // The title for the link or image.
362
+ //
363
+ // Note this field is used for the `title` attribute by the HTML formatter even for images;
364
+ // `alt` text is supplied in the image inline text.
365
+ title: title.unwrap_or_default(),
366
+ })
367
+ }
368
+ "image" => {
369
+ let kwargs = scan_args::get_kwargs::<_, (String,), (Option<String>,), ()>(
370
+ args.keywords,
371
+ &["url"],
372
+ &["title"],
373
+ )?;
374
+
375
+ let (url,) = kwargs.required;
376
+ let (title,) = kwargs.optional;
377
+
378
+ ComrakNodeValue::Image(NodeLink {
379
+ // The URL for the link destination or image source.
380
+ url,
381
+ // The title for the link or image.
382
+ //
383
+ // Note this field is used for the `title` attribute by the HTML formatter even for images;
384
+ // `alt` text is supplied in the image inline text.
385
+ title: title.unwrap_or_default(),
386
+ })
387
+ }
388
+ "footnote_reference" => {
389
+ let kwargs = scan_args::get_kwargs::<_, (String,), (Option<u32>, Option<u32>), ()>(
390
+ args.keywords,
391
+ &["name"],
392
+ &["ref_num", "ix"],
393
+ )?;
394
+
395
+ let (name,) = kwargs.required;
396
+ let (ref_num, ix) = kwargs.optional;
397
+
398
+ ComrakNodeValue::FootnoteReference(NodeFootnoteReference {
399
+ // The name of the footnote.
400
+ name,
401
+ // The index of reference to the same footnote
402
+ ref_num: ref_num.unwrap_or(0),
403
+ // The index of the footnote in the document.
404
+ ix: ix.unwrap_or(0),
405
+ })
406
+ }
407
+ // #[cfg(feature = "shortcodes")]
408
+ "shortcode" => {
409
+ let kwargs =
410
+ scan_args::get_kwargs::<_, (String,), (), ()>(args.keywords, &["code"], &[])?;
411
+
412
+ let (code,) = kwargs.required;
413
+
414
+ match NodeShortCode::resolve(code.as_str()) {
415
+ Some(shortcode) => ComrakNodeValue::ShortCode(shortcode),
416
+ None => {
417
+ return Err(magnus::Error::new(
418
+ magnus::exception::arg_error(),
419
+ "could not resolve shortcode",
420
+ ));
421
+ }
422
+ }
423
+ }
424
+ "math" => {
425
+ let kwargs = scan_args::get_kwargs::<_, (bool, bool, String), (), ()>(
426
+ args.keywords,
427
+ &["dollar_math", "display_math", "literal"],
428
+ &[],
429
+ )?;
430
+
431
+ let (dollar_math, display_math, literal) = kwargs.required;
432
+
433
+ ComrakNodeValue::Math(NodeMath {
434
+ // Whether this is dollar math (`$` or `$$`).
435
+ // `false` indicates it is code math
436
+ dollar_math,
437
+
438
+ // Whether this is display math (using `$$`)
439
+ display_math,
440
+
441
+ // The literal contents of the math span.
442
+ // As the contents are not interpreted as Markdown at all,
443
+ // they are contained within this structure,
444
+ // rather than inserted into a child inline of any kind.
445
+ literal,
446
+ })
447
+ }
448
+ "multiline_block_quote" => {
449
+ let kwargs = scan_args::get_kwargs::<_, (usize, usize), (), ()>(
450
+ args.keywords,
451
+ &["fence_length", "fence_offset"],
452
+ &[],
453
+ )?;
454
+
455
+ let (fence_length, fence_offset) = kwargs.required;
456
+
457
+ ComrakNodeValue::MultilineBlockQuote(NodeMultilineBlockQuote {
458
+ // The length of the fence.
459
+ fence_length,
460
+ // The indentation level of the fence marker.
461
+ fence_offset,
462
+ })
463
+ }
464
+
465
+ "escaped" => ComrakNodeValue::Escaped,
466
+
467
+ "wikilink" => {
468
+ let kwargs =
469
+ scan_args::get_kwargs::<_, (String,), (), ()>(args.keywords, &["url"], &[])?;
470
+
471
+ let (url,) = kwargs.required;
472
+
473
+ ComrakNodeValue::WikiLink(NodeWikiLink { url })
474
+ }
475
+
476
+ "raw" => {
477
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<String>,), ()>(
478
+ args.keywords,
479
+ &[],
480
+ &["content"],
481
+ )?;
482
+
483
+ let (content,) = kwargs.optional;
484
+
485
+ ComrakNodeValue::Raw(content.unwrap_or_default())
486
+ }
487
+
488
+ "alert" => {
489
+ let kwargs = scan_args::get_kwargs::<
490
+ _,
491
+ (Symbol,),
492
+ (Option<String>, Option<bool>, Option<usize>, Option<usize>),
493
+ (),
494
+ >(
495
+ args.keywords,
496
+ &["type"],
497
+ &["title", "multiline", "fence_length", "fence_offset"],
498
+ )?;
499
+
500
+ let (alert_name,) = kwargs.required;
501
+ let (title, multiline, fence_length, fence_offset) = kwargs.optional;
502
+
503
+ let alert_type = match alert_name.to_string().as_str() {
504
+ "note" => AlertType::Note,
505
+ "tip" => AlertType::Tip,
506
+ "important" => AlertType::Important,
507
+ "warning" => AlertType::Warning,
508
+ _ => {
509
+ return Err(magnus::Error::new(
510
+ magnus::exception::arg_error(),
511
+ "alert type must be `note`, `tip`, `important`, or `warning`",
512
+ ));
513
+ }
514
+ };
515
+
516
+ ComrakNodeValue::Alert(NodeAlert {
517
+ alert_type,
518
+ // Overridden title. If None, then use the default title.
519
+ title,
520
+ // Originated from a multiline blockquote.
521
+ multiline: multiline.unwrap_or(false),
522
+ // The length of the fence (multiline only).
523
+ fence_length: fence_length.unwrap_or(0),
524
+ // The indentation level of the fence marker (multiline only)
525
+ fence_offset: fence_offset.unwrap_or(0),
526
+ })
527
+ }
528
+
529
+ _ => panic!("unknown node type {}", node_type),
530
+ };
531
+
532
+ Ok(CommonmarkerNode {
533
+ inner: Node::new(CommonmarkerAst {
534
+ data: ComrakAst::new(node, (0, 0).into()),
535
+ }),
536
+ })
537
+ }
538
+
539
+ pub fn new_from_comrak_node<'a>(
540
+ comrak_root_node: &'a ComrakAstNode<'a>,
541
+ ) -> Result<CommonmarkerNode, magnus::Error> {
542
+ let comrak_ast = comrak_root_node.data.clone().into_inner();
543
+
544
+ fn iter_nodes<'a>(comrak_node: &'a ComrakAstNode<'a>) -> CommonmarkerNode {
545
+ let comrak_node_ast = comrak_node.data.clone().into_inner();
546
+ let commonmark_node = CommonmarkerNode {
547
+ inner: Node::new(CommonmarkerAst {
548
+ data: comrak_node_ast,
549
+ }),
550
+ };
551
+
552
+ for c in comrak_node.children() {
553
+ match commonmark_node.append_child_node(&iter_nodes(c)) {
554
+ Ok(_) => {}
555
+ Err(e) => {
556
+ panic!("cannot append node: {}", e);
557
+ }
558
+ }
559
+ }
560
+
561
+ commonmark_node
562
+ }
563
+
564
+ let commonmarker_root_node = CommonmarkerNode {
565
+ inner: Node::new(CommonmarkerAst { data: comrak_ast }),
566
+ };
567
+
568
+ for child in comrak_root_node.children() {
569
+ let new_child = iter_nodes(child);
570
+
571
+ commonmarker_root_node.append_child_node(&new_child)?;
572
+ }
573
+
574
+ Ok(commonmarker_root_node)
575
+ }
576
+
577
+ fn type_to_symbol(&self) -> Symbol {
578
+ let node = self.inner.borrow();
579
+ match node.data.value {
580
+ ComrakNodeValue::Document => Symbol::new("document"),
581
+ ComrakNodeValue::BlockQuote => Symbol::new("block_quote"),
582
+ ComrakNodeValue::FootnoteDefinition(_) => Symbol::new("footnote_definition"),
583
+ ComrakNodeValue::List(..) => Symbol::new("list"),
584
+ ComrakNodeValue::DescriptionList => Symbol::new("description_list"),
585
+ ComrakNodeValue::DescriptionItem(_) => Symbol::new("description_item"),
586
+ ComrakNodeValue::DescriptionTerm => Symbol::new("description_term"),
587
+ ComrakNodeValue::DescriptionDetails => Symbol::new("description_details"),
588
+ ComrakNodeValue::Item(..) => Symbol::new("item"),
589
+ ComrakNodeValue::CodeBlock(..) => Symbol::new("code_block"),
590
+ ComrakNodeValue::HtmlBlock(..) => Symbol::new("html_block"),
591
+ ComrakNodeValue::Paragraph => Symbol::new("paragraph"),
592
+ ComrakNodeValue::Heading(..) => Symbol::new("heading"),
593
+ ComrakNodeValue::ThematicBreak => Symbol::new("thematic_break"),
594
+ ComrakNodeValue::Table(..) => Symbol::new("table"),
595
+ ComrakNodeValue::TableRow(..) => Symbol::new("table_row"),
596
+ ComrakNodeValue::TableCell => Symbol::new("table_cell"),
597
+ ComrakNodeValue::Text(..) => Symbol::new("text"),
598
+ ComrakNodeValue::SoftBreak => Symbol::new("softbreak"),
599
+ ComrakNodeValue::LineBreak => Symbol::new("linebreak"),
600
+ ComrakNodeValue::Image(..) => Symbol::new("image"),
601
+ ComrakNodeValue::Link(..) => Symbol::new("link"),
602
+ ComrakNodeValue::Emph => Symbol::new("emph"),
603
+ ComrakNodeValue::Raw(..) => Symbol::new("raw"),
604
+ ComrakNodeValue::Strong => Symbol::new("strong"),
605
+ ComrakNodeValue::Code(..) => Symbol::new("code"),
606
+ ComrakNodeValue::HtmlInline(..) => Symbol::new("html_inline"),
607
+ ComrakNodeValue::Strikethrough => Symbol::new("strikethrough"),
608
+ ComrakNodeValue::FrontMatter(_) => Symbol::new("frontmatter"),
609
+ ComrakNodeValue::TaskItem { .. } => Symbol::new("taskitem"),
610
+ ComrakNodeValue::Superscript => Symbol::new("superscript"),
611
+ ComrakNodeValue::FootnoteReference(..) => Symbol::new("footnote_reference"),
612
+ ComrakNodeValue::ShortCode(_) => Symbol::new("shortcode"),
613
+ ComrakNodeValue::MultilineBlockQuote(_) => Symbol::new("multiline_block_quote"),
614
+ ComrakNodeValue::Escaped => Symbol::new("escaped"),
615
+ ComrakNodeValue::Math(..) => Symbol::new("math"),
616
+ ComrakNodeValue::WikiLink(..) => Symbol::new("wikilink"),
617
+ ComrakNodeValue::Underline => Symbol::new("underline"),
618
+ ComrakNodeValue::Subscript => Symbol::new("subscript"),
619
+ ComrakNodeValue::SpoileredText => Symbol::new("spoilered_text"),
620
+ ComrakNodeValue::EscapedTag(_) => Symbol::new("escaped_tag"),
621
+ ComrakNodeValue::Alert(..) => Symbol::new("alert"),
622
+ }
623
+ }
624
+
625
+ fn get_parent(&self) -> Option<CommonmarkerNode> {
626
+ self.inner.parent().map(|n| CommonmarkerNode { inner: n })
627
+ }
628
+
629
+ fn get_previous_sibling(&self) -> Option<CommonmarkerNode> {
630
+ self.inner
631
+ .previous_sibling()
632
+ .map(|n| CommonmarkerNode { inner: n })
633
+ }
634
+
635
+ fn get_next_sibling(&self) -> Option<CommonmarkerNode> {
636
+ self.inner
637
+ .next_sibling()
638
+ .map(|n| CommonmarkerNode { inner: n })
639
+ }
640
+
641
+ fn get_first_child(&self) -> Option<CommonmarkerNode> {
642
+ self.inner
643
+ .first_child()
644
+ .map(|n| CommonmarkerNode { inner: n })
645
+ }
646
+
647
+ fn get_last_child(&self) -> Option<CommonmarkerNode> {
648
+ self.inner
649
+ .last_child()
650
+ .map(|n| CommonmarkerNode { inner: n })
651
+ }
652
+
653
+ fn prepend_child_node(&self, new_child: &CommonmarkerNode) -> Result<bool, magnus::Error> {
654
+ let node = new_child.inner.clone();
655
+ node.detach();
656
+ self.inner.prepend(node);
657
+
658
+ Ok(true)
659
+ }
660
+
661
+ fn append_child_node(&self, new_child: &CommonmarkerNode) -> Result<bool, magnus::Error> {
662
+ let node = new_child.inner.clone();
663
+ node.detach();
664
+ self.inner.append(node);
665
+
666
+ Ok(true)
667
+ }
668
+
669
+ fn detach_node(&self) -> Result<CommonmarkerNode, magnus::Error> {
670
+ let node = self.inner.make_copy().borrow().data.clone();
671
+ self.inner.detach();
672
+
673
+ Ok(CommonmarkerNode {
674
+ inner: Node::new(CommonmarkerAst { data: node }),
675
+ })
676
+ }
677
+
678
+ fn get_sourcepos(&self) -> Result<RHash, magnus::Error> {
679
+ let node = self.inner.borrow();
680
+
681
+ let result = RHash::new();
682
+ result.aset(Symbol::new("start_line"), node.data.sourcepos.start.line)?;
683
+ result.aset(
684
+ Symbol::new("start_column"),
685
+ node.data.sourcepos.start.column,
686
+ )?;
687
+ result.aset(Symbol::new("end_line"), node.data.sourcepos.end.line)?;
688
+ result.aset(Symbol::new("end_column"), node.data.sourcepos.end.column)?;
689
+
690
+ Ok(result)
691
+ }
692
+
693
+ fn replace_node(&self, new_node: &CommonmarkerNode) -> Result<bool, magnus::Error> {
694
+ self.insert_node_after(new_node)?;
695
+ match self.detach_node() {
696
+ Ok(_) => Ok(true),
697
+ Err(e) => Err(e),
698
+ }
699
+ }
700
+
701
+ fn insert_node_before(&self, new_sibling: &CommonmarkerNode) -> Result<bool, magnus::Error> {
702
+ let node = new_sibling.inner.clone();
703
+ node.detach();
704
+ self.inner.insert_before(node);
705
+
706
+ Ok(true)
707
+ }
708
+
709
+ fn insert_node_after(&self, new_sibling: &CommonmarkerNode) -> Result<bool, magnus::Error> {
710
+ let node = new_sibling.inner.clone();
711
+ node.detach();
712
+ self.inner.insert_after(node);
713
+
714
+ Ok(true)
715
+ }
716
+
717
+ fn get_url(&self) -> Result<String, magnus::Error> {
718
+ let node = self.inner.borrow();
719
+
720
+ match &node.data.value {
721
+ ComrakNodeValue::Link(link) => Ok(link.url.to_string()),
722
+ ComrakNodeValue::Image(image) => Ok(image.url.to_string()),
723
+ _ => Err(magnus::Error::new(
724
+ magnus::exception::type_error(),
725
+ "node is not an image or link node",
726
+ )),
727
+ }
728
+ }
729
+
730
+ fn set_url(&self, new_url: String) -> Result<bool, magnus::Error> {
731
+ let mut node = self.inner.borrow_mut();
732
+
733
+ match node.data.value {
734
+ ComrakNodeValue::Link(ref mut link) => {
735
+ link.url = new_url;
736
+ Ok(true)
737
+ }
738
+ ComrakNodeValue::Image(ref mut image) => {
739
+ image.url = new_url;
740
+ Ok(true)
741
+ }
742
+ _ => Err(magnus::Error::new(
743
+ magnus::exception::type_error(),
744
+ "node is not an image or link node",
745
+ )),
746
+ }
747
+ }
748
+
749
+ fn get_string_content(&self) -> Result<String, magnus::Error> {
750
+ let node = self.inner.borrow();
751
+
752
+ match node.data.value {
753
+ ComrakNodeValue::Code(ref code) => return Ok(code.literal.to_string()),
754
+ ComrakNodeValue::CodeBlock(ref code_block) => {
755
+ return Ok(code_block.literal.to_string())
756
+ }
757
+ _ => {}
758
+ }
759
+
760
+ match node.data.value.text() {
761
+ Some(s) => Ok(s.to_string()),
762
+ None => Err(magnus::Error::new(
763
+ magnus::exception::type_error(),
764
+ "node does not have string content",
765
+ )),
766
+ }
767
+ }
768
+
769
+ fn set_string_content(&self, new_content: String) -> Result<bool, magnus::Error> {
770
+ let mut node = self.inner.borrow_mut();
771
+
772
+ match node.data.value {
773
+ ComrakNodeValue::Code(ref mut code) => {
774
+ code.literal = new_content;
775
+ return Ok(true);
776
+ }
777
+ ComrakNodeValue::CodeBlock(ref mut code_block) => {
778
+ code_block.literal = new_content;
779
+ return Ok(true);
780
+ }
781
+ _ => {}
782
+ }
783
+
784
+ match node.data.value.text_mut() {
785
+ Some(s) => {
786
+ *s = new_content;
787
+ Ok(true)
788
+ }
789
+ None => Err(magnus::Error::new(
790
+ magnus::exception::type_error(),
791
+ "node does not have string content",
792
+ )),
793
+ }
794
+ }
795
+
796
+ fn get_title(&self) -> Result<String, magnus::Error> {
797
+ let node = self.inner.borrow();
798
+
799
+ match &node.data.value {
800
+ ComrakNodeValue::Link(link) => Ok(link.title.to_string()),
801
+ ComrakNodeValue::Image(image) => Ok(image.title.to_string()),
802
+ _ => Err(magnus::Error::new(
803
+ magnus::exception::type_error(),
804
+ "node is not an image or link node",
805
+ )),
806
+ }
807
+ }
808
+
809
+ fn set_title(&self, new_title: String) -> Result<bool, magnus::Error> {
810
+ let mut node = self.inner.borrow_mut();
811
+
812
+ match node.data.value {
813
+ ComrakNodeValue::Link(ref mut link) => {
814
+ link.title = new_title;
815
+ Ok(true)
816
+ }
817
+ ComrakNodeValue::Image(ref mut image) => {
818
+ image.title = new_title;
819
+ Ok(true)
820
+ }
821
+ _ => Err(magnus::Error::new(
822
+ magnus::exception::type_error(),
823
+ "node is not an image or link node",
824
+ )),
825
+ }
826
+ }
827
+
828
+ fn get_header_level(&self) -> Result<u8, magnus::Error> {
829
+ let node = self.inner.borrow();
830
+
831
+ match &node.data.value {
832
+ ComrakNodeValue::Heading(heading) => Ok(heading.level),
833
+ _ => Err(magnus::Error::new(
834
+ magnus::exception::type_error(),
835
+ "node is not a heading node",
836
+ )),
837
+ }
838
+ }
839
+
840
+ fn set_header_level(&self, new_level: u8) -> Result<bool, magnus::Error> {
841
+ let mut node = self.inner.borrow_mut();
842
+
843
+ match node.data.value {
844
+ ComrakNodeValue::Heading(ref mut heading) => {
845
+ heading.level = new_level;
846
+ Ok(true)
847
+ }
848
+ _ => Err(magnus::Error::new(
849
+ magnus::exception::type_error(),
850
+ "node is not a heading node",
851
+ )),
852
+ }
853
+ }
854
+
855
+ fn get_list_type(&self) -> Result<Symbol, magnus::Error> {
856
+ let node = self.inner.borrow();
857
+
858
+ match &node.data.value {
859
+ ComrakNodeValue::List(list) => match list.list_type {
860
+ comrak::nodes::ListType::Bullet => Ok(Symbol::new("bullet")),
861
+ comrak::nodes::ListType::Ordered => Ok(Symbol::new("ordered")),
862
+ },
863
+ _ => Err(magnus::Error::new(
864
+ magnus::exception::type_error(),
865
+ "node is not a list node",
866
+ )),
867
+ }
868
+ }
869
+
870
+ fn set_list_type(&self, new_type: Symbol) -> Result<bool, magnus::Error> {
871
+ let mut node = self.inner.borrow_mut();
872
+
873
+ match node.data.value {
874
+ ComrakNodeValue::List(ref mut list) => {
875
+ match new_type.to_string().as_str() {
876
+ "bullet" => list.list_type = comrak::nodes::ListType::Bullet,
877
+ "ordered" => list.list_type = comrak::nodes::ListType::Ordered,
878
+ _ => return Ok(false),
879
+ }
880
+ Ok(true)
881
+ }
882
+ _ => Err(magnus::Error::new(
883
+ magnus::exception::type_error(),
884
+ "node is not a list node",
885
+ )),
886
+ }
887
+ }
888
+
889
+ fn get_list_start(&self) -> Result<usize, magnus::Error> {
890
+ let node = self.inner.borrow();
891
+
892
+ match &node.data.value {
893
+ ComrakNodeValue::List(list) => Ok(list.start),
894
+ _ => Err(magnus::Error::new(
895
+ magnus::exception::type_error(),
896
+ "node is not a list node",
897
+ )),
898
+ }
899
+ }
900
+
901
+ fn set_list_start(&self, new_start: usize) -> Result<bool, magnus::Error> {
902
+ let mut node = self.inner.borrow_mut();
903
+
904
+ match node.data.value {
905
+ ComrakNodeValue::List(ref mut list) => {
906
+ list.start = new_start;
907
+ Ok(true)
908
+ }
909
+ _ => Err(magnus::Error::new(
910
+ magnus::exception::type_error(),
911
+ "node is not a list node",
912
+ )),
913
+ }
914
+ }
915
+
916
+ fn get_list_tight(&self) -> Result<bool, magnus::Error> {
917
+ let node = self.inner.borrow();
918
+
919
+ match &node.data.value {
920
+ ComrakNodeValue::List(list) => Ok(list.tight),
921
+ _ => Err(magnus::Error::new(
922
+ magnus::exception::type_error(),
923
+ "node is not a list node",
924
+ )),
925
+ }
926
+ }
927
+
928
+ fn set_list_tight(&self, new_tight: bool) -> Result<bool, magnus::Error> {
929
+ let mut node = self.inner.borrow_mut();
930
+
931
+ match node.data.value {
932
+ ComrakNodeValue::List(ref mut list) => {
933
+ list.tight = new_tight;
934
+ Ok(true)
935
+ }
936
+ _ => Err(magnus::Error::new(
937
+ magnus::exception::type_error(),
938
+ "node is not a list node",
939
+ )),
940
+ }
941
+ }
942
+
943
+ fn get_fence_info(&self) -> Result<String, magnus::Error> {
944
+ let node = self.inner.borrow();
945
+
946
+ match &node.data.value {
947
+ ComrakNodeValue::CodeBlock(code_block) => Ok(code_block.info.to_string()),
948
+ _ => Err(magnus::Error::new(
949
+ magnus::exception::type_error(),
950
+ "node is not a code block node",
951
+ )),
952
+ }
953
+ }
954
+
955
+ fn set_fence_info(&self, new_info: String) -> Result<bool, magnus::Error> {
956
+ let mut node = self.inner.borrow_mut();
957
+
958
+ match node.data.value {
959
+ ComrakNodeValue::CodeBlock(ref mut code_block) => {
960
+ code_block.info = new_info;
961
+ Ok(true)
962
+ }
963
+ _ => Err(magnus::Error::new(
964
+ magnus::exception::type_error(),
965
+ "node is not a code block node",
966
+ )),
967
+ }
968
+ }
969
+
970
+ fn to_html(&self, args: &[Value]) -> Result<String, magnus::Error> {
971
+ let args = scan_args::scan_args::<(), (), (), (), _, ()>(args)?;
972
+
973
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<RHash>, Option<RHash>), ()>(
974
+ args.keywords,
975
+ &[],
976
+ &["options", "plugins"],
977
+ )?;
978
+ let (rb_options, rb_plugins) = kwargs.optional;
979
+
980
+ let comrak_options = match format_options(rb_options) {
981
+ Ok(options) => options,
982
+ Err(err) => return Err(err),
983
+ };
984
+
985
+ let mut comrak_plugins = comrak::Plugins::default();
986
+
987
+ let syntect_adapter = match construct_syntax_highlighter_from_plugin(rb_plugins) {
988
+ Ok(Some(adapter)) => Some(adapter),
989
+ Ok(None) => None,
990
+ Err(err) => return Err(err),
991
+ };
992
+
993
+ match syntect_adapter {
994
+ Some(ref adapter) => comrak_plugins.render.codefence_syntax_highlighter = Some(adapter),
995
+ None => comrak_plugins.render.codefence_syntax_highlighter = None,
996
+ }
997
+
998
+ let arena: Arena<ComrakAstNode> = Arena::new();
999
+ fn iter_nodes<'a>(
1000
+ arena: &'a Arena<comrak::arena_tree::Node<'a, RefCell<ComrakAst>>>,
1001
+ node: &CommonmarkerNode,
1002
+ ) -> &'a comrak::arena_tree::Node<'a, std::cell::RefCell<comrak::nodes::Ast>> {
1003
+ let comrak_node: &'a mut ComrakAstNode = arena.alloc(ComrakNode::new(RefCell::new(
1004
+ node.inner.borrow().data.clone(),
1005
+ )));
1006
+
1007
+ for c in node.inner.children() {
1008
+ let child = CommonmarkerNode { inner: c };
1009
+ let child_node = iter_nodes(arena, &child);
1010
+ comrak_node.append(child_node);
1011
+ }
1012
+
1013
+ comrak_node
1014
+ }
1015
+
1016
+ let comrak_root_node: ComrakNode<RefCell<ComrakAst>> =
1017
+ ComrakNode::new(RefCell::new(self.inner.borrow().data.clone()));
1018
+
1019
+ for c in self.inner.children() {
1020
+ let child = CommonmarkerNode { inner: c };
1021
+
1022
+ let new_child = iter_nodes(&arena, &child);
1023
+
1024
+ comrak_root_node.append(new_child);
1025
+ }
1026
+
1027
+ let mut output = vec![];
1028
+ match comrak::format_html_with_plugins(
1029
+ &comrak_root_node,
1030
+ &comrak_options,
1031
+ &mut output,
1032
+ &comrak_plugins,
1033
+ ) {
1034
+ Ok(_) => {}
1035
+ Err(e) => {
1036
+ return Err(magnus::Error::new(
1037
+ magnus::exception::runtime_error(),
1038
+ format!("cannot convert into html: {}", e),
1039
+ ));
1040
+ }
1041
+ }
1042
+
1043
+ match std::str::from_utf8(&output) {
1044
+ Ok(s) => Ok(s.to_string()),
1045
+ Err(_e) => Err(magnus::Error::new(
1046
+ magnus::exception::runtime_error(),
1047
+ "cannot convert into utf-8",
1048
+ )),
1049
+ }
1050
+ }
1051
+
1052
+ fn to_commonmark(&self, args: &[Value]) -> Result<String, magnus::Error> {
1053
+ let args = scan_args::scan_args::<(), (), (), (), _, ()>(args)?;
1054
+
1055
+ let kwargs = scan_args::get_kwargs::<_, (), (Option<RHash>, Option<RHash>), ()>(
1056
+ args.keywords,
1057
+ &[],
1058
+ &["options", "plugins"],
1059
+ )?;
1060
+ let (rb_options, rb_plugins) = kwargs.optional;
1061
+
1062
+ let _comrak_options = format_options(rb_options);
1063
+ let comrak_options = match format_options(rb_options) {
1064
+ Ok(options) => options,
1065
+ Err(err) => return Err(err),
1066
+ };
1067
+
1068
+ let mut comrak_plugins = comrak::Plugins::default();
1069
+
1070
+ let syntect_adapter = match construct_syntax_highlighter_from_plugin(rb_plugins) {
1071
+ Ok(Some(adapter)) => Some(adapter),
1072
+ Ok(None) => None,
1073
+ Err(err) => return Err(err),
1074
+ };
1075
+
1076
+ match syntect_adapter {
1077
+ Some(ref adapter) => comrak_plugins.render.codefence_syntax_highlighter = Some(adapter),
1078
+ None => comrak_plugins.render.codefence_syntax_highlighter = None,
1079
+ }
1080
+
1081
+ let arena: Arena<ComrakAstNode> = Arena::new();
1082
+ fn iter_nodes<'a>(
1083
+ arena: &'a Arena<comrak::arena_tree::Node<'a, RefCell<ComrakAst>>>,
1084
+ node: &CommonmarkerNode,
1085
+ ) -> &'a comrak::arena_tree::Node<'a, std::cell::RefCell<comrak::nodes::Ast>> {
1086
+ let comrak_node: &'a mut ComrakAstNode = arena.alloc(ComrakNode::new(RefCell::new(
1087
+ node.inner.borrow().data.clone(),
1088
+ )));
1089
+
1090
+ for c in node.inner.children() {
1091
+ let child = CommonmarkerNode { inner: c };
1092
+ let child_node = iter_nodes(arena, &child);
1093
+ comrak_node.append(child_node);
1094
+ }
1095
+
1096
+ comrak_node
1097
+ }
1098
+
1099
+ let comrak_root_node: ComrakNode<RefCell<ComrakAst>> =
1100
+ ComrakNode::new(RefCell::new(self.inner.borrow().data.clone()));
1101
+
1102
+ for c in self.inner.children() {
1103
+ let child = CommonmarkerNode { inner: c };
1104
+
1105
+ let new_child = iter_nodes(&arena, &child);
1106
+
1107
+ comrak_root_node.append(new_child);
1108
+ }
1109
+
1110
+ let mut output = vec![];
1111
+ match comrak::format_commonmark_with_plugins(
1112
+ &comrak_root_node,
1113
+ &comrak_options,
1114
+ &mut output,
1115
+ &comrak_plugins,
1116
+ ) {
1117
+ Ok(_) => {}
1118
+ Err(e) => {
1119
+ return Err(magnus::Error::new(
1120
+ magnus::exception::runtime_error(),
1121
+ format!("cannot convert into html: {}", e),
1122
+ ));
1123
+ }
1124
+ }
1125
+
1126
+ match std::str::from_utf8(&output) {
1127
+ Ok(s) => Ok(s.to_string()),
1128
+ Err(_e) => Err(magnus::Error::new(
1129
+ magnus::exception::runtime_error(),
1130
+ "cannot convert into utf-8",
1131
+ )),
1132
+ }
1133
+ }
1134
+ }
1135
+
1136
+ pub fn init(m_commonmarker: RModule) -> Result<(), magnus::Error> {
1137
+ let c_node = m_commonmarker
1138
+ .define_class("Node", magnus::class::object())
1139
+ .expect("cannot define class Commonmarker::Node");
1140
+
1141
+ c_node.define_singleton_method("new", function!(CommonmarkerNode::new, -1))?;
1142
+
1143
+ c_node.define_method("type", method!(CommonmarkerNode::type_to_symbol, 0))?;
1144
+ c_node.define_method("parent", method!(CommonmarkerNode::get_parent, 0))?;
1145
+ c_node.define_method("first_child", method!(CommonmarkerNode::get_first_child, 0))?;
1146
+ c_node.define_method("last_child", method!(CommonmarkerNode::get_last_child, 0))?;
1147
+ c_node.define_method(
1148
+ "previous_sibling",
1149
+ method!(CommonmarkerNode::get_previous_sibling, 0),
1150
+ )?;
1151
+ c_node.define_method(
1152
+ "next_sibling",
1153
+ method!(CommonmarkerNode::get_next_sibling, 0),
1154
+ )?;
1155
+
1156
+ c_node.define_method("node_to_html", method!(CommonmarkerNode::to_html, -1))?;
1157
+ c_node.define_method(
1158
+ "node_to_commonmark",
1159
+ method!(CommonmarkerNode::to_commonmark, -1),
1160
+ )?;
1161
+
1162
+ c_node.define_method("replace", method!(CommonmarkerNode::replace_node, 1))?;
1163
+
1164
+ c_node.define_method(
1165
+ "insert_before",
1166
+ method!(CommonmarkerNode::insert_node_before, 1),
1167
+ )?;
1168
+ c_node.define_method(
1169
+ "insert_after",
1170
+ method!(CommonmarkerNode::insert_node_after, 1),
1171
+ )?;
1172
+
1173
+ c_node.define_method(
1174
+ "prepend_child",
1175
+ method!(CommonmarkerNode::prepend_child_node, 1),
1176
+ )?;
1177
+ c_node.define_method(
1178
+ "append_child",
1179
+ method!(CommonmarkerNode::append_child_node, 1),
1180
+ )?;
1181
+
1182
+ c_node.define_method("delete", method!(CommonmarkerNode::detach_node, 0))?;
1183
+
1184
+ c_node.define_method(
1185
+ "source_position",
1186
+ method!(CommonmarkerNode::get_sourcepos, 0),
1187
+ )?;
1188
+
1189
+ c_node.define_method(
1190
+ "string_content",
1191
+ method!(CommonmarkerNode::get_string_content, 0),
1192
+ )?;
1193
+ c_node.define_method(
1194
+ "string_content=",
1195
+ method!(CommonmarkerNode::set_string_content, 1),
1196
+ )?;
1197
+
1198
+ c_node.define_method("url", method!(CommonmarkerNode::get_url, 0))?;
1199
+ c_node.define_method("url=", method!(CommonmarkerNode::set_url, 1))?;
1200
+ c_node.define_method("title", method!(CommonmarkerNode::get_title, 0))?;
1201
+ c_node.define_method("title=", method!(CommonmarkerNode::set_title, 1))?;
1202
+
1203
+ c_node.define_method(
1204
+ "header_level",
1205
+ method!(CommonmarkerNode::get_header_level, 0),
1206
+ )?;
1207
+ c_node.define_method(
1208
+ "header_level=",
1209
+ method!(CommonmarkerNode::set_header_level, 1),
1210
+ )?;
1211
+ c_node.define_method("list_type", method!(CommonmarkerNode::get_list_type, 0))?;
1212
+ c_node.define_method("list_type=", method!(CommonmarkerNode::set_list_type, 1))?;
1213
+ c_node.define_method("list_start", method!(CommonmarkerNode::get_list_start, 0))?;
1214
+ c_node.define_method("list_start=", method!(CommonmarkerNode::set_list_start, 1))?;
1215
+ c_node.define_method("list_tight", method!(CommonmarkerNode::get_list_tight, 0))?;
1216
+ c_node.define_method("list_tight=", method!(CommonmarkerNode::set_list_tight, 1))?;
1217
+ c_node.define_method("fence_info", method!(CommonmarkerNode::get_fence_info, 0))?;
1218
+ c_node.define_method("fence_info=", method!(CommonmarkerNode::set_fence_info, 1))?;
1219
+
1220
+ Ok(())
1221
+ }