deno_doc 0.198.0

doc generation for deno
Documentation
use super::DocNodeWithContext;
use super::GenerateCtx;
use super::RenderContext;
use super::UrlResolveKind;
use crate::html::util::Id;
use crate::html::util::IdBuilder;
use crate::html::util::IdKind;
use crate::js_doc::JsDocTag;
use serde::Deserialize;
use serde::Serialize;
use serde_json::json;
use std::borrow::Cow;

#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SlimKindCtx {
  pub char: char,
  pub kind: Cow<'static, str>,
  pub title: Cow<'static, str>,
}

#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SearchIndexNode {
  pub id: Id,
  pub kind: Vec<SlimKindCtx>,
  pub name: Box<str>,
  pub file: Box<str>,
  pub doc: Box<str>,
  pub url: Box<str>,
  #[serde(skip_serializing_if = "is_empty", default)]
  pub category: Box<str>,
  pub deprecated: bool,
}

fn is_empty(s: &str) -> bool {
  s.is_empty()
}

pub fn doc_nodes_into_search_index_node(
  ctx: &RenderContext,
  symbol: &DocNodeWithContext,
  parent_id: Option<Id>,
) -> Vec<SearchIndexNode> {
  let kinds = symbol
    .get_kind_ctxs()
    .into_iter()
    .map(|kind| SlimKindCtx {
      char: kind.char,
      kind: kind.kind,
      title: kind.title,
    })
    .collect();
  let deprecated = super::util::all_deprecated(
    &symbol.declarations.iter().collect::<Vec<_>>(),
  );

  let doc = super::jsdoc::strip(
    ctx,
    &symbol.declarations[0]
      .js_doc
      .doc
      .clone()
      .unwrap_or_default(),
  )
  .into_boxed_str();

  let name = symbol.get_qualified_name();

  let abs_url = ctx.ctx.resolve_path(
    super::UrlResolveKind::Root,
    super::UrlResolveKind::Symbol {
      file: &symbol.origin,
      symbol: name,
    },
  );

  let category = symbol.declarations[0]
    .js_doc
    .tags
    .iter()
    .find_map(|x| {
      if let JsDocTag::Category { doc } = x {
        Some(doc.clone())
      } else {
        None
      }
    })
    .unwrap_or_default();

  let id = parent_id.unwrap_or_else(|| {
    IdBuilder::new(ctx)
      .kind(IdKind::Namespace)
      .name(name)
      .build_unregistered()
  });

  let mut out = vec![SearchIndexNode {
    id: id.clone(),
    kind: kinds,
    name: html_escape::encode_text(name).into(),
    file: html_escape::encode_double_quoted_attribute(&symbol.origin.path)
      .into(),
    doc,
    url: abs_url.into_boxed_str(),
    category,
    deprecated,
  }];

  if let Some(drilldowns) = symbol.get_drilldown_symbols() {
    out.extend(drilldowns.into_iter().flat_map(|drilldown_node| {
      doc_nodes_into_search_index_node(ctx, &drilldown_node, Some(id.clone()))
    }));
  }

  out
}

pub fn generate_search_index(ctx: &GenerateCtx) -> serde_json::Value {
  let doc_nodes = ctx.doc_nodes.values().flat_map(|nodes| {
    crate::html::partition::flatten_namespace(
      ctx,
      Box::new(nodes.iter().map(Cow::Borrowed)),
    )
  });

  let render_ctx = RenderContext::new(ctx, &[], UrlResolveKind::AllSymbols);

  let mut seen = std::collections::HashSet::new();
  let mut doc_nodes = doc_nodes
    .flat_map(|node| doc_nodes_into_search_index_node(&render_ctx, &node, None))
    .filter(|node| seen.insert((node.name.clone(), node.file.clone())))
    .collect::<Vec<_>>();

  doc_nodes.sort_by(|a, b| a.file.cmp(&b.file));

  let search_index = json!({
    "kind": "search",
    "nodes": doc_nodes
  });

  search_index
}

pub(crate) fn get_search_index_file(
  ctx: &GenerateCtx,
) -> Result<String, anyhow::Error> {
  let search_index = generate_search_index(ctx);
  let search_index_str = serde_json::to_string(&search_index)?;

  let index = format!(
    r#"(function () {{
  window.DENO_DOC_SEARCH_INDEX = {};
}})()"#,
    search_index_str
  );
  Ok(index)
}