use crate::rule::{Fix, LintError, LintResult, LintWarning, Rule, RuleCategory, Severity};
use crate::utils::range_utils::{LineIndex, calculate_match_range};
use regex::Regex;
use std::collections::HashMap;
use std::ops::Range;
use std::sync::LazyLock;
use std::sync::RwLock;
mod md026_config;
use md026_config::{DEFAULT_PUNCTUATION, MD026Config};
static ATX_HEADING_UNIFIED: LazyLock<Regex> =
LazyLock::new(|| Regex::new(r"^( {0,3})(#{1,6})(\s+)(.+?)(\s+#{1,6})?$").unwrap());
static QUICK_PUNCTUATION_CHECK: LazyLock<Regex> =
LazyLock::new(|| Regex::new(&format!(r"[{}]", regex::escape(DEFAULT_PUNCTUATION))).unwrap());
static PUNCTUATION_REGEX_CACHE: LazyLock<RwLock<HashMap<String, Regex>>> =
LazyLock::new(|| RwLock::new(HashMap::new()));
#[derive(Clone, Default)]
pub struct MD026NoTrailingPunctuation {
config: MD026Config,
}
impl MD026NoTrailingPunctuation {
pub fn new(punctuation: Option<String>) -> Self {
Self {
config: MD026Config {
punctuation: punctuation.unwrap_or_else(|| DEFAULT_PUNCTUATION.to_string()),
},
}
}
pub fn from_config_struct(config: MD026Config) -> Self {
Self { config }
}
#[inline]
fn get_punctuation_regex(&self) -> Result<Regex, regex::Error> {
{
let cache = PUNCTUATION_REGEX_CACHE.read().unwrap();
if let Some(cached_regex) = cache.get(&self.config.punctuation) {
return Ok(cached_regex.clone());
}
}
let pattern = format!(r"([{}]+)$", regex::escape(&self.config.punctuation));
let regex = Regex::new(&pattern)?;
{
let mut cache = PUNCTUATION_REGEX_CACHE.write().unwrap();
cache.insert(self.config.punctuation.clone(), regex.clone());
}
Ok(regex)
}
#[inline]
fn has_trailing_punctuation(&self, text: &str, re: &Regex) -> bool {
let trimmed = text.trim();
re.is_match(trimmed)
}
#[inline]
fn get_line_byte_range(&self, content: &str, line_num: usize, line_index: &LineIndex) -> Range<usize> {
let start_pos = line_index.get_line_start_byte(line_num).unwrap_or(content.len());
let line = content.lines().nth(line_num - 1).unwrap_or("");
Range {
start: start_pos,
end: start_pos + line.len(),
}
}
#[inline]
fn remove_trailing_punctuation(&self, text: &str, re: &Regex) -> String {
re.replace_all(text.trim(), "").to_string()
}
#[inline]
fn fix_atx_heading(&self, line: &str, re: &Regex) -> String {
if let Some(captures) = ATX_HEADING_UNIFIED.captures(line) {
let indentation = captures.get(1).unwrap().as_str();
let hashes = captures.get(2).unwrap().as_str();
let space = captures.get(3).unwrap().as_str();
let content = captures.get(4).unwrap().as_str();
let fixed_content = if let Some(id_pos) = content.rfind(" {#") {
let before_id = &content[..id_pos];
let id_part = &content[id_pos..];
let fixed_before = self.remove_trailing_punctuation(before_id, re);
format!("{fixed_before}{id_part}")
} else {
self.remove_trailing_punctuation(content, re)
};
if let Some(trailing) = captures.get(5) {
return format!(
"{}{}{}{}{}",
indentation,
hashes,
space,
fixed_content,
trailing.as_str()
);
}
return format!("{indentation}{hashes}{space}{fixed_content}");
}
line.to_string()
}
#[inline]
fn fix_setext_heading(&self, content_line: &str, re: &Regex) -> String {
let trimmed = content_line.trim_end();
let mut whitespace = "";
if content_line.len() > trimmed.len() {
whitespace = &content_line[trimmed.len()..];
}
format!("{}{}", self.remove_trailing_punctuation(trimmed, re), whitespace)
}
}
impl Rule for MD026NoTrailingPunctuation {
fn name(&self) -> &'static str {
"MD026"
}
fn description(&self) -> &'static str {
"Trailing punctuation in heading"
}
fn category(&self) -> RuleCategory {
RuleCategory::Heading
}
fn should_skip(&self, ctx: &crate::lint_context::LintContext) -> bool {
if !ctx.likely_has_headings() {
return true;
}
let punctuation = &self.config.punctuation;
!punctuation.chars().any(|p| ctx.content.contains(p))
}
fn check(&self, ctx: &crate::lint_context::LintContext) -> LintResult {
let content = ctx.content;
if content.is_empty() {
return Ok(Vec::new());
}
if self.config.punctuation == DEFAULT_PUNCTUATION {
if !QUICK_PUNCTUATION_CHECK.is_match(content) {
return Ok(Vec::new());
}
} else {
let has_custom_punctuation = self.config.punctuation.chars().any(|c| content.contains(c));
if !has_custom_punctuation {
return Ok(Vec::new());
}
}
let has_headings = ctx.lines.iter().any(|line| line.heading.is_some());
if !has_headings {
return Ok(Vec::new());
}
let mut warnings = Vec::new();
let Ok(re) = self.get_punctuation_regex() else {
return Ok(warnings);
};
let line_index = &ctx.line_index;
for (line_num, line_info) in ctx.lines.iter().enumerate() {
if let Some(heading) = &line_info.heading {
if !heading.is_valid {
continue;
}
if line_info.visual_indent >= 4 && matches!(heading.style, crate::lint_context::HeadingStyle::ATX) {
continue;
}
let text_to_check = heading.text.clone();
if self.has_trailing_punctuation(&text_to_check, &re) {
if let Some(punctuation_match) = re.find(&text_to_check) {
let line = line_info.content(ctx.content);
let punctuation_pos_in_text = punctuation_match.start();
let text_pos_in_line = line.find(&heading.text).unwrap_or(heading.content_column);
let punctuation_start_in_line = text_pos_in_line + punctuation_pos_in_text;
let punctuation_len = punctuation_match.len();
let (start_line, start_col, end_line, end_col) = calculate_match_range(
line_num + 1, line,
punctuation_start_in_line,
punctuation_len,
);
let last_char = text_to_check.chars().last().unwrap_or(' ');
warnings.push(LintWarning {
rule_name: Some(self.name().to_string()),
line: start_line,
column: start_col,
end_line,
end_column: end_col,
message: format!("Heading '{text_to_check}' ends with punctuation '{last_char}'"),
severity: Severity::Warning,
fix: Some(Fix {
range: self.get_line_byte_range(content, line_num + 1, line_index),
replacement: if matches!(heading.style, crate::lint_context::HeadingStyle::ATX) {
self.fix_atx_heading(line, &re)
} else {
self.fix_setext_heading(line, &re)
},
}),
});
}
}
}
}
Ok(warnings)
}
fn fix(&self, ctx: &crate::lint_context::LintContext) -> Result<String, LintError> {
if self.should_skip(ctx) {
return Ok(ctx.content.to_string());
}
let warnings = self.check(ctx)?;
if warnings.is_empty() {
return Ok(ctx.content.to_string());
}
let warnings =
crate::utils::fix_utils::filter_warnings_by_inline_config(warnings, ctx.inline_config(), self.name());
crate::utils::fix_utils::apply_warning_fixes(ctx.content, &warnings)
.map_err(crate::rule::LintError::InvalidInput)
}
fn as_any(&self) -> &dyn std::any::Any {
self
}
fn default_config_section(&self) -> Option<(String, toml::Value)> {
let json_value = serde_json::to_value(&self.config).ok()?;
Some((
self.name().to_string(),
crate::rule_config_serde::json_to_toml_value(&json_value)?,
))
}
fn from_config(config: &crate::config::Config) -> Box<dyn Rule>
where
Self: Sized,
{
let rule_config = crate::rule_config_serde::load_rule_config::<MD026Config>(config);
Box::new(Self::from_config_struct(rule_config))
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::lint_context::LintContext;
#[test]
fn test_no_trailing_punctuation() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# This is a heading\n\n## Another heading";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert!(result.is_empty(), "Headings without punctuation should not be flagged");
}
#[test]
fn test_trailing_period() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# This is a heading.\n\n## Another one.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2);
assert_eq!(result[0].line, 1);
assert_eq!(result[0].column, 20);
assert!(result[0].message.contains("ends with punctuation '.'"));
assert_eq!(result[1].line, 3);
assert_eq!(result[1].column, 15);
}
#[test]
fn test_trailing_comma() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Heading,\n## Sub-heading,";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2);
assert!(result[0].message.contains("ends with punctuation ','"));
}
#[test]
fn test_trailing_semicolon() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title;\n## Subtitle;";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2);
assert!(result[0].message.contains("ends with punctuation ';'"));
}
#[test]
fn test_custom_punctuation() {
let rule = MD026NoTrailingPunctuation::new(Some("!".to_string()));
let content = "# Important!\n## Regular heading.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 1, "Only exclamation should be flagged with custom config");
assert_eq!(result[0].line, 1);
assert!(result[0].message.contains("ends with punctuation '!'"));
}
#[test]
fn test_legitimate_question_mark() {
let rule = MD026NoTrailingPunctuation::new(Some(".,;?".to_string()));
let content = "# What is this?\n# This is bad.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2, "Both should be flagged with custom punctuation");
}
#[test]
fn test_question_marks_not_in_default() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# What is Rust?\n# How does it work?\n# Is it fast?";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert!(result.is_empty(), "Question marks are not in default punctuation list");
}
#[test]
fn test_colons_in_default() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# FAQ:\n# API Reference:\n# Step 1:\n# Version 2.0:";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(
result.len(),
4,
"Colons are in default punctuation list and should be flagged"
);
}
#[test]
fn test_fix_atx_headings() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title.\n## Subtitle,\n### Sub-subtitle;";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let fixed = rule.fix(&ctx).unwrap();
assert_eq!(fixed, "# Title\n## Subtitle\n### Sub-subtitle");
}
#[test]
fn test_fix_setext_headings() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "Title.\n======\n\nSubtitle,\n---------";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let fixed = rule.fix(&ctx).unwrap();
assert_eq!(fixed, "Title\n======\n\nSubtitle\n---------");
}
#[test]
fn test_fix_preserves_trailing_hashes() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title. #\n## Subtitle, ##";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let fixed = rule.fix(&ctx).unwrap();
assert_eq!(fixed, "# Title #\n## Subtitle ##");
}
#[test]
fn test_indented_headings() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = " # Title.\n ## Subtitle.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2, "Indented headings (< 4 spaces) should be checked");
}
#[test]
fn test_deeply_indented_ignored() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = " # This is code.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert!(result.is_empty(), "Deeply indented lines (4+ spaces) should be ignored");
}
#[test]
fn test_multiple_punctuation() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title...";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 1);
assert_eq!(result[0].column, 8); }
#[test]
fn test_empty_content() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert!(result.is_empty());
}
#[test]
fn test_no_headings() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "This is just text.\nMore text with punctuation.";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert!(result.is_empty(), "Non-heading lines should not be checked");
}
#[test]
fn test_get_punctuation_regex() {
let rule = MD026NoTrailingPunctuation::new(Some("!?".to_string()));
let regex = rule.get_punctuation_regex().unwrap();
assert!(regex.is_match("text!"));
assert!(regex.is_match("text?"));
assert!(!regex.is_match("text."));
}
#[test]
fn test_regex_caching() {
let rule1 = MD026NoTrailingPunctuation::new(Some("!".to_string()));
let rule2 = MD026NoTrailingPunctuation::new(Some("!".to_string()));
let _regex1 = rule1.get_punctuation_regex().unwrap();
let _regex2 = rule2.get_punctuation_regex().unwrap();
let cache = PUNCTUATION_REGEX_CACHE.read().unwrap();
assert!(cache.contains_key("!"));
}
#[test]
fn test_config_from_toml() {
let mut config = crate::config::Config::default();
let mut rule_config = crate::config::RuleConfig::default();
rule_config
.values
.insert("punctuation".to_string(), toml::Value::String("!?".to_string()));
config.rules.insert("MD026".to_string(), rule_config);
let rule = MD026NoTrailingPunctuation::from_config(&config);
let content = "# Title!\n# Another?";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let result = rule.check(&ctx).unwrap();
assert_eq!(result.len(), 2, "Custom punctuation from config should be used");
}
#[test]
fn test_fix_removes_punctuation() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title. \n## Subtitle, ";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let fixed = rule.fix(&ctx).unwrap();
assert_eq!(fixed, "# Title\n## Subtitle");
}
#[test]
fn test_final_newline_preservation() {
let rule = MD026NoTrailingPunctuation::new(None);
let content = "# Title.\n";
let ctx = LintContext::new(content, crate::config::MarkdownFlavor::Standard, None);
let fixed = rule.fix(&ctx).unwrap();
assert_eq!(fixed, "# Title\n");
let content_no_newline = "# Title.";
let ctx2 = LintContext::new(content_no_newline, crate::config::MarkdownFlavor::Standard, None);
let fixed2 = rule.fix(&ctx2).unwrap();
assert_eq!(fixed2, "# Title");
}
}