use std::{borrow::Borrow, iter::Iterator, mem}; use num_bigint::BigInt; use num_rational::{BigRational, Rational64}; use num_traits::{pow, One, ToPrimitive}; use codemap::{Span, Spanned}; use peekmore::PeekMore; use crate::{ builtin::GLOBAL_FUNCTIONS, color::{Color, NAMED_COLORS}, common::{Brackets, Identifier, ListSeparator, Op, QuoteKind}, error::SassResult, unit::Unit, utils::{ as_hex, devour_whitespace, eat_number, hex_char_for, is_name, peek_ident_no_interpolation, peek_until_closing_curly_brace, peek_whitespace, read_until_char, read_until_closing_paren, read_until_closing_square_brace, IsWhitespace, }, value::Value, value::{Number, SassMap}, Token, }; use super::Parser; impl<'a> Parser<'a> { pub(super) fn parse_value(&mut self) -> SassResult> { self.whitespace(); let span = match self.toks.peek() { Some(Token { pos, .. }) => *pos, None => return Err(("Expected expression.", self.span_before).into()), }; let mut last_was_whitespace = false; let mut space_separated = Vec::new(); let mut comma_separated = Vec::new(); let mut iter = IntermediateValueIterator::new(self); while let Some(val) = iter.next() { let val = val?; match val.node { IntermediateValue::Value(v) => { last_was_whitespace = false; space_separated.push(v.span(val.span)) } IntermediateValue::Op(op) => { iter.eat_op( Spanned { node: op, span: val.span, }, &mut space_separated, last_was_whitespace, )?; } IntermediateValue::Whitespace => { last_was_whitespace = true; continue; } IntermediateValue::Comma => { last_was_whitespace = false; if space_separated.len() == 1 { comma_separated.push(space_separated.pop().unwrap()); } else { let mut span = space_separated .get(0) .ok_or(("Expected expression.", val.span))? .span; comma_separated.push( Value::List( mem::take(&mut space_separated) .into_iter() .map(|a| { span = span.merge(a.span); a.node }) .collect(), ListSeparator::Space, Brackets::None, ) .span(span), ); } } IntermediateValue::Bracketed(t) => { last_was_whitespace = false; if t.is_empty() { space_separated.push( Value::List(Vec::new(), ListSeparator::Space, Brackets::Bracketed) .span(val.span), ); continue; } space_separated.push(match iter.parser.parse_value_from_vec(t)?.node { Value::List(v, sep, Brackets::None) => { Value::List(v, sep, Brackets::Bracketed).span(val.span) } v => Value::List(vec![v], ListSeparator::Space, Brackets::Bracketed) .span(val.span), }) } IntermediateValue::Paren(t) => { last_was_whitespace = false; space_separated.push(iter.parse_paren(Spanned { node: t, span: val.span, })?); } } } Ok(if !comma_separated.is_empty() { if space_separated.len() == 1 { comma_separated.push(space_separated.pop().unwrap()); } else if !space_separated.is_empty() { comma_separated.push( Value::List( space_separated.into_iter().map(|a| a.node).collect(), ListSeparator::Space, Brackets::None, ) .span(span), ); } Value::List( comma_separated.into_iter().map(|a| a.node).collect(), ListSeparator::Comma, Brackets::None, ) .span(span) } else if space_separated.len() == 1 { space_separated.pop().unwrap() } else { Value::List( space_separated.into_iter().map(|a| a.node).collect(), ListSeparator::Space, Brackets::None, ) .span(span) }) } pub(super) fn parse_value_from_vec(&mut self, toks: Vec) -> SassResult> { Parser { toks: &mut toks.into_iter().peekmore(), map: self.map, path: self.path, scopes: self.scopes, global_scope: self.global_scope, super_selectors: self.super_selectors, span_before: self.span_before, content: self.content.clone(), in_mixin: self.in_mixin, in_function: self.in_function, in_control_flow: self.in_control_flow, at_root: self.at_root, at_root_has_selector: self.at_root_has_selector, extender: self.extender, } .parse_value() } fn parse_ident_value(&mut self) -> SassResult> { let Spanned { node: mut s, span } = self.parse_identifier()?; self.span_before = span; let lower = s.to_ascii_lowercase(); if lower == "progid" && self.toks.peek().is_some() && self.toks.peek().unwrap().kind == ':' { s = lower; self.toks.next(); s.push(':'); s.push_str(&self.eat_progid()?); return Ok(Spanned { node: IntermediateValue::Value(Value::String(s, QuoteKind::None)), span, }); } if let Some(Token { kind: '(', .. }) = self.toks.peek() { self.toks.next(); if lower == "min" { match self.try_parse_min_max("min", true)? { Some((val, len)) => { self.toks.take(len).for_each(drop); return Ok( IntermediateValue::Value(Value::String(val, QuoteKind::None)) .span(span), ); } None => { self.toks.reset_cursor(); } } } else if lower == "max" { match self.try_parse_min_max("max", true)? { Some((val, len)) => { self.toks.take(len).for_each(drop); return Ok( IntermediateValue::Value(Value::String(val, QuoteKind::None)) .span(span), ); } None => { self.toks.reset_cursor(); } } } let as_ident = Identifier::from(&s); let ident_as_string = as_ident.clone().into_inner(); let func = match self.scopes.last().get_fn( Spanned { node: as_ident, span, }, self.global_scope, ) { Ok(f) => f, Err(_) => { if let Some(f) = GLOBAL_FUNCTIONS.get(ident_as_string.as_str()) { return Ok( IntermediateValue::Value(f.0(self.parse_call_args()?, self)?) .span(span), ); } else { // check for special cased CSS functions match lower.as_str() { "calc" | "element" | "expression" => { s = lower; self.eat_calc_args(&mut s)?; } "url" => match self.try_eat_url()? { Some(val) => s = val, None => s.push_str(&self.parse_call_args()?.to_css_string(self)?), }, _ => s.push_str(&self.parse_call_args()?.to_css_string(self)?), } return Ok( IntermediateValue::Value(Value::String(s, QuoteKind::None)).span(span) ); } } }; let call_args = self.parse_call_args()?; return Ok(IntermediateValue::Value(self.eval_function(func, call_args)?).span(span)); } // check for named colors if let Some(c) = NAMED_COLORS.get_by_name(lower.as_str()) { return Ok(IntermediateValue::Value(Value::Color(Box::new(Color::new( c[0], c[1], c[2], c[3], s, )))) .span(span)); } // check for keywords Ok(match lower.as_str() { "true" => IntermediateValue::Value(Value::True), "false" => IntermediateValue::Value(Value::False), "null" => IntermediateValue::Value(Value::Null), "not" => IntermediateValue::Op(Op::Not), "and" => IntermediateValue::Op(Op::And), "or" => IntermediateValue::Op(Op::Or), _ => IntermediateValue::Value(Value::String(s, QuoteKind::None)), } .span(span)) } fn next_is_hypen(&mut self) -> bool { self.toks.peek_forward(1).is_some() && matches!(self.toks.peek().unwrap().kind, '-' | '_' | 'a'..='z' | 'A'..='Z') } fn parse_intermediate_value(&mut self) -> Option>> { let (kind, span) = match self.toks.peek() { Some(v) => (v.kind, v.pos()), None => return None, }; self.span_before = span; if self.whitespace() { return Some(Ok(Spanned { node: IntermediateValue::Whitespace, span, })); } Some(Ok(match kind { _ if kind.is_ascii_alphabetic() || kind == '_' || kind == '\\' || (!kind.is_ascii() && !kind.is_control()) || (kind == '-' && self.next_is_hypen()) => { return Some(self.parse_ident_value()); } '0'..='9' | '.' => { let Spanned { node: val, mut span, } = match eat_number(self.toks) { Ok(v) => v, Err(e) => return Some(Err(e)), }; let unit = if let Some(tok) = self.toks.peek() { let Token { kind, .. } = *tok; match kind { 'a'..='z' | 'A'..='Z' | '_' | '\\' | '\u{7f}'..=std::char::MAX => { let u = match self.parse_identifier_no_interpolation(true) { Ok(v) => v, Err(e) => return Some(Err(e)), }; span = span.merge(u.span); Unit::from(u.node) } '%' => { span = span.merge(self.toks.next().unwrap().pos()); Unit::Percent } _ => Unit::None, } } else { Unit::None }; let n = if val.dec_len == 0 { if val.num.len() <= 18 && val.times_ten.is_empty() { let n = Rational64::new_raw(parse_i64(&val.num), 1); return Some(Ok(IntermediateValue::Value(Value::Dimension( Number::new_machine(n), unit, )) .span(span))); } BigRational::new_raw(val.num.parse::().unwrap(), BigInt::one()) } else { if val.num.len() <= 18 && val.times_ten.is_empty() { let n = Rational64::new(parse_i64(&val.num), pow(10, val.dec_len)); return Some(Ok(IntermediateValue::Value(Value::Dimension( Number::new_machine(n), unit, )) .span(span))); } BigRational::new(val.num.parse().unwrap(), pow(BigInt::from(10), val.dec_len)) }; if val.times_ten.is_empty() { return Some(Ok(IntermediateValue::Value(Value::Dimension( Number::new_big(n), unit, )) .span(span))); } let times_ten = pow( BigInt::from(10), match val .times_ten .parse::() .unwrap() .to_usize() .ok_or(("Exponent too large (expected usize).", span)) { Ok(v) => v, Err(e) => return Some(Err(e.into())), }, ); let times_ten = if val.times_ten_is_postive { BigRational::new_raw(times_ten, BigInt::one()) } else { BigRational::new(BigInt::one(), times_ten) }; IntermediateValue::Value(Value::Dimension(Number::new_big(n * times_ten), unit)) .span(span) } '(' => { let mut span = self.toks.next().unwrap().pos(); let mut inner = match read_until_closing_paren(self.toks) { Ok(v) => v, Err(e) => return Some(Err(e)), }; // todo: the above shouldn't eat the closing paren if let Some(last_tok) = inner.pop() { if last_tok.kind != ')' { return Some(Err(("expected \")\".", span).into())); } span = span.merge(last_tok.pos()); } IntermediateValue::Paren(inner).span(span) } '&' => { let span = self.toks.next().unwrap().pos(); if self.super_selectors.is_empty() && !self.at_root_has_selector && !self.at_root { IntermediateValue::Value(Value::Null).span(span) } else { IntermediateValue::Value(self.super_selectors.last().clone().into_value()) .span(span) } } '#' => { if let Some(Token { kind: '{', pos }) = self.toks.peek_forward(1) { self.span_before = *pos; self.toks.reset_cursor(); return Some(self.parse_ident_value()); } self.toks.reset_cursor(); self.toks.next(); let hex = match self.parse_hex() { Ok(v) => v, Err(e) => return Some(Err(e)), }; IntermediateValue::Value(hex.node).span(hex.span) } q @ '"' | q @ '\'' => { let span_start = self.toks.next().unwrap().pos(); let Spanned { node, span } = match self.parse_quoted_string(q) { Ok(v) => v, Err(e) => return Some(Err(e)), }; IntermediateValue::Value(node).span(span_start.merge(span)) } '[' => { let mut span = self.toks.next().unwrap().pos(); let mut inner = match read_until_closing_square_brace(self.toks) { Ok(v) => v, Err(e) => return Some(Err(e)), }; if let Some(last_tok) = inner.pop() { if last_tok.kind != ']' { return Some(Err(("expected \"]\".", span).into())); } span = span.merge(last_tok.pos()); } IntermediateValue::Bracketed(inner).span(span) } '$' => { self.toks.next(); let val = match self.parse_identifier_no_interpolation(false) { Ok(v) => v, Err(e) => return Some(Err(e)), }; let span = val.span; IntermediateValue::Value( match self.scopes.last().get_var(val, self.global_scope) { Ok(v) => v, Err(e) => return Some(Err(e)), } .node, ) .span(span) } '+' => { let span = self.toks.next().unwrap().pos(); IntermediateValue::Op(Op::Plus).span(span) } '-' => { let span = self.toks.next().unwrap().pos(); IntermediateValue::Op(Op::Minus).span(span) } '*' => { let span = self.toks.next().unwrap().pos(); IntermediateValue::Op(Op::Mul).span(span) } '%' => { let span = self.toks.next().unwrap().pos(); IntermediateValue::Op(Op::Rem).span(span) } ',' => { self.toks.next(); IntermediateValue::Comma.span(span) } q @ '>' | q @ '<' => { let mut span = self.toks.next().unwrap().pos; #[allow(clippy::eval_order_dependence)] IntermediateValue::Op(if let Some(Token { kind: '=', .. }) = self.toks.peek() { span = span.merge(self.toks.next().unwrap().pos); match q { '>' => Op::GreaterThanEqual, '<' => Op::LessThanEqual, _ => unreachable!(), } } else { match q { '>' => Op::GreaterThan, '<' => Op::LessThan, _ => unreachable!(), } }) .span(span) } '=' => { let mut span = self.toks.next().unwrap().pos(); if let Some(Token { kind: '=', pos }) = self.toks.next() { span = span.merge(pos); IntermediateValue::Op(Op::Equal).span(span) } else { return Some(Err(("expected \"=\".", span).into())); } } '!' => { let mut span = self.toks.next().unwrap().pos(); if let Some(Token { kind: '=', .. }) = self.toks.peek() { span = span.merge(self.toks.next().unwrap().pos()); return Some(Ok(IntermediateValue::Op(Op::NotEqual).span(span))); } self.whitespace(); let v = match self.parse_identifier() { Ok(v) => v, Err(e) => return Some(Err(e)), }; span = span.merge(v.span); // TODO: we return `None` when encountering `optional` here as a hack for // supporting `!optional` in `@extend`. In the future, we should have a better // check for `!optional` as this technically allows `!optional` everywhere match v.node.to_ascii_lowercase().as_str() { "important" => IntermediateValue::Value(Value::Important).span(span), "optional" => return None, _ => return Some(Err(("Expected \"important\".", span).into())), } } '/' => { let span = self.toks.next().unwrap().pos(); match self.toks.peek() { Some(Token { kind: '/', .. }) | Some(Token { kind: '*', .. }) => { let span = match self.parse_comment() { Ok(c) => c.span, Err(e) => return Some(Err(e)), }; IntermediateValue::Whitespace.span(span) } Some(..) => IntermediateValue::Op(Op::Div).span(span), None => return Some(Err(("Expected expression.", span).into())), } } ';' | '}' | '{' => return None, ':' | '?' | ')' | '@' | '^' | ']' | '|' => { return Some(Err(("expected \";\".", span).into())) } '\u{0}'..='\u{8}' | '\u{b}'..='\u{1f}' | '\u{7f}'..=std::char::MAX | '`' | '~' => { return Some(Err(("Expected expression.", span).into())) } ' ' | '\n' | '\t' => unreachable!("whitespace is checked prior to this match"), 'A'..='Z' | 'a'..='z' | '_' | '\\' => { unreachable!("these chars are checked in an if stmt") } })) } fn parse_hex(&mut self) -> SassResult> { let mut s = String::with_capacity(7); s.push('#'); let first_char = self .toks .peek() .ok_or(("Expected identifier.", self.span_before))? .kind; let first_is_digit = first_char.is_ascii_digit(); let first_is_hexdigit = first_char.is_ascii_hexdigit(); if first_is_digit { while let Some(c) = self.toks.peek() { if !c.kind.is_ascii_hexdigit() || s.len() == 9 { break; } let tok = self.toks.next().unwrap(); self.span_before = self.span_before.merge(tok.pos()); s.push(tok.kind); } // this branch exists so that we can emit `#` combined with // identifiers. e.g. `#ooobar` should be emitted exactly as written; // that is, `#ooobar`. } else { let ident = self.parse_identifier()?; if first_is_hexdigit && ident.node.chars().all(|c| c.is_ascii_hexdigit()) && matches!(ident.node.len(), 3 | 4 | 6 | 8) { s.push_str(&ident.node); } else { return Ok(Spanned { node: Value::String(format!("#{}", ident.node), QuoteKind::None), span: ident.span, }); } } let v = match u32::from_str_radix(&s[1..], 16) { Ok(a) => a, Err(_) => return Ok(Value::String(s, QuoteKind::None).span(self.span_before)), }; let (red, green, blue, alpha) = match s.len().saturating_sub(1) { 3 => ( (((v & 0x0f00) >> 8) * 0x11) as u8, (((v & 0x00f0) >> 4) * 0x11) as u8, ((v & 0x000f) * 0x11) as u8, 1, ), 4 => ( (((v & 0xf000) >> 12) * 0x11) as u8, (((v & 0x0f00) >> 8) * 0x11) as u8, (((v & 0x00f0) >> 4) * 0x11) as u8, ((v & 0x000f) * 0x11) as u8, ), 6 => ( ((v & 0x00ff_0000) >> 16) as u8, ((v & 0x0000_ff00) >> 8) as u8, (v & 0x0000_00ff) as u8, 1, ), 8 => ( ((v & 0xff00_0000) >> 24) as u8, ((v & 0x00ff_0000) >> 16) as u8, ((v & 0x0000_ff00) >> 8) as u8, (v & 0x0000_00ff) as u8, ), _ => return Err(("Expected hex digit.", self.span_before).into()), }; let color = Color::new(red, green, blue, alpha, s); Ok(Value::Color(Box::new(color)).span(self.span_before)) } fn eat_calc_args(&mut self, buf: &mut String) -> SassResult<()> { buf.reserve(2); buf.push('('); let mut nesting = 0; while let Some(tok) = self.toks.next() { match tok.kind { ' ' | '\t' | '\n' => { self.whitespace(); buf.push(' '); } '#' => { if let Some(Token { kind: '{', pos }) = self.toks.peek() { self.span_before = *pos; self.toks.next(); let interpolation = self.parse_interpolation()?; buf.push_str(&interpolation.node.to_css_string(interpolation.span)?); } else { buf.push('#'); } } '(' => { nesting += 1; buf.push('('); } ')' => { if nesting == 0 { break; } else { nesting -= 1; buf.push(')'); } } c => buf.push(c), } } buf.push(')'); Ok(()) } fn eat_progid(&mut self) -> SassResult { let mut string = String::new(); let mut span = self.toks.peek().unwrap().pos(); while let Some(tok) = self.toks.next() { span = span.merge(tok.pos()); match tok.kind { 'a'..='z' | 'A'..='Z' | '.' => { string.push(tok.kind); } '(' => { self.eat_calc_args(&mut string)?; break; } _ => return Err(("expected \"(\".", span).into()), } } Ok(string) } fn try_eat_url(&mut self) -> SassResult> { let mut buf = String::from("url("); let mut peek_counter = 0; peek_counter += peek_whitespace(self.toks); while let Some(tok) = self.toks.peek() { let kind = tok.kind; self.toks.advance_cursor(); peek_counter += 1; if kind == '!' || kind == '%' || kind == '&' || (kind >= '*' && kind <= '~') || kind as u32 >= 0x0080 { buf.push(kind); } else if kind == '\\' { buf.push_str(&self.peek_escape()?); } else if kind == '#' { if let Some(Token { kind: '{', .. }) = self.toks.peek() { self.toks.advance_cursor(); peek_counter += 1; let (interpolation, count) = self.peek_interpolation()?; peek_counter += count; match interpolation.node { Value::String(ref s, ..) => buf.push_str(s), v => buf.push_str(v.to_css_string(interpolation.span)?.borrow()), }; } else { buf.push('#'); } } else if kind == ')' { buf.push(')'); self.toks.take(peek_counter).for_each(drop); return Ok(Some(buf)); } else if kind.is_whitespace() { peek_counter += peek_whitespace(self.toks); let next = match self.toks.peek() { Some(v) => v, None => break, }; if next.kind == ')' { buf.push(')'); self.toks.take(peek_counter + 1).for_each(drop); return Ok(Some(buf)); } else { break; } } else { break; } } self.toks.reset_cursor(); Ok(None) } fn peek_number(&mut self) -> SassResult> { let mut buf = String::new(); let mut peek_counter = 0; let (num, count) = self.peek_whole_number(); peek_counter += count; buf.push_str(&num); self.toks.advance_cursor(); if let Some(Token { kind: '.', .. }) = self.toks.peek() { self.toks.advance_cursor(); let (num, count) = self.peek_whole_number(); if count == 0 { return Ok(None); } peek_counter += count; buf.push_str(&num); } else { self.toks.move_cursor_back().unwrap(); } let next = match self.toks.peek() { Some(tok) => tok, None => return Ok(Some((buf, peek_counter))), }; match next.kind { 'a'..='z' | 'A'..='Z' | '-' | '_' | '\\' => { let unit = peek_ident_no_interpolation(self.toks, true, self.span_before)?.node; buf.push_str(&unit); peek_counter += unit.chars().count(); } '%' => { self.toks.advance_cursor(); peek_counter += 1; buf.push('%'); } _ => {} } Ok(Some((buf, peek_counter))) } fn peek_whole_number(&mut self) -> (String, usize) { let mut buf = String::new(); let mut peek_counter = 0; while let Some(tok) = self.toks.peek() { if tok.kind.is_ascii_digit() { buf.push(tok.kind); peek_counter += 1; self.toks.advance_cursor(); } else { return (buf, peek_counter); } } (buf, peek_counter) } fn try_parse_min_max( &mut self, fn_name: &str, allow_comma: bool, ) -> SassResult> { let mut buf = if allow_comma { format!("{}(", fn_name) } else { String::new() }; let mut peek_counter = 0; peek_counter += peek_whitespace(self.toks); while let Some(tok) = self.toks.peek() { let kind = tok.kind; peek_counter += 1; match kind { '+' | '-' | '0'..='9' => { self.toks.advance_cursor(); if let Some((number, count)) = self.peek_number()? { buf.push(kind); buf.push_str(&number); peek_counter += count; } else { return Ok(None); } } '#' => { self.toks.advance_cursor(); if let Some(Token { kind: '{', .. }) = self.toks.peek() { self.toks.advance_cursor(); peek_counter += 1; let (interpolation, count) = self.peek_interpolation()?; peek_counter += count; match interpolation.node { Value::String(ref s, ..) => buf.push_str(s), v => buf.push_str(v.to_css_string(interpolation.span)?.borrow()), }; } else { return Ok(None); } } 'c' | 'C' => { if let Some((name, additional_peek_count)) = self.try_parse_min_max_function("calc")? { peek_counter += additional_peek_count; buf.push_str(&name); } else { return Ok(None); } } 'e' | 'E' => { if let Some((name, additional_peek_count)) = self.try_parse_min_max_function("env")? { peek_counter += additional_peek_count; buf.push_str(&name); } else { return Ok(None); } } 'v' | 'V' => { if let Some((name, additional_peek_count)) = self.try_parse_min_max_function("var")? { peek_counter += additional_peek_count; buf.push_str(&name); } else { return Ok(None); } } '(' => { self.toks.advance_cursor(); buf.push('('); if let Some((val, len)) = self.try_parse_min_max(fn_name, false)? { buf.push_str(&val); peek_counter += len; } else { return Ok(None); } } 'm' | 'M' => { self.toks.advance_cursor(); match self.toks.peek() { Some(Token { kind: 'i', .. }) | Some(Token { kind: 'I', .. }) => { self.toks.advance_cursor(); if !matches!(self.toks.peek(), Some(Token { kind: 'n', .. }) | Some(Token { kind: 'N', .. })) { return Ok(None); } buf.push_str("min(") } Some(Token { kind: 'a', .. }) | Some(Token { kind: 'A', .. }) => { self.toks.advance_cursor(); if !matches!(self.toks.peek(), Some(Token { kind: 'x', .. }) | Some(Token { kind: 'X', .. })) { return Ok(None); } buf.push_str("max(") } _ => return Ok(None), } self.toks.advance_cursor(); if !matches!(self.toks.peek(), Some(Token { kind: '(', .. })) { return Ok(None); } peek_counter += 1; if let Some((val, len)) = self.try_parse_min_max(fn_name, false)? { buf.push_str(&val); peek_counter += len; } else { return Ok(None); } } _ => return Ok(None), } peek_counter += peek_whitespace(self.toks); let next = match self.toks.peek() { Some(tok) => tok, None => return Ok(None), }; match next.kind { ')' => { peek_counter += 1; self.toks.advance_cursor(); buf.push(')'); return Ok(Some((buf, peek_counter))); } '+' | '-' | '*' | '/' => { buf.push(' '); buf.push(next.kind); buf.push(' '); self.toks.advance_cursor(); } ',' => { if !allow_comma { return Ok(None); } self.toks.advance_cursor(); buf.push(','); buf.push(' '); } _ => return Ok(None), } peek_counter += peek_whitespace(self.toks); } Ok(Some((buf, peek_counter))) } #[allow(dead_code, unused_mut, unused_variables, unused_assignments)] fn try_parse_min_max_function( &mut self, fn_name: &'static str, ) -> SassResult> { let mut ident = peek_ident_no_interpolation(self.toks, false, self.span_before)?.node; let mut peek_counter = ident.chars().count(); ident.make_ascii_lowercase(); if ident != fn_name { return Ok(None); } if !matches!(self.toks.peek(), Some(Token { kind: '(', .. })) { return Ok(None); } self.toks.advance_cursor(); ident.push('('); peek_counter += 1; todo!("special functions inside `min()` or `max()`") } fn peek_interpolation(&mut self) -> SassResult<(Spanned, usize)> { let vec = peek_until_closing_curly_brace(self.toks)?; let peek_counter = vec.len(); self.toks.advance_cursor(); let val = self.parse_value_from_vec(vec)?; Ok(( Spanned { node: val.node.eval(val.span)?.node.unquote(), span: val.span, }, peek_counter, )) } fn peek_escape(&mut self) -> SassResult { let mut value = 0; let first = match self.toks.peek() { Some(t) => *t, None => return Ok(String::new()), }; let mut span = first.pos; if first.kind == '\n' { return Err(("Expected escape sequence.", first.pos()).into()); } else if first.kind.is_ascii_hexdigit() { for _ in 0..6 { let next = match self.toks.peek() { Some(t) => t, None => break, }; if !next.kind.is_ascii_hexdigit() { break; } value *= 16; value += as_hex(next.kind); span = span.merge(next.pos); self.toks.peek_forward(1); } if self.toks.peek().is_some() && self.toks.peek().unwrap().kind.is_whitespace() { self.toks.peek_forward(1); } } else { value = self.toks.peek_forward(1).unwrap().kind as u32; } let c = std::char::from_u32(value).ok_or(("Invalid escape sequence.", span))?; if is_name(c) { Ok(c.to_string()) } else if value <= 0x1F || value == 0x7F { let mut buf = String::with_capacity(4); buf.push('\\'); if value > 0xF { buf.push(hex_char_for(value >> 4)); } buf.push(hex_char_for(value & 0xF)); buf.push(' '); Ok(buf) } else { Ok(format!("\\{}", c)) } } } struct IntermediateValueIterator<'a, 'b: 'a> { parser: &'a mut Parser<'b>, peek: Option>>, } impl<'a, 'b: 'a> Iterator for IntermediateValueIterator<'a, 'b> { type Item = SassResult>; fn next(&mut self) -> Option { if self.peek.is_some() { self.peek.take() } else { self.parser.parse_intermediate_value() } } } impl<'a, 'b: 'a> IntermediateValueIterator<'a, 'b> { pub fn new(parser: &'a mut Parser<'b>) -> Self { Self { parser, peek: None } } fn peek(&mut self) -> &Option>> { self.peek = self.next(); &self.peek } fn whitespace(&mut self) -> bool { let mut found_whitespace = false; while let Some(w) = self.peek() { if !w.is_whitespace() { break; } found_whitespace = true; self.next(); } found_whitespace } fn eat_op( &mut self, op: Spanned, space_separated: &mut Vec>, last_was_whitespace: bool, ) -> SassResult<()> { match op.node { Op::Not => { self.whitespace(); let right = self.single_value()?; space_separated.push(Spanned { node: Value::UnaryOp(op.node, Box::new(right.node)), span: right.span, }); } Op::Div => { self.whitespace(); let right = self.single_value()?; if let Some(left) = space_separated.pop() { space_separated.push(Spanned { node: Value::BinaryOp(Box::new(left.node), op.node, Box::new(right.node)), span: left.span.merge(right.span), }); } else { self.whitespace(); space_separated.push(Spanned { node: Value::String( format!("/{}", right.node.to_css_string(right.span)?), QuoteKind::None, ), span: op.span.merge(right.span), }); } } Op::Plus => { if let Some(left) = space_separated.pop() { self.whitespace(); let right = self.single_value()?; space_separated.push(Spanned { node: Value::BinaryOp(Box::new(left.node), op.node, Box::new(right.node)), span: left.span.merge(right.span), }); } else { self.whitespace(); let right = self.single_value()?; space_separated.push(Spanned { node: Value::UnaryOp(op.node, Box::new(right.node)), span: right.span, }); } } Op::Minus => { if self.whitespace() || !last_was_whitespace { let right = self.single_value()?; if let Some(left) = space_separated.pop() { space_separated.push(Spanned { node: Value::BinaryOp( Box::new(left.node), op.node, Box::new(right.node), ), span: left.span.merge(right.span), }); } else { space_separated .push(right.map_node(|n| Value::UnaryOp(op.node, Box::new(n)))); } } else { let right = self.single_value()?; space_separated.push(right.map_node(|n| Value::UnaryOp(op.node, Box::new(n)))); } } Op::And => { self.whitespace(); // special case when the value is literally "and" if self.peek().is_none() { space_separated .push(Value::String(op.to_string(), QuoteKind::None).span(op.span)); } else if let Some(left) = space_separated.pop() { self.whitespace(); if left.node.is_true(left.span)? { let right = self.single_value()?; space_separated.push( Value::BinaryOp(Box::new(left.node), op.node, Box::new(right.node)) .span(left.span.merge(right.span)), ); } else { // we explicitly ignore errors here as a workaround for short circuiting while let Some(value) = self.peek() { if let Ok(Spanned { node: IntermediateValue::Comma, .. }) = value { break; } self.next(); } space_separated.push(left); } } else { return Err(("Expected expression.", op.span).into()); } } Op::Or => { self.whitespace(); // special case when the value is literally "or" if self.peek().is_none() { space_separated .push(Value::String(op.to_string(), QuoteKind::None).span(op.span)); } else if let Some(left) = space_separated.pop() { self.whitespace(); if left.node.is_true(left.span)? { // we explicitly ignore errors here as a workaround for short circuiting while let Some(value) = self.peek() { if let Ok(Spanned { node: IntermediateValue::Comma, .. }) = value { break; } self.next(); } space_separated.push(left); } else { let right = self.single_value()?; space_separated.push( Value::BinaryOp(Box::new(left.node), op.node, Box::new(right.node)) .span(left.span.merge(right.span)), ); } } else { return Err(("Expected expression.", op.span).into()); } } _ => { if let Some(left) = space_separated.pop() { self.whitespace(); let right = self.single_value()?; space_separated.push( Value::BinaryOp(Box::new(left.node), op.node, Box::new(right.node)) .span(left.span.merge(right.span)), ); } else { return Err(("Expected expression.", op.span).into()); } } } Ok(()) } fn single_value(&mut self) -> SassResult> { let next = self .next() .ok_or(("Expected expression.", self.parser.span_before))??; Ok(match next.node { IntermediateValue::Value(v) => v.span(next.span), IntermediateValue::Op(op) => match op { Op::Minus => { self.whitespace(); let val = self.single_value()?; Spanned { node: val.node.neg(val.span)?, span: next.span.merge(val.span), } } Op::Not => { self.whitespace(); let val = self.single_value()?; Spanned { node: Value::UnaryOp(Op::Not, Box::new(val.node)), span: next.span.merge(val.span), } } Op::Plus => { self.whitespace(); self.single_value()? } Op::Div => { self.whitespace(); let val = self.single_value()?; Spanned { node: Value::String( format!("/{}", val.node.to_css_string(val.span)?), QuoteKind::None, ), span: next.span.merge(val.span), } } Op::And => Spanned { node: Value::String("and".into(), QuoteKind::None), span: next.span, }, Op::Or => Spanned { node: Value::String("or".into(), QuoteKind::None), span: next.span, }, _ => { return Err(("Expected expression.", next.span).into()); } }, IntermediateValue::Whitespace => unreachable!(), IntermediateValue::Comma => { return Err(("Expected expression.", self.parser.span_before).into()) } IntermediateValue::Bracketed(t) => { let v = self.parser.parse_value_from_vec(t)?; match v.node { Value::List(v, sep, Brackets::None) => Value::List(v, sep, Brackets::Bracketed), v => Value::List(vec![v], ListSeparator::Space, Brackets::Bracketed), } .span(v.span) } IntermediateValue::Paren(t) => { let val = self.parse_paren(Spanned { node: t, span: next.span, })?; Spanned { node: Value::Paren(Box::new(val.node)), span: val.span, } } }) } fn parse_paren(&mut self, t: Spanned>) -> SassResult> { if t.is_empty() { return Ok(Value::List(Vec::new(), ListSeparator::Space, Brackets::None).span(t.span)); } let paren_toks = &mut t.node.into_iter().peekmore(); let mut map = SassMap::new(); let key = self .parser .parse_value_from_vec(read_until_char(paren_toks, ':')?)?; if paren_toks.peek().is_none() { return Ok(Spanned { node: Value::Paren(Box::new(key.node)), span: key.span, }); } let val = self .parser .parse_value_from_vec(read_until_char(paren_toks, ',')?)?; map.insert(key.node, val.node); if paren_toks.peek().is_none() { return Ok(Spanned { node: Value::Map(map), span: key.span.merge(val.span), }); } let mut span = key.span; loop { let key = self .parser .parse_value_from_vec(read_until_char(paren_toks, ':')?)?; devour_whitespace(paren_toks); let val = self .parser .parse_value_from_vec(read_until_char(paren_toks, ',')?)?; span = span.merge(val.span); devour_whitespace(paren_toks); if map.insert(key.node, val.node) { return Err(("Duplicate key.", key.span).into()); } if paren_toks.peek().is_none() { break; } } Ok(Spanned { node: Value::Map(map), span, }) } } impl IsWhitespace for SassResult> { fn is_whitespace(&self) -> bool { match self { Ok(v) => v.node.is_whitespace(), _ => false, } } } #[derive(Clone, Debug, Eq, PartialEq)] enum IntermediateValue { Value(Value), Op(Op), Bracketed(Vec), Paren(Vec), Comma, Whitespace, } impl IntermediateValue { const fn span(self, span: Span) -> Spanned { Spanned { node: self, span } } } impl IsWhitespace for IntermediateValue { fn is_whitespace(&self) -> bool { if self == &IntermediateValue::Whitespace { return true; } false } } fn parse_i64(s: &str) -> i64 { s.as_bytes() .iter() .fold(0, |total, this| total * 10 + i64::from(this - b'0')) }