diff --git a/fuzz/fuzz_targets/all_options.rs b/fuzz/fuzz_targets/all_options.rs index 2af39c81..0684a197 100644 --- a/fuzz/fuzz_targets/all_options.rs +++ b/fuzz/fuzz_targets/all_options.rs @@ -35,13 +35,13 @@ fuzz_target!(|s: &str| { parse.default_info_string = Some("rust".to_string()); parse.relaxed_tasklist_matching = true; parse.relaxed_autolinks = true; - let mut cb = |link_ref: BrokenLinkReference| { + let cb = |link_ref: BrokenLinkReference| { Some(ResolvedReference { url: link_ref.normalized.to_string(), title: link_ref.original.to_string(), }) }; - parse.broken_link_callback = Some(Arc::new(Mutex::new(&mut cb))); + parse.broken_link_callback = Some(Arc::new(cb)); let mut render = RenderOptions::default(); render.hardbreaks = true; diff --git a/src/cm.rs b/src/cm.rs index 9d46d387..10cba543 100644 --- a/src/cm.rs +++ b/src/cm.rs @@ -47,9 +47,9 @@ pub fn format_document_with_plugins<'a>( Ok(()) } -struct CommonMarkFormatter<'a, 'o, 'c> { +struct CommonMarkFormatter<'a, 'o> { node: &'a AstNode<'a>, - options: &'o Options<'c>, + options: &'o Options, v: Vec, prefix: Vec, column: usize, @@ -72,7 +72,7 @@ enum Escaping { Title, } -impl<'a, 'o, 'c> Write for CommonMarkFormatter<'a, 'o, 'c> { +impl<'a, 'o> Write for CommonMarkFormatter<'a, 'o> { fn write(&mut self, buf: &[u8]) -> std::io::Result { self.output(buf, false, Escaping::Literal); Ok(buf.len()) @@ -83,8 +83,8 @@ impl<'a, 'o, 'c> Write for CommonMarkFormatter<'a, 'o, 'c> { } } -impl<'a, 'o, 'c> CommonMarkFormatter<'a, 'o, 'c> { - fn new(node: &'a AstNode<'a>, options: &'o Options<'c>) -> Self { +impl<'a, 'o> CommonMarkFormatter<'a, 'o> { + fn new(node: &'a AstNode<'a>, options: &'o Options) -> Self { CommonMarkFormatter { node, options, diff --git a/src/html.rs b/src/html.rs index b7c1edf5..c82abe6f 100644 --- a/src/html.rs +++ b/src/html.rs @@ -126,9 +126,9 @@ impl Anchorizer { } } -struct HtmlFormatter<'o, 'c> { +struct HtmlFormatter<'o> { output: &'o mut WriteWithLast<'o>, - options: &'o Options<'c>, + options: &'o Options, anchorizer: Anchorizer, footnote_ix: u32, written_footnote_ix: u32, @@ -361,12 +361,8 @@ where Ok(()) } -impl<'o, 'c: 'o> HtmlFormatter<'o, 'c> { - fn new( - options: &'o Options<'c>, - output: &'o mut WriteWithLast<'o>, - plugins: &'o Plugins, - ) -> Self { +impl<'o> HtmlFormatter<'o> { + fn new(options: &'o Options, output: &'o mut WriteWithLast<'o>, plugins: &'o Plugins) -> Self { HtmlFormatter { options, output, diff --git a/src/lib.rs b/src/lib.rs index 16f0abe6..3abdb656 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -101,9 +101,9 @@ pub use xml::format_document_with_plugins as format_xml_with_plugins; /// Legacy naming of [`ExtensionOptions`] pub type ComrakExtensionOptions = ExtensionOptions; /// Legacy naming of [`Options`] -pub type ComrakOptions<'c> = Options<'c>; +pub type ComrakOptions = Options; /// Legacy naming of [`ParseOptions`] -pub type ComrakParseOptions<'c> = ParseOptions<'c>; +pub type ComrakParseOptions = ParseOptions; /// Legacy naming of [`Plugins`] pub type ComrakPlugins<'a> = Plugins<'a>; /// Legacy naming of [`RenderOptions`] diff --git a/src/parser/inlines.rs b/src/parser/inlines.rs index acea0ea0..5601cfb7 100644 --- a/src/parser/inlines.rs +++ b/src/parser/inlines.rs @@ -25,9 +25,9 @@ const MAXBACKTICKS: usize = 80; const MAX_LINK_LABEL_LENGTH: usize = 1000; const MAX_MATH_DOLLARS: usize = 2; -pub struct Subject<'a: 'd, 'r, 'o, 'c, 'd, 'i> { +pub struct Subject<'a: 'd, 'r, 'o, 'd, 'i> { pub arena: &'a Arena>, - options: &'o Options<'c>, + options: &'o Options, pub input: &'i [u8], line: usize, pub pos: usize, @@ -110,10 +110,10 @@ struct WikilinkComponents<'i> { link_label: Option<(&'i [u8], usize, usize)>, } -impl<'a, 'r, 'o, 'c, 'd, 'i> Subject<'a, 'r, 'o, 'c, 'd, 'i> { +impl<'a, 'r, 'o, 'd, 'i> Subject<'a, 'r, 'o, 'd, 'i> { pub fn new( arena: &'a Arena>, - options: &'o Options<'c>, + options: &'o Options, input: &'i [u8], line: usize, refmap: &'r mut RefMap, @@ -1548,7 +1548,7 @@ impl<'a, 'r, 'o, 'c, 'd, 'i> Subject<'a, 'r, 'o, 'c, 'd, 'i> { // Attempt to use the provided broken link callback if a reference cannot be resolved if reff.is_none() { if let Some(callback) = &self.options.parse.broken_link_callback { - reff = callback.lock().unwrap()(BrokenLinkReference { + reff = callback.resolve(BrokenLinkReference { normalized: &lab, original: &unfolded_lab, }); diff --git a/src/parser/mod.rs b/src/parser/mod.rs index 280e10df..5b13fc42 100644 --- a/src/parser/mod.rs +++ b/src/parser/mod.rs @@ -26,7 +26,7 @@ use std::fmt::{self, Debug, Formatter}; use std::mem; use std::panic::RefUnwindSafe; use std::str; -use std::sync::{Arc, Mutex}; +use std::sync::Arc; use typed_arena::Arena; use crate::adapters::HeadingAdapter; @@ -80,16 +80,16 @@ pub fn parse_document<'a>( /// [`ParseOptions::broken_link_callback`]. #[deprecated( since = "0.25.0", - note = "The broken link callback has been moved into ParseOptions<'c>." + note = "The broken link callback has been moved into ParseOptions." )] -pub fn parse_document_with_broken_link_callback<'a, 'c>( +pub fn parse_document_with_broken_link_callback<'a>( arena: &'a Arena>, buffer: &str, - options: &Options<'c>, - callback: Option>, + options: &Options, + callback: Arc, ) -> &'a AstNode<'a> { let mut options_with_callback = options.clone(); - options_with_callback.parse.broken_link_callback = callback.map(|cb| Arc::new(Mutex::new(cb))); + options_with_callback.parse.broken_link_callback = Some(callback); parse_document(arena, buffer, &options_with_callback) } @@ -100,8 +100,26 @@ pub fn parse_document_with_broken_link_callback<'a, 'c>( /// [`BrokenLinkReference`] argument. If a [`ResolvedReference`] is returned, it /// is used as the link; otherwise, no link is made and the reference text is /// preserved in its entirety. -pub type BrokenLinkCallback<'c> = - &'c mut dyn FnMut(BrokenLinkReference) -> Option; +pub trait BrokenLinkCallback: RefUnwindSafe + Send + Sync { + /// Potentially resolve a single broken link reference. + fn resolve(&self, broken_link_reference: BrokenLinkReference) -> Option; +} + +impl Debug for dyn BrokenLinkCallback { + fn fmt(&self, formatter: &mut Formatter<'_>) -> Result<(), fmt::Error> { + formatter.write_str("") + } +} + +impl BrokenLinkCallback for F +where + F: Fn(BrokenLinkReference) -> Option, + F: RefUnwindSafe + Send + Sync, +{ + fn resolve(&self, broken_link_reference: BrokenLinkReference) -> Option { + self(broken_link_reference) + } +} /// Struct to the broken link callback, containing details on the link reference /// which failed to find a match. @@ -116,7 +134,7 @@ pub struct BrokenLinkReference<'l> { pub original: &'l str, } -pub struct Parser<'a, 'o, 'c> { +pub struct Parser<'a, 'o> { arena: &'a Arena>, refmap: RefMap, root: &'a AstNode<'a>, @@ -135,19 +153,18 @@ pub struct Parser<'a, 'o, 'c> { last_line_length: usize, last_buffer_ended_with_cr: bool, total_size: usize, - options: &'o Options<'c>, + options: &'o Options, } #[derive(Default, Debug, Clone)] #[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))] -/// Umbrella options struct. `'c` represents the lifetime of any callback -/// closure options may take. -pub struct Options<'c> { +/// Umbrella options struct. +pub struct Options { /// Enable CommonMark extensions. pub extension: ExtensionOptions, /// Configure parse-time options. - pub parse: ParseOptions<'c>, + pub parse: ParseOptions, /// Configure render-time options. pub render: RenderOptions, @@ -581,10 +598,10 @@ pub struct ExtensionOptions { } #[non_exhaustive] -#[derive(Default, Clone, Builder)] +#[derive(Default, Clone, Debug, Builder)] #[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))] /// Options for parser functions. -pub struct ParseOptions<'c> { +pub struct ParseOptions { /// Punctuation (quotes, full-stops and hyphens) are converted into 'smart' punctuation. /// /// ``` @@ -643,18 +660,18 @@ pub struct ParseOptions<'c> { /// used as the link destination and title if not [`None`]. /// /// ``` - /// # use std::{str, sync::{Arc, Mutex}}; + /// # use std::{str, sync::Arc}; /// # use comrak::{markdown_to_html, BrokenLinkReference, Options, ResolvedReference}; - /// let mut cb = |link_ref: BrokenLinkReference| match link_ref.normalized { + /// let cb = |link_ref: BrokenLinkReference| match link_ref.normalized { /// "foo" => Some(ResolvedReference { /// url: "https://www.rust-lang.org/".to_string(), /// title: "The Rust Language".to_string(), /// }), /// _ => None, /// }; - /// + /// /// let mut options = Options::default(); - /// options.parse.broken_link_callback = Some(Arc::new(Mutex::new(&mut cb))); + /// options.parse.broken_link_callback = Some(Arc::new(cb)); /// /// let output = markdown_to_html( /// "# Cool input!\nWow look at this cool [link][foo]. A [broken link] renders as text.", @@ -666,22 +683,7 @@ pub struct ParseOptions<'c> { /// link. \ /// A [broken link] renders as text.

\n"); #[cfg_attr(feature = "arbitrary", arbitrary(default))] - pub broken_link_callback: Option>>>, -} - -impl<'c> fmt::Debug for ParseOptions<'c> { - fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> { - let mut struct_fmt = f.debug_struct("ParseOptions"); - struct_fmt.field("smart", &self.smart); - struct_fmt.field("default_info_string", &self.default_info_string); - struct_fmt.field("relaxed_tasklist_matching", &self.relaxed_tasklist_matching); - struct_fmt.field("relaxed_autolinks", &self.relaxed_autolinks); - struct_fmt.field( - "broken_link_callback.is_some()", - &self.broken_link_callback.is_some(), - ); - struct_fmt.finish() - } + pub broken_link_callback: Option>, } #[non_exhaustive] @@ -1088,8 +1090,8 @@ struct FootnoteDefinition<'a> { total_references: u32, } -impl<'a, 'o, 'c: 'o> Parser<'a, 'o, 'c> { - fn new(arena: &'a Arena>, root: &'a AstNode<'a>, options: &'o Options<'c>) -> Self { +impl<'a, 'o> Parser<'a, 'o> { + fn new(arena: &'a Arena>, root: &'a AstNode<'a>, options: &'o Options) -> Self { Parser { arena, refmap: RefMap::new(), diff --git a/src/parser/table.rs b/src/parser/table.rs index 4d6f8610..50127085 100644 --- a/src/parser/table.rs +++ b/src/parser/table.rs @@ -13,7 +13,7 @@ use super::inlines::count_newlines; const MAX_AUTOCOMPLETED_CELLS: usize = 500_000; pub fn try_opening_block<'a>( - parser: &mut Parser<'a, '_, '_>, + parser: &mut Parser<'a, '_>, container: &'a AstNode<'a>, line: &[u8], ) -> Option<(&'a AstNode<'a>, bool, bool)> { @@ -30,7 +30,7 @@ pub fn try_opening_block<'a>( } fn try_opening_header<'a>( - parser: &mut Parser<'a, '_, '_>, + parser: &mut Parser<'a, '_>, container: &'a AstNode<'a>, line: &[u8], ) -> Option<(&'a AstNode<'a>, bool, bool)> { @@ -133,7 +133,7 @@ fn try_opening_header<'a>( } fn try_opening_row<'a>( - parser: &mut Parser<'a, '_, '_>, + parser: &mut Parser<'a, '_>, container: &'a AstNode<'a>, alignments: &[TableAlignment], line: &[u8], @@ -280,7 +280,7 @@ fn row(string: &[u8], spoiler: bool) -> Option { } fn try_inserting_table_header_paragraph<'a>( - parser: &mut Parser<'a, '_, '_>, + parser: &mut Parser<'a, '_>, container: &'a AstNode<'a>, paragraph_offset: usize, ) { diff --git a/src/tests/api.rs b/src/tests/api.rs index b06e2a24..07808209 100644 --- a/src/tests/api.rs +++ b/src/tests/api.rs @@ -32,14 +32,15 @@ fn exercise_full_api() { let _: &AstNode = parse_document(&arena, "document", &default_options); // Ensure the closure can modify its context. - let mut blr_ctx_0 = 0; + let blr_ctx_0 = Arc::new(Mutex::new(0)); + let blr_ctx_1 = blr_ctx_0.clone(); #[allow(deprecated)] let _: &AstNode = parse_document_with_broken_link_callback( &arena, "document", &Options::default(), - Some(&mut |blr: BrokenLinkReference| { - blr_ctx_0 += 1; + Arc::new(move |blr: BrokenLinkReference| { + *blr_ctx_1.lock().unwrap() += 1; let _: &str = blr.normalized; let _: &str = blr.original; Some(ResolvedReference { @@ -80,17 +81,16 @@ fn exercise_full_api() { .relaxed_tasklist_matching(false) .relaxed_autolinks(false); - let mut blr_ctx_1 = 0; - let _parse = - parse.broken_link_callback(Arc::new(Mutex::new(&mut |blr: BrokenLinkReference| { - blr_ctx_1 += 1; - let _: &str = blr.normalized; - let _: &str = blr.original; - Some(ResolvedReference { - url: String::new(), - title: String::new(), - }) - }))); + let blr_ctx_1 = blr_ctx_0.clone(); + let _parse = parse.broken_link_callback(Arc::new(move |blr: BrokenLinkReference| { + *blr_ctx_1.lock().unwrap() += 1; + let _: &str = blr.normalized; + let _: &str = blr.original; + Some(ResolvedReference { + url: String::new(), + title: String::new(), + }) + })); let _render = RenderOptions::builder() .hardbreaks(false) diff --git a/src/tests/options.rs b/src/tests/options.rs index 4f07bef2..d3a0ae33 100644 --- a/src/tests/options.rs +++ b/src/tests/options.rs @@ -1,4 +1,4 @@ -use std::sync::{Arc, Mutex}; +use std::sync::Arc; use super::*; @@ -67,32 +67,22 @@ fn smart_chars() { #[test] fn broken_link_callback() { - let arena = Arena::new(); - - let mut cb = |link_ref: BrokenLinkReference| match link_ref.normalized { + let cb = |link_ref: BrokenLinkReference| match link_ref.normalized { "foo" => Some(ResolvedReference { url: "https://www.rust-lang.org/".to_string(), title: "The Rust Language".to_string(), }), _ => None, }; - let options = Options { - parse: ParseOptions::builder() - .broken_link_callback(Arc::new(Mutex::new(&mut cb))) - .build(), - ..Default::default() - }; + let mut options = Options::default(); + options.parse.broken_link_callback = Some(Arc::new(cb)); - let root = parse_document( - &arena, + let output = markdown_to_html( "# Cool input!\nWow look at this cool [link][foo]. A [broken link] renders as text.", &options, ); - let mut output = Vec::new(); - format_html(root, &Options::default(), &mut output).unwrap(); - let output_str = std::str::from_utf8(&output).unwrap(); assert_eq!( - output_str, + output, "

Cool input!

\n

Wow look at this cool \ link. \ A [broken link] renders as text.

\n" diff --git a/src/xml.rs b/src/xml.rs index 1afa5cf8..19721d2e 100644 --- a/src/xml.rs +++ b/src/xml.rs @@ -30,15 +30,15 @@ pub fn format_document_with_plugins<'a>( XmlFormatter::new(options, output, plugins).format(root, false) } -struct XmlFormatter<'o, 'c> { +struct XmlFormatter<'o> { output: &'o mut dyn Write, - options: &'o Options<'c>, + options: &'o Options, _plugins: &'o Plugins<'o>, indent: u32, } -impl<'o, 'c> XmlFormatter<'o, 'c> { - fn new(options: &'o Options<'c>, output: &'o mut dyn Write, plugins: &'o Plugins) -> Self { +impl<'o> XmlFormatter<'o> { + fn new(options: &'o Options, output: &'o mut dyn Write, plugins: &'o Plugins) -> Self { XmlFormatter { options, output,