use std::path::PathBuf;
use std::sync::Arc;
use anyhow::{anyhow, Context, Result};
use async_std::fs;
use async_std::task::{spawn, spawn_blocking, JoinHandle};
use nipper::Document;
use super::{AssetFile, HashedFileOutput, LinkAttrs, TrunkLinkPipelineOutput};
use super::{ATTR_HREF, ATTR_INLINE};
use crate::config::RtcBuild;
pub struct Sass {
id: usize,
cfg: Arc<RtcBuild>,
asset: AssetFile,
use_inline: bool,
}
impl Sass {
pub const TYPE_SASS: &'static str = "sass";
pub const TYPE_SCSS: &'static str = "scss";
pub async fn new(cfg: Arc<RtcBuild>, html_dir: Arc<PathBuf>, attrs: LinkAttrs, id: usize) -> Result<Self> {
let href_attr = attrs
.get(ATTR_HREF)
.context(r#"required attr `href` missing for <link data-trunk rel="sass|scss" .../> element"#)?;
let mut path = PathBuf::new();
path.extend(href_attr.split('/'));
let asset = AssetFile::new(&html_dir, path).await?;
let use_inline = attrs.get(ATTR_INLINE).is_some();
Ok(Self { id, cfg, asset, use_inline })
}
#[tracing::instrument(level = "trace", skip(self))]
pub fn spawn(self) -> JoinHandle<Result<TrunkLinkPipelineOutput>> {
spawn(self.run())
}
#[tracing::instrument(level = "trace", skip(self))]
async fn run(self) -> Result<TrunkLinkPipelineOutput> {
let rel_path = crate::common::strip_prefix(&self.asset.path);
tracing::info!(path = ?rel_path, "compiling sass/scss");
let path_str = self.asset.path.to_string_lossy().to_string();
let mut opts = sass_rs::Options::default();
if self.cfg.release {
opts.output_style = sass_rs::OutputStyle::Compressed;
}
let css = spawn_blocking(move || sass_rs::compile_file(&path_str, opts)).await.map_err(|err| {
eprintln!("{}", err);
anyhow!("error compiling sass for {:?}", &self.asset.path)
})?;
let css_ref = if self.use_inline {
CssRef::Inline(css)
} else {
let hash = seahash::hash(css.as_bytes());
let file_name = format!("{}-{:x}.css", &self.asset.file_stem.to_string_lossy(), hash);
let file_path = self.cfg.staging_dist.join(&file_name);
fs::write(&file_path, css).await.context("error writing SASS pipeline output")?;
CssRef::File(HashedFileOutput { hash, file_path, file_name })
};
tracing::info!(path = ?rel_path, "finished compiling sass/scss");
Ok(TrunkLinkPipelineOutput::Sass(SassOutput {
cfg: self.cfg.clone(),
id: self.id,
css_ref,
}))
}
}
pub struct SassOutput {
pub cfg: Arc<RtcBuild>,
pub id: usize,
pub css_ref: CssRef,
}
pub enum CssRef {
Inline(String),
File(HashedFileOutput),
}
impl SassOutput {
pub async fn finalize(self, dom: &mut Document) -> Result<()> {
let html = match self.css_ref {
CssRef::Inline(css) => format!(r#"<style type="text/css">{}</style>"#, css),
CssRef::File(file) => {
format!(
r#"<link rel="stylesheet" href="{base}{file}"/>"#,
base = &self.cfg.public_url,
file = file.file_name
)
}
};
dom.select(&super::trunk_id_selector(self.id)).replace_with_html(html);
Ok(())
}
}