/*! # grass An implementation of the Sass specification in pure rust. Spec progress as of 2020-07-24: | Passing | Failing | Total | |---------|---------|-------| | 2935 | 2158 | 5093 | ## Use as library ``` fn main() -> Result<(), Box> { let sass = grass::from_string("a { b { color: &; } }".to_string(), &grass::Options::default())?; assert_eq!(sass, "a b {\n color: a b;\n}\n"); Ok(()) } ``` ## Use as binary ```bash cargo install grass grass input.scss ``` */ #![warn( clippy::all, clippy::restriction, clippy::pedantic, clippy::nursery, clippy::cargo )] #![deny(missing_debug_implementations)] #![allow( // explicit return makes some things look ugly clippy::implicit_return, clippy::use_self, clippy::missing_docs_in_private_items, clippy::unreachable, // this disallows binding as well, e.g. `v => ...` clippy::wildcard_enum_match_arm, clippy::module_name_repetitions, // it is sometimes useful to break up `impl`s clippy::multiple_inherent_impl, // filter isn't fallible clippy::filter_map, clippy::else_if_without_else, clippy::new_ret_no_self, renamed_and_removed_lints, clippy::unknown_clippy_lints, clippy::replace_consts, clippy::single_match, clippy::float_arithmetic, // temporarily allowed while under heavy development. // eventually these allows should be refactored away // to no longer be necessary clippy::as_conversions, clippy::todo, clippy::too_many_lines, clippy::panic, clippy::unwrap_used, clippy::option_unwrap_used, clippy::result_unwrap_used, clippy::cast_possible_truncation, clippy::single_match_else, clippy::indexing_slicing, clippy::redundant_pub_crate, // the api is changing too often to allot this clippy::missing_errors_doc, clippy::integer_arithmetic, clippy::string_add, clippy::get_unwrap, clippy::wrong_self_convention, clippy::items_after_statements, clippy::shadow_reuse, clippy::shadow_unrelated, // this is only available on nightly clippy::unnested_or_patterns, )] #![cfg_attr(feature = "nightly", feature(track_caller))] #![cfg_attr(feature = "profiling", inline(never))] use std::{fs, path::Path}; #[cfg(feature = "wasm")] use wasm_bindgen::prelude::*; pub(crate) use beef::lean::Cow; use codemap::CodeMap; use peekmore::PeekMore; pub use crate::error::{SassError as Error, SassResult as Result}; pub(crate) use crate::token::Token; use crate::{ builtin::modules::Modules, lexer::Lexer, output::Css, parse::{ common::{ContextFlags, NeverEmptyVec}, Parser, }, scope::{Scope, Scopes}, selector::{Extender, Selector}, }; mod args; mod atrule; mod builtin; mod color; mod common; mod error; mod interner; mod lexer; mod output; mod parse; mod scope; mod selector; mod style; mod token; mod unit; mod utils; mod value; #[non_exhaustive] #[derive(Debug)] pub enum OutputStyle { /// The default style, this mode writes each /// selector and declaration on its own line. Expanded, /// Ideal for release builds, this mode removes /// as many extra characters as possible and /// writes the entire stylesheet on a single line. Compressed, } /// Configuration for Sass compilation /// /// The simplest usage is `grass::Options::default()`; /// however, a builder pattern is also exposed to offer /// more control. #[derive(Debug)] pub struct Options<'a> { style: OutputStyle, load_paths: Vec<&'a Path>, allows_charset: bool, unicode_error_messages: bool, quiet: bool, } impl<'a> Default for Options<'a> { #[inline] fn default() -> Self { Self { style: OutputStyle::Expanded, load_paths: Vec::new(), allows_charset: true, unicode_error_messages: true, quiet: false, } } } #[allow(clippy::missing_const_for_fn)] impl<'a> Options<'a> { /// `grass` currently offers 2 different output styles /// /// - `OutputStyle::Expanded` writes each selector and declaration on its own line. /// - `OutputStyle::Compressed` removes as many extra characters as possible /// and writes the entire stylesheet on a single line. /// /// By default, output is expanded. #[must_use] #[inline] pub fn style(mut self, style: OutputStyle) -> Self { self.style = style; self } /// This flag tells Sass not to emit any warnings /// when compiling. By default, Sass emits warnings /// when deprecated features are used or when the /// `@warn` rule is encountered. It also silences the /// `@debug` rule. /// /// By default, this value is `false` and warnings are emitted. #[must_use] #[inline] pub fn quiet(mut self, quiet: bool) -> Self { self.quiet = quiet; self } /// All Sass implementations allow users to provide /// load paths: paths on the filesystem that Sass /// will look in when locating modules. For example, /// if you pass `node_modules/susy/sass` as a load path, /// you can use `@import "susy"` to load `node_modules/susy/sass/susy.scss`. /// /// Imports will always be resolved relative to the current /// file first, though. Load paths will only be used if no /// relative file exists that matches the module's URL. This /// ensures that you can't accidentally mess up your relative /// imports when you add a new library. /// /// This method will append a single path to the list. #[must_use] #[inline] pub fn load_path(mut self, path: &'a Path) -> Self { self.load_paths.push(path); self } /// Append multiple loads paths /// /// Note that this method does *not* remove existing load paths /// /// See [`Options::load_path`](Options::load_path) for more information about load paths #[must_use] #[inline] pub fn load_paths(mut self, paths: &'a [&'a Path]) -> Self { self.load_paths.extend_from_slice(paths); self } /// This flag tells Sass whether to emit a `@charset` /// declaration or a UTF-8 byte-order mark. /// /// By default, Sass will insert either a `@charset` /// declaration (in expanded output mode) or a byte-order /// mark (in compressed output mode) if the stylesheet /// contains any non-ASCII characters. #[must_use] #[inline] pub fn allows_charset(mut self, allows_charset: bool) -> Self { self.allows_charset = allows_charset; self } /// This flag tells Sass only to emit ASCII characters as /// part of error messages. /// /// By default Sass will emit non-ASCII characters for /// these messages. /// /// This flag does not affect the CSS output. #[must_use] #[inline] pub fn unicode_error_messages(mut self, unicode_error_messages: bool) -> Self { self.unicode_error_messages = unicode_error_messages; self } } fn raw_to_parse_error(map: &CodeMap, err: Error, unicode: bool) -> Box { let (message, span) = err.raw(); Box::new(Error::from_loc(message, map.look_up_span(span), unicode)) } /// Compile CSS from a path /// /// ``` /// fn main() -> Result<(), Box> { /// let sass = grass::from_path("input.scss", &grass::Options::default())?; /// Ok(()) /// } /// ``` /// (grass does not currently allow files or paths that are not valid UTF-8) #[cfg_attr(feature = "profiling", inline(never))] #[cfg_attr(not(feature = "profiling"), inline)] #[cfg(not(feature = "wasm"))] pub fn from_path(p: &str, options: &Options) -> Result { let mut map = CodeMap::new(); let file = map.add_file(p.into(), String::from_utf8(fs::read(p)?)?); let empty_span = file.span.subspan(0, 0); let stmts = Parser { toks: &mut Lexer::new(&file) .collect::>() .into_iter() .peekmore(), map: &mut map, path: p.as_ref(), scopes: &mut Scopes::new(), global_scope: &mut Scope::new(), super_selectors: &mut NeverEmptyVec::new(Selector::new(empty_span)), span_before: empty_span, content: &mut Vec::new(), flags: ContextFlags::empty(), at_root: true, at_root_has_selector: false, extender: &mut Extender::new(empty_span), content_scopes: &mut Scopes::new(), options, modules: &mut Modules::default(), } .parse() .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages))?; Css::from_stmts(stmts, false, options.allows_charset) .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages))? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages)) } /// Compile CSS from a string /// /// ``` /// fn main() -> Result<(), Box> { /// let sass = grass::from_string("a { b { color: &; } }".to_string(), &grass::Options::default())?; /// assert_eq!(sass, "a b {\n color: a b;\n}\n"); /// Ok(()) /// } /// ``` #[cfg_attr(feature = "profiling", inline(never))] #[cfg_attr(not(feature = "profiling"), inline)] #[cfg(not(feature = "wasm"))] pub fn from_string(p: String, options: &Options) -> Result { let mut map = CodeMap::new(); let file = map.add_file("stdin".into(), p); let empty_span = file.span.subspan(0, 0); let stmts = Parser { toks: &mut Lexer::new(&file) .collect::>() .into_iter() .peekmore(), map: &mut map, path: Path::new(""), scopes: &mut Scopes::new(), global_scope: &mut Scope::new(), super_selectors: &mut NeverEmptyVec::new(Selector::new(empty_span)), span_before: empty_span, content: &mut Vec::new(), flags: ContextFlags::empty(), at_root: true, at_root_has_selector: false, extender: &mut Extender::new(empty_span), content_scopes: &mut Scopes::new(), options, modules: &mut Modules::default(), } .parse() .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages))?; Css::from_stmts(stmts, false, options.allows_charset) .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages))? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e, options.unicode_error_messages)) } #[cfg(feature = "wasm")] #[wasm_bindgen] pub fn from_string(p: String) -> std::result::Result { let mut map = CodeMap::new(); let file = map.add_file("stdin".into(), p); let empty_span = file.span.subspan(0, 0); let stmts = Parser { toks: &mut Lexer::new(&file) .collect::>() .into_iter() .peekmore(), map: &mut map, path: Path::new(""), scopes: &mut Scopes::new(), global_scope: &mut Scope::new(), super_selectors: &mut NeverEmptyVec::new(Selector::new(empty_span)), span_before: empty_span, content: &mut Vec::new(), flags: ContextFlags::empty(), at_root: true, at_root_has_selector: false, extender: &mut Extender::new(empty_span), content_scopes: &mut Scopes::new(), options: &Options::default(), modules: &mut Modules::default(), } .parse() .map_err(|e| raw_to_parse_error(&map, *e, true).to_string())?; Ok(Css::from_stmts(stmts, false, true) .map_err(|e| raw_to_parse_error(&map, *e, true).to_string())? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e, true).to_string())?) }