/*! # grass An implementation of the Sass specification in pure rust. Spec progress as of 2020-07-04: | Passing | Failing | Total | |---------|---------|-------| | 2880 | 2213 | 5093 | ## Use as library ``` fn main() -> Result<(), Box> { let sass = grass::from_string("a { b { color: &; } }".to_string())?; 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, // 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, )] #![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::{ lexer::Lexer, output::Css, parse::{ common::{ContextFlags, NeverEmptyVec}, Parser, }, scope::Scope, selector::{Extender, Selector}, }; mod args; mod atrule; mod builtin; mod color; mod common; mod error; mod lexer; mod output; mod parse; mod scope; mod selector; mod style; mod token; mod unit; mod utils; mod value; fn raw_to_parse_error(map: &CodeMap, err: Error) -> Box { let (message, span) = err.raw(); Box::new(Error::from_loc(message, map.look_up_span(span))) } /// Write CSS to `buf`, constructed from a path /// /// ``` /// fn main() -> Result<(), Box> { /// let sass = grass::from_path("input.scss")?; /// Ok(()) /// } /// ``` #[cfg_attr(feature = "profiling", inline(never))] #[cfg_attr(not(feature = "profiling"), inline)] #[cfg(not(feature = "wasm"))] pub fn from_path(p: &str) -> 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 NeverEmptyVec::new(Scope::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), } .parse() .map_err(|e| raw_to_parse_error(&map, *e))?; Css::from_stmts(stmts, false) .map_err(|e| raw_to_parse_error(&map, *e))? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e)) } /// Write CSS to `buf`, constructed from a string /// /// ``` /// fn main() -> Result<(), Box> { /// let sass = grass::from_string("a { b { color: &; } }".to_string())?; /// 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) -> 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 NeverEmptyVec::new(Scope::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), } .parse() .map_err(|e| raw_to_parse_error(&map, *e))?; Css::from_stmts(stmts, false) .map_err(|e| raw_to_parse_error(&map, *e))? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e)) } #[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 NeverEmptyVec::new(Scope::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), } .parse() .map_err(|e| raw_to_parse_error(&map, *e).to_string())?; Ok(Css::from_stmts(stmts, false) .map_err(|e| raw_to_parse_error(&map, *e).to_string())? .pretty_print(&map) .map_err(|e| raw_to_parse_error(&map, *e).to_string())?) }