use super::{
data_target_path, AssetFile, AttrWriter, Attrs, TrunkAssetPipelineOutput, ATTR_CONFIG,
ATTR_HREF, ATTR_INLINE, ATTR_NO_MINIFY,
};
use crate::{
common::{self, dist_relative, html_rewrite::Document, nonce, target_path},
config::rt::RtcBuild,
processing::integrity::{IntegrityType, OutputDigest},
tools::{self, Application},
};
use anyhow::{Context, Result};
use std::{path::PathBuf, sync::Arc};
use tokio::{fs, task::JoinHandle};
pub struct TailwindCssExtra {
id: usize,
cfg: Arc<RtcBuild>,
asset: AssetFile,
use_inline: bool,
attrs: Attrs,
integrity: IntegrityType,
no_minify: bool,
target_path: Option<PathBuf>,
tailwind_config: Option<String>,
}
impl TailwindCssExtra {
pub const TYPE_TAILWIND_CSS_EXTRA: &'static str = "tailwind-css-extra";
pub async fn new(
cfg: Arc<RtcBuild>,
html_dir: Arc<PathBuf>,
attrs: Attrs,
id: usize,
) -> Result<Self> {
let href_attr = attrs.get(ATTR_HREF).context(
r#"required attr `href` missing for <link data-trunk rel="tailwind-css-extra" .../> element"#,
)?;
let tailwind_config = attrs.get(ATTR_CONFIG).map(|attr| &attr.value).cloned();
let mut path = PathBuf::new();
path.extend(href_attr.split('/'));
let asset = AssetFile::new(&html_dir, path).await?;
let use_inline = attrs.contains_key(ATTR_INLINE);
let integrity = IntegrityType::from_attrs(&attrs, &cfg)?;
let no_minify = attrs.contains_key(ATTR_NO_MINIFY);
let target_path = data_target_path(&attrs)?;
Ok(Self {
id,
cfg,
asset,
use_inline,
integrity,
attrs,
no_minify,
target_path,
tailwind_config,
})
}
#[tracing::instrument(level = "trace", skip(self))]
pub fn spawn(self) -> JoinHandle<Result<TrunkAssetPipelineOutput>> {
tokio::spawn(self.run())
}
#[tracing::instrument(level = "trace", skip(self))]
async fn run(self) -> Result<TrunkAssetPipelineOutput> {
let version = self.cfg.tools.tailwindcss.as_deref();
let tailwind = tools::get(
Application::TailwindCssExtra,
version,
self.cfg.offline,
&self.cfg.client_options(),
)
.await?;
let path_str = dunce::simplified(&self.asset.path).display().to_string();
let file_name = format!("{}.css", &self.asset.file_stem.to_string_lossy());
let file_path = dunce::simplified(&self.cfg.staging_dist.join(&file_name))
.display()
.to_string();
let mut args = vec!["--input", &path_str, "--output", &file_path];
if let Some(tailwind_config) = self.tailwind_config.as_ref() {
args.push("--config");
args.push(tailwind_config);
}
if self.cfg.minify_asset(self.no_minify) {
args.push("--minify");
}
let rel_path = common::strip_prefix(&self.asset.path);
tracing::debug!(path = ?rel_path, "compiling tailwind css");
common::run_command(
Application::TailwindCssExtra.name(),
&tailwind,
&args,
&self.cfg.core.working_directory,
)
.await?;
let css = fs::read_to_string(&file_path).await?;
fs::remove_file(&file_path).await?;
let css_ref = if self.use_inline {
CssExtraRef::Inline(css)
} else {
let hash = seahash::hash(css.as_bytes());
let file_name = if self.cfg.filehash {
format!("{}-{:x}.css", &self.asset.file_stem.to_string_lossy(), hash)
} else {
file_name
};
let result_dir =
target_path(&self.cfg.staging_dist, self.target_path.as_deref(), None).await?;
let file_path = result_dir.join(&file_name);
let file_href = dist_relative(&self.cfg.staging_dist, &file_path)?;
let integrity = OutputDigest::generate_from(self.integrity, css.as_bytes());
fs::write(&file_path, css)
.await
.context("error writing tailwind css pipeline output")?;
CssExtraRef::File(file_href, integrity)
};
tracing::debug!(path = ?rel_path, "finished compiling tailwind css");
Ok(TrunkAssetPipelineOutput::TailwindCssExtra(
TailwindCssExtraOutput {
cfg: self.cfg.clone(),
id: self.id,
css_ref,
attrs: self.attrs,
},
))
}
}
pub struct TailwindCssExtraOutput {
pub cfg: Arc<RtcBuild>,
pub id: usize,
pub css_ref: CssExtraRef,
pub attrs: Attrs,
}
pub enum CssExtraRef {
Inline(String),
File(String, OutputDigest),
}
impl TailwindCssExtraOutput {
pub async fn finalize(self, dom: &mut Document) -> Result<()> {
let html = match self.css_ref {
CssExtraRef::Inline(css) => format!(
r#"<style {attrs} nonce="{}">{css}</style>"#,
nonce()?,
attrs = AttrWriter::new(&self.attrs, AttrWriter::EXCLUDE_CSS_INLINE)
),
CssExtraRef::File(file, integrity) => {
let mut attrs = self.attrs.clone();
integrity.insert_into(&mut attrs);
format!(
r#"<link rel="stylesheet" href="{base}{file}"{attrs}/>"#,
base = &self.cfg.public_url,
attrs = AttrWriter::new(&attrs, AttrWriter::EXCLUDE_CSS_LINK)
)
}
};
dom.replace_with_html(&super::trunk_id_selector(self.id), &html)
}
}