ast-grep-config 0.43.0

Search and Rewrite code at large scale using precise AST pattern
Documentation
use ast_grep_core::{Doc, Node, NodeMatch};
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use std::{collections::HashMap, ops::Range};

#[derive(Serialize, Deserialize, Clone, JsonSchema, PartialEq, Eq, Debug)]
#[serde(rename_all = "camelCase")]
pub enum LabelStyle {
  /// Labels that describe the primary cause of a diagnostic.
  Primary,
  /// Labels that provide additional context for a diagnostic.
  Secondary,
}

#[derive(Serialize, Deserialize, Clone, JsonSchema)]
pub struct LabelConfig {
  pub style: LabelStyle,
  pub message: Option<String>,
}

/// A label is a way to mark a specific part of the code with a styled message.
/// It is used to provide diagnostic information in LSP or CLI.
/// 'r represents a lifetime for the message string from `rule`.
/// 't represents a lifetime for the node from a ast `tree`.
pub struct Label<'r, 't, D: Doc> {
  pub style: LabelStyle,
  pub message: Option<&'r str>,
  pub start_node: Node<'t, D>,
  pub end_node: Node<'t, D>,
}

impl<'t, D: Doc> Label<'_, 't, D> {
  fn primary(n: &Node<'t, D>) -> Self {
    Self {
      style: LabelStyle::Primary,
      start_node: n.clone(),
      end_node: n.clone(),
      message: None,
    }
  }
  fn secondary(n: &Node<'t, D>) -> Self {
    Self {
      style: LabelStyle::Secondary,
      start_node: n.clone(),
      end_node: n.clone(),
      message: None,
    }
  }

  pub fn range(&self) -> Range<usize> {
    let start = self.start_node.range().start;
    let end = self.end_node.range().end;
    start..end
  }
}

pub fn get_labels_from_config<'r, 't, D: Doc>(
  config: &'r HashMap<String, LabelConfig>,
  node_match: &NodeMatch<'t, D>,
) -> Vec<Label<'r, 't, D>> {
  let env = node_match.get_env();
  config
    .iter()
    .filter_map(|(var, conf)| {
      let (start, end) = if let Some(n) = env.get_match(var) {
        (n.clone(), n.clone())
      } else {
        let ns = env.get_multiple_matches(var);
        let start = ns.first()?.clone();
        let end = ns.last()?.clone();
        (start, end)
      };
      Some(Label {
        style: conf.style.clone(),
        message: conf.message.as_deref(),
        start_node: start,
        end_node: end,
      })
    })
    .collect()
}

pub fn get_default_labels<'t, D: Doc>(n: &NodeMatch<'t, D>) -> Vec<Label<'static, 't, D>> {
  let mut ret = vec![Label::primary(n)];
  if let Some(secondary) = n.get_env().get_labels("secondary") {
    ret.extend(secondary.iter().map(Label::secondary));
  }
  ret
}

#[cfg(test)]
mod tests {
  use super::*;
  use crate::test::TypeScript;
  use ast_grep_core::matcher::Pattern;
  use ast_grep_core::tree_sitter::LanguageExt;
  use ast_grep_core::tree_sitter::StrDoc;

  #[test]
  fn test_label_primary_secondary() {
    let doc = TypeScript::Tsx.ast_grep("let a = 1;");
    let root = doc.root();
    let label = Label::primary(&root);
    assert_eq!(label.style, LabelStyle::Primary);
    assert_eq!(label.range(), root.range());
    let label2 = Label::<'_, '_, StrDoc<TypeScript>>::secondary(&root);
    assert_eq!(label2.style, LabelStyle::Secondary);
  }

  #[test]
  fn test_get_labels_from_config_single() {
    let doc = TypeScript::Tsx.ast_grep("let foo = 42;");
    let pattern = Pattern::try_new("let $A = $B;", TypeScript::Tsx).unwrap();
    let m = doc.root().find(pattern).unwrap();
    let mut config = std::collections::HashMap::new();
    config.insert(
      "A".to_string(),
      LabelConfig {
        style: LabelStyle::Primary,
        message: Some("var label".to_string()),
      },
    );
    let labels = get_labels_from_config(&config, &m);
    assert_eq!(labels.len(), 1);
    assert_eq!(labels[0].style, LabelStyle::Primary);
    assert_eq!(labels[0].message, Some("var label"));
  }

  #[test]
  fn test_get_labels_from_config_multiple() {
    let doc = TypeScript::Tsx.ast_grep("let foo = 42, bar = 99;");
    let pattern = Pattern::try_new("let $A = $B, $C = $D;", TypeScript::Tsx).unwrap();
    let m = doc.root().find(pattern).unwrap();
    let mut config = std::collections::HashMap::new();
    config.insert(
      "A".to_string(),
      LabelConfig {
        style: LabelStyle::Secondary,
        message: None,
      },
    );
    let labels = get_labels_from_config(&config, &m);
    assert_eq!(labels.len(), 1);
    assert_eq!(labels[0].style, LabelStyle::Secondary);
  }

  #[test]
  fn test_get_default_labels() {
    let doc = TypeScript::Tsx.ast_grep("let foo = 42;");
    let pattern = Pattern::try_new("let $A = $B;", TypeScript::Tsx).unwrap();
    let m = doc.root().find(pattern).unwrap();
    let labels = get_default_labels(&m);
    assert!(!labels.is_empty());
    assert_eq!(labels[0].style, LabelStyle::Primary);
  }
}