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.
- checksums.yaml +4 -4
- data/Cargo.lock +1156 -0
- data/Cargo.toml +7 -0
- data/README.md +237 -172
- data/ext/commonmarker/Cargo.toml +20 -0
- data/ext/commonmarker/extconf.rb +3 -6
- data/ext/commonmarker/src/lib.rs +103 -0
- data/ext/commonmarker/src/node.rs +1221 -0
- data/ext/commonmarker/src/options.rs +220 -0
- data/ext/commonmarker/src/plugins/syntax_highlighting.rs +166 -0
- data/ext/commonmarker/src/plugins.rs +6 -0
- data/ext/commonmarker/src/utils.rs +8 -0
- data/lib/commonmarker/config.rb +92 -40
- data/lib/commonmarker/constants.rb +7 -0
- data/lib/commonmarker/extension.rb +14 -0
- data/lib/commonmarker/node/ast.rb +8 -0
- data/lib/commonmarker/node/inspect.rb +14 -4
- data/lib/commonmarker/node.rb +29 -47
- data/lib/commonmarker/renderer.rb +1 -127
- data/lib/commonmarker/utils.rb +22 -0
- data/lib/commonmarker/version.rb +2 -2
- data/lib/commonmarker.rb +27 -25
- metadata +38 -191
- data/Rakefile +0 -109
- data/bin/commonmarker +0 -118
- data/commonmarker.gemspec +0 -38
- data/ext/commonmarker/arena.c +0 -104
- data/ext/commonmarker/autolink.c +0 -508
- data/ext/commonmarker/autolink.h +0 -8
- data/ext/commonmarker/blocks.c +0 -1622
- data/ext/commonmarker/buffer.c +0 -278
- data/ext/commonmarker/buffer.h +0 -116
- data/ext/commonmarker/case_fold_switch.inc +0 -4327
- data/ext/commonmarker/chunk.h +0 -135
- data/ext/commonmarker/cmark-gfm-core-extensions.h +0 -54
- data/ext/commonmarker/cmark-gfm-extension_api.h +0 -737
- data/ext/commonmarker/cmark-gfm-extensions_export.h +0 -42
- data/ext/commonmarker/cmark-gfm.h +0 -833
- data/ext/commonmarker/cmark-gfm_export.h +0 -42
- data/ext/commonmarker/cmark-gfm_version.h +0 -7
- data/ext/commonmarker/cmark.c +0 -55
- data/ext/commonmarker/cmark_ctype.c +0 -44
- data/ext/commonmarker/cmark_ctype.h +0 -33
- data/ext/commonmarker/commonmark.c +0 -514
- data/ext/commonmarker/commonmarker.c +0 -1308
- data/ext/commonmarker/commonmarker.h +0 -16
- data/ext/commonmarker/config.h +0 -76
- data/ext/commonmarker/core-extensions.c +0 -27
- data/ext/commonmarker/entities.inc +0 -2138
- data/ext/commonmarker/ext_scanners.c +0 -879
- data/ext/commonmarker/ext_scanners.h +0 -24
- data/ext/commonmarker/footnotes.c +0 -63
- data/ext/commonmarker/footnotes.h +0 -27
- data/ext/commonmarker/houdini.h +0 -57
- data/ext/commonmarker/houdini_href_e.c +0 -100
- data/ext/commonmarker/houdini_html_e.c +0 -66
- data/ext/commonmarker/houdini_html_u.c +0 -149
- data/ext/commonmarker/html.c +0 -502
- data/ext/commonmarker/html.h +0 -27
- data/ext/commonmarker/inlines.c +0 -1788
- data/ext/commonmarker/inlines.h +0 -29
- data/ext/commonmarker/iterator.c +0 -159
- data/ext/commonmarker/iterator.h +0 -26
- data/ext/commonmarker/latex.c +0 -468
- data/ext/commonmarker/linked_list.c +0 -37
- data/ext/commonmarker/man.c +0 -274
- data/ext/commonmarker/map.c +0 -129
- data/ext/commonmarker/map.h +0 -44
- data/ext/commonmarker/node.c +0 -1045
- data/ext/commonmarker/node.h +0 -167
- data/ext/commonmarker/parser.h +0 -59
- data/ext/commonmarker/plaintext.c +0 -218
- data/ext/commonmarker/plugin.c +0 -36
- data/ext/commonmarker/plugin.h +0 -34
- data/ext/commonmarker/references.c +0 -43
- data/ext/commonmarker/references.h +0 -26
- data/ext/commonmarker/registry.c +0 -63
- data/ext/commonmarker/registry.h +0 -24
- data/ext/commonmarker/render.c +0 -213
- data/ext/commonmarker/render.h +0 -62
- data/ext/commonmarker/scanners.c +0 -14056
- data/ext/commonmarker/scanners.h +0 -70
- data/ext/commonmarker/scanners.re +0 -341
- data/ext/commonmarker/strikethrough.c +0 -167
- data/ext/commonmarker/strikethrough.h +0 -9
- data/ext/commonmarker/syntax_extension.c +0 -149
- data/ext/commonmarker/syntax_extension.h +0 -34
- data/ext/commonmarker/table.c +0 -917
- data/ext/commonmarker/table.h +0 -12
- data/ext/commonmarker/tagfilter.c +0 -60
- data/ext/commonmarker/tagfilter.h +0 -8
- data/ext/commonmarker/tasklist.c +0 -156
- data/ext/commonmarker/tasklist.h +0 -8
- data/ext/commonmarker/utf8.c +0 -317
- data/ext/commonmarker/utf8.h +0 -35
- data/ext/commonmarker/xml.c +0 -182
- 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
|
+
}
|