Fixing Transclusion Link Handling in Eleventy.js

This note documents the process of debugging and fixing an issue with transclusion link handling in the Digital Garden Eleventy setup.

Problem Statement

Transclusion links using the

Transclusion error: File "filename" not found.
syntax were not being processed correctly. Instead of transcluding the content of the referenced file, they were being rendered as regular internal links.

Debugging Steps

  1. The link filter in .eleventy.js was modified to prioritize handling transclusion links before other link types. However, this change alone did not resolve the issue, indicating something else was interfering.

  2. A grep search was performed across the codebase to find other code that might be processing the

    Transclusion error: File "filename" not found.
    syntax. This revealed that the obsidian-image-embeds transform was also capturing these links.

The scenic path to identifying the real problem

initially, I thought that the obsidian-image-embeds transform, intended to handle Obsidian-style image links like

Transclusion error: File "image.png" not found.
, had a regular expression that was too broad:

/!\[\[(.*?\.(?:png|jpg|jpeg|gif|svg|webp))\]\]/gi

The (.*?\. part of this regex was matching any characters up until a dot, even if those characters included square brackets []. This caused it to accidentally match non-image transclusion links like

Transclusion error: File "note.md" not found.
.

Since this transform ran after the link filter, it was overriding the transclusion link handling.

Proposed Solution

To fix this, the regex in obsidian-image-embeds was modified to be more specific:

/!\[\[([^[\]]*\.(?:png|jpg|jpeg|gif|svg|webp))\]\]/gi

The key change is ([^[\]]*\. which ensures it only matches filenames without [] characters. This prevents it from capturing

Transclusion error: File "note.md" not found.
type links.

After deploying the changes, it turned out this wasn't the issue, because

Transclusion error: File "file" not found.
syntax was being rendered as an internal link, and ignoring the ! symbol. Maybe it's the filters and transforms?

adding a explicit transclusion filter

a new transclusion filter was created to handle the

Transclusion error: File "filename" not found.
syntax before the link filter:

eleventyConfig.addFilter("transclusion", function (str) {
  return (
    str &&
    str.replace(/!\[\[(.*?)\]\]/g, (match, filename) => {
      const filePath = findFile(filename);
      if (filePath) {
        const fileContent = fs.readFileSync(filePath, 'utf8');
        
        // Parse the frontmatter and content
        const parsed = matter(fileContent);
        
        // Render only the content part (excluding frontmatter)
        return `<div class="transclusion">${markdownLib.render(parsed.content)}</div>`;
      } else {
        return `<div class="transclusion-error">Transclusion error: File "${filename}" not found.</div>`;
      }
    })
  );
});

The first implementation of this filter exposed the frontmatter of transcluded notes, which was undesirable. This was fixed by using the matter() function to parse the file content and extract only the content part (excluding the frontmatter) before rendering it.

Implementation Details

  1. The transclusion filter looks for all occurrences of
    Transclusion error: File "filename" not found.
    in the content using a regex pattern.
  2. For each match, it tries to find the corresponding file using the findFile function.
  3. If the file is found, it reads the content, parses it using matter() to separate the frontmatter from the actual content, and renders only the content part.
  4. The rendered content is wrapped in a <div class="transclusion"> to allow for styling.
  5. If the file is not found, it returns an error message.

The templates were also modified to include the transclusion filter in the filter chain:

{{ content | hideDataview | taggify | transclusion | link | safe}}

By placing the transclusion filter before the link filter, we ensure that all transclusion links are processed first, then any remaining regular links are handled by the link filter.

Key Takeaways

By understanding how the different parts of the Eleventy config interact and carefully adjusting the regex patterns and filter order, we were able to get the transclusion link handling working as intended, with proper handling of frontmatter.

module.exports = function (eleventyConfig) {
  // ... other configuration ...

  // Transclusion transform
  eleventyConfig.addTransform("transclusion", function (content, outputPath) {
    // ... transclusion transform code ...
  });

  // Link filter
  eleventyConfig.addFilter("link", function (str) {
    // ... link filter code ...
  });

  // ... other transforms and filters ...
};

I made charts. everyone loves charts

flowchart TD
    A[Raw Markdown Content] --> B[hideDataview filter]
    B --> C[taggify filter]
    C --> D[transclusion filter]
    D --> E[link filter]
    E --> F[safe filter]
    F --> G[HTML Template Processing]

    subgraph "Template Filter Chain"
        A & B & C & D & E & F & G
    end

    subgraph "Transclusion Process"
        D1[transclusion filter] --> D2[Find file using findFile]
        D2 --> D3{File found?}
        D3 -->|Yes| D4[Read file content]
        D4 --> D5[Parse with matter]
        D5 --> D6[Extract content only]
        D6 --> D7[Render markdown]
        D7 --> D8[Wrap in div.transclusion]
        D3 -->|No| D9[Return error message]
    end

    subgraph "Transform Chain"
        G --> TR1[Various transforms]
        TR1 --> TR2[dataview-js-links transform]
        TR2 --> TR3[callout-block transform]
        TR3 --> TR4[picture transform]
        TR4 --> TR5[table transform]
        TR5 --> TR6[obsidian-image-embeds transform]
        TR6 --> TR7[htmlMinifier transform]
        TR7 --> H[Final HTML Output]
    end

    style D fill:#f9f,stroke:#333,stroke-width:2px,color:#000
    style E fill:#bbf,stroke:#333,stroke-width:2px,color:#000
    style TR6 fill:#fbb,stroke:#333,stroke-width:2px,color:#000

These charts outline the flow of logic for the transclusion filter, and where it fits in within the broader template filter chain used to generate the final HTML for the site.