sqruff_lib/rules/layout/
lt10.rsuse ahash::AHashMap;
use itertools::chain;
use sqruff_lib_core::dialects::syntax::{SyntaxKind, SyntaxSet};
use sqruff_lib_core::lint_fix::LintFix;
use sqruff_lib_core::parser::segments::base::{ErasedSegment, SegmentBuilder};
use crate::core::config::Value;
use crate::core::rules::base::{Erased, ErasedRule, LintResult, Rule, RuleGroups};
use crate::core::rules::context::RuleContext;
use crate::core::rules::crawlers::{Crawler, SegmentSeekerCrawler};
use crate::utils::functional::context::FunctionalContext;
#[derive(Debug, Default, Clone)]
pub struct RuleLT10;
impl Rule for RuleLT10 {
fn load_from_config(&self, _config: &AHashMap<String, Value>) -> Result<ErasedRule, String> {
Ok(RuleLT10.erased())
}
fn name(&self) -> &'static str {
"layout.select_modifiers"
}
fn description(&self) -> &'static str {
"'SELECT' modifiers (e.g. 'DISTINCT') must be on the same line as 'SELECT'."
}
fn long_description(&self) -> &'static str {
r#"
**Anti-pattern**
In this example, the `DISTINCT` modifier is on the next line after the `SELECT` keyword.
```sql
select
distinct a,
b
from x
```
**Best practice**
Move the `DISTINCT` modifier to the same line as the `SELECT` keyword.
```sql
select distinct
a,
b
from x
```
"#
}
fn groups(&self) -> &'static [RuleGroups] {
&[RuleGroups::All, RuleGroups::Core, RuleGroups::Layout]
}
fn eval(&self, context: RuleContext) -> Vec<LintResult> {
let child_segments = FunctionalContext::new(context.clone())
.segment()
.children(None);
let select_keyword = child_segments.first().unwrap();
let select_clause_modifier_seg = child_segments.find_first(Some(|sp: &ErasedSegment| {
sp.is_type(SyntaxKind::SelectClauseModifier)
}));
if select_clause_modifier_seg.is_empty() {
return Vec::new();
}
let leading_newline_segments = child_segments.select(
Some(|seg: &ErasedSegment| seg.is_type(SyntaxKind::Newline)),
Some(|seg| seg.is_whitespace() || seg.is_meta()),
select_keyword.into(),
None,
);
if leading_newline_segments.is_empty() {
return Vec::new();
}
let select_clause_modifier = select_clause_modifier_seg.first().unwrap();
let leading_whitespace_segments = child_segments.select(
Some(|seg: &ErasedSegment| seg.is_type(SyntaxKind::Whitespace)),
Some(|seg| seg.is_whitespace() || seg.is_meta()),
select_keyword.into(),
None,
);
let trailing_newline_segments = child_segments.select(
Some(|seg: &ErasedSegment| seg.is_type(SyntaxKind::Newline)),
Some(|seg| seg.is_whitespace() || seg.is_meta()),
select_clause_modifier.into(),
None,
);
let mut edit_segments = vec![
SegmentBuilder::whitespace(context.tables.next_id(), " "),
select_clause_modifier.clone(),
];
if trailing_newline_segments.is_empty() {
edit_segments.push(SegmentBuilder::newline(context.tables.next_id(), "\n"));
}
let mut fixes = Vec::new();
fixes.push(LintFix::create_after(
select_keyword.clone(),
edit_segments,
None,
));
if trailing_newline_segments.is_empty() {
fixes.extend(leading_newline_segments.into_iter().map(LintFix::delete));
} else {
let segments = chain(leading_newline_segments, leading_whitespace_segments);
fixes.extend(segments.map(LintFix::delete));
}
let trailing_whitespace_segments = child_segments.select(
Some(|segment: &ErasedSegment| segment.is_whitespace()),
Some(|seg| seg.is_type(SyntaxKind::Whitespace) || seg.is_meta()),
select_clause_modifier.into(),
None,
);
if !trailing_whitespace_segments.is_empty() {
fixes.extend(
trailing_whitespace_segments
.into_iter()
.map(LintFix::delete),
);
}
fixes.push(LintFix::delete(select_clause_modifier.clone()));
vec![LintResult::new(context.segment.into(), fixes, None, None)]
}
fn is_fix_compatible(&self) -> bool {
true
}
fn crawl_behaviour(&self) -> Crawler {
SegmentSeekerCrawler::new(const { SyntaxSet::new(&[SyntaxKind::SelectClause]) }).into()
}
}