rspack_plugin_html 0.7.11

rspack html plugin
Documentation
use std::str::FromStr;

use rspack_core::rspack_sources::{BoxSource, Source};
use rspack_util::base64;
use serde::Serialize;
use sha2::{Digest, Sha256, Sha384, Sha512};

use crate::tag::{HtmlPluginAttribute, HtmlPluginTag};

#[derive(Serialize, Debug, Clone, Copy)]
#[serde(rename_all = "snake_case")]
pub enum HtmlSriHashFunction {
  Sha256,
  Sha384,
  Sha512,
}

impl FromStr for HtmlSriHashFunction {
  type Err = anyhow::Error;

  fn from_str(s: &str) -> anyhow::Result<HtmlSriHashFunction> {
    if s.eq("sha256") {
      Ok(HtmlSriHashFunction::Sha256)
    } else if s.eq("sha384") {
      Ok(HtmlSriHashFunction::Sha384)
    } else if s.eq("sha512") {
      Ok(HtmlSriHashFunction::Sha512)
    } else {
      Err(anyhow::Error::msg(
        "sri hash function in html config only support 'sha256', 'sha384' or 'sha512'",
      ))
    }
  }
}

pub fn create_digest_from_asset(hash_func: &HtmlSriHashFunction, asset: &BoxSource) -> String {
  let byte_content = asset.buffer();
  match hash_func {
    HtmlSriHashFunction::Sha384 => {
      let mut hasher = Sha384::new();
      hasher.update(byte_content);
      let digest = &hasher.finalize()[..];
      format!("sha384-{}", base64::encode_to_string(digest))
    }
    HtmlSriHashFunction::Sha256 => {
      let mut hasher = Sha256::new();
      hasher.update(byte_content);
      let digest = &hasher.finalize()[..];
      format!("sha256-{}", base64::encode_to_string(digest))
    }
    HtmlSriHashFunction::Sha512 => {
      let mut hasher = Sha512::new();
      hasher.update(byte_content);
      let digest = &hasher.finalize()[..];
      format!("sha512-{}", base64::encode_to_string(digest))
    }
  }
}

pub fn add_sri(tag: &mut HtmlPluginTag, sri: &String) {
  let mut has_crossorigin = false;
  let mut has_integrity = false;
  tag.attributes.iter_mut().for_each(|attribute| {
    if attribute.attr_name.eq("integrity") {
      has_integrity = true;
      attribute.attr_value = Some(sri.to_string());
    }
    if attribute.attr_name.eq("crossorigin") {
      has_crossorigin = true;
      attribute.attr_value = Some("anonymous".to_string());
    }
  });
  if !has_crossorigin {
    tag.attributes.push(HtmlPluginAttribute {
      attr_name: "crossorigin".to_string(),
      attr_value: Some("anonymous".to_string()),
    });
  }
  if !has_integrity {
    tag.attributes.push(HtmlPluginAttribute {
      attr_name: "integrity".to_string(),
      attr_value: Some(sri.to_string()),
    });
  }
}