deny functions with reserved names or in control flow
This commit is contained in:
parent
ac2d15b776
commit
aea7c9c408
@ -14,18 +14,48 @@ use crate::{
|
|||||||
|
|
||||||
use super::{NeverEmptyVec, Parser, Stmt};
|
use super::{NeverEmptyVec, Parser, Stmt};
|
||||||
|
|
||||||
impl<'a> Parser<'a> {
|
/// Names that functions are not allowed to have
|
||||||
pub(super) fn parse_function(&mut self) -> SassResult<()> {
|
const FORBIDDEN_IDENTIFIERS: [&str; 7] =
|
||||||
if self.in_mixin {
|
["calc", "element", "expression", "url", "and", "or", "not"];
|
||||||
return Err((
|
|
||||||
"Mixins may not contain function declarations.",
|
fn unvendor<'a>(name: &'a str) -> &'a str {
|
||||||
self.span_before,
|
let mut chars = name.chars();
|
||||||
)
|
if !matches!(chars.next(), Some('-')) {
|
||||||
.into());
|
return name;
|
||||||
|
}
|
||||||
|
if matches!(chars.next(), Some('-')) {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
if name.chars().count() < 2 {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
let mut pos = 2;
|
||||||
|
for c in chars {
|
||||||
|
if c == '-' {
|
||||||
|
return &name[pos..];
|
||||||
|
}
|
||||||
|
pos += 1;
|
||||||
|
}
|
||||||
|
name
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<'a> Parser<'a> {
|
||||||
|
pub(super) fn parse_function(&mut self) -> SassResult<()> {
|
||||||
self.whitespace_or_comment();
|
self.whitespace_or_comment();
|
||||||
let Spanned { node: name, span } = self.parse_identifier()?;
|
let Spanned { node: name, span } = self.parse_identifier()?;
|
||||||
|
|
||||||
|
if self.in_mixin {
|
||||||
|
return Err(("Mixins may not contain function declarations.", span).into());
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.in_control_flow {
|
||||||
|
return Err(("Functions may not be declared in control directives.", span).into());
|
||||||
|
}
|
||||||
|
|
||||||
|
if FORBIDDEN_IDENTIFIERS.contains(&unvendor(&name)) {
|
||||||
|
return Err(("Invalid function name.", span).into());
|
||||||
|
}
|
||||||
|
|
||||||
self.whitespace_or_comment();
|
self.whitespace_or_comment();
|
||||||
let args = match self.toks.next() {
|
let args = match self.toks.next() {
|
||||||
Some(Token { kind: '(', .. }) => self.parse_func_args()?,
|
Some(Token { kind: '(', .. }) => self.parse_func_args()?,
|
||||||
|
@ -121,3 +121,12 @@ test!(
|
|||||||
}",
|
}",
|
||||||
"a {\n color: \"success!\";\n}\n"
|
"a {\n color: \"success!\";\n}\n"
|
||||||
);
|
);
|
||||||
|
error!(
|
||||||
|
denies_function_declaration_in_control_flow,
|
||||||
|
"@if true {\n @function foo() {}\n}\n",
|
||||||
|
"Error: Functions may not be declared in control directives."
|
||||||
|
);
|
||||||
|
error!(
|
||||||
|
denies_function_declaration_with_reserved_name,
|
||||||
|
"@function url() {}", "Error: Invalid function name."
|
||||||
|
);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user