Fixing Wiki-Link Handling in Eleventy.js

This document outlines the changes made to .eleventy.js to fix issues with wiki-link handling in the Digital Garden.

Additional Fixes

This document has been updated to include an additional fix for the root file path resolution issue documented here.

Issues Fixed

  1. Wiki-links using permalinks (like Daily Notes) were causing build errors
  2. The system was incorrectly prepending the base path (src/site/notes/) multiple times
  3. Automatic creation of stub files for missing links was causing bloat and confusion
  4. File read operations on directories were causing EISDIR errors
  5. A permalink format issue in one file was causing file/directory conflicts
  6. Wiki-links to root files were incorrectly resolving to /notes/filename instead of using the permalink

Detailed Changes

Issue: Links in the format Title were being treated as file paths with slashes instead of as permalinks.

Before:

function getAnchorAttributes(filePath, linkTitle) {
  let fileName = filePath.replaceAll("&", "&");
  let header = "";
  let headerLinkPath = "";
  
  if (filePath.includes("#")) {
    [fileName, header] = filePath.split("#");
    headerLinkPath = `#${headerToId(header)}`;
  }

  let noteIcon = process.env.NOTE_ICON_DEFAULT;
  const title = linkTitle ? linkTitle : fileName;
  let permalink = `/notes/${slugify(fileName)}`;
  let deadLink = false;
  // ... rest of function

After:

function getAnchorAttributes(filePath, linkTitle) {
  let fileName = filePath.replaceAll("&", "&");
  let header = "";
  let headerLinkPath = "";
  
  // Handle permalinks in wiki-links
  if (filePath.startsWith('/') && filePath.endsWith('/')) {
    // This is a permalink-style link, extract without the slashes
    const permalinkPath = filePath.slice(1, -1);
    let noteIcon = process.env.NOTE_ICON_DEFAULT;
    const title = linkTitle ? linkTitle : permalinkPath;
    
    // Simply use the permalink directly
    return {
      attributes: {
        "class": "internal-link",
        "target": "",
        "data-note-icon": noteIcon,
        "href": `${filePath}`,
      },
      innerHTML: title,
    }
  }
  
  if (filePath.includes("#")) {
    [fileName, header] = filePath.split("#");
    headerLinkPath = `#${headerToId(header)}`;
  }
  // ... rest of function

Explanation: This change adds special handling for wiki-links that use the permalink format (Title). When detected, it immediately returns the correct attributes without attempting to resolve the file path, preventing errors when trying to read directories.

2. Preventing Duplicate Path Prefixes

Issue: The system was incorrectly adding src/site/notes/ to paths that already included this prefix.

Before:

  // If the path already contains a directory structure, treat it differently
  if (fileName.includes('/')) {
    // Try to get or create the file if it doesn't exist
    try {
      fullPath = getOrCreateNoteStub(fileName);
    } catch (e) {
      fullPath = fileName.endsWith(".md") 
        ? `${startPath}${fileName}` 
        : `${startPath}${fileName}.md`;
    }
  }

After:

  // Make sure we don't duplicate the base path
  if (fileName.startsWith('src/site/notes/')) {
    fileName = fileName.replace('src/site/notes/', '');
  }
  
  // If the path already contains a directory structure, treat it differently
  if (fileName.includes('/')) {
    // Don't attempt to create stub files, just construct the path
    fullPath = fileName.endsWith(".md") 
      ? `${startPath}${fileName}` 
      : `${startPath}${fileName}.md`;
  }

Explanation: This change checks if the file path already includes the base path and removes it if found. This prevents file paths like src/site/notes/src/site/notes/file.md which were causing errors.

3. Removing Automatic Stub File Creation

Issue: The system was automatically creating stub files for links that pointed to non-existent files.

Before:

// Helper function to handle missing files by creating stubs
function getOrCreateNoteStub(filePath) {
  const startPath = "./src/site/notes/";
  const fullPath = filePath.endsWith(".md") 
    ? `${startPath}${filePath}` 
    : `${startPath}${filePath}.md`;
  
  try {
    // Check if file exists first
    fs.readFileSync(fullPath, "utf8");
    return fullPath; // Return the path if it exists
  } catch (e) {
    // File doesn't exist, try to create a stub
    try {
      const fileName = filePath.split("/").pop();
      const dirPath = fullPath.substring(0, fullPath.lastIndexOf("/"));
      
      // Create directory if it doesn't exist
      if (!fs.existsSync(dirPath)) {
        fs.mkdirSync(dirPath, { recursive: true });
      }
      
      // Create a minimal stub file
      const stubContent = `---
dg-publish: true
permalink: /${slugify(fileName)}/
---
# ${fileName}

This is a stub page. Content will be added later.
`;
      // Only write the file if it doesn't exist
      if (!fs.existsSync(fullPath)) {
        fs.writeFileSync(fullPath, stubContent);
        console.log(`Created stub file: ${fullPath}`);
      }
      return fullPath;
    } catch (stubError) {
      console.warn(`Error creating stub file: ${stubError.message}`);
      throw e; // Re-throw the original error
    }
  }
}

After:

// Helper function to handle missing files by creating stubs
function getOrCreateNoteStub(filePath) {
  const startPath = "./src/site/notes/";
  const fullPath = filePath.endsWith(".md") 
    ? `${startPath}${filePath}` 
    : `${startPath}${filePath}.md`;
  
  try {
    // Check if file exists first
    fs.readFileSync(fullPath, "utf8");
    return fullPath; // Return the path if it exists
  } catch (e) {
    // File doesn't exist, but don't create a stub - just return the path
    // This disables the automatic stub file creation
    console.warn(`File not found: ${fullPath}`);
    return fullPath;
  }
}

Explanation: This change eliminates the automatic creation of stub files, which was causing confusion and bloat in the repository. Now it simply logs a warning and returns the path.

4. Adding File Existence Check Before Reading

Issue: The system was attempting to read files that didn't exist, causing errors.

Before:

  const file = fs.readFileSync(fullPath, "utf8");
  const frontMatter = matter(file);
  
  // Use permalink from frontmatter if available
  if (frontMatter.data.permalink) {
    permalink = frontMatter.data.permalink;
  }
  // ... more code

After:

  // Only try to read the file if it exists
  if (fs.existsSync(fullPath)) {
    const file = fs.readFileSync(fullPath, "utf8");
    const frontMatter = matter(file);
    
    // Use permalink from frontmatter if available
    if (frontMatter.data.permalink) {
      permalink = frontMatter.data.permalink;
    }
    // ... more code
  } else {
    console.warn(`File not found: ${fullPath}`);
    deadLink = true;
  }

Explanation: This change adds a check to verify that a file exists before attempting to read it, preventing file system errors.

Issue: The file 202503261845.md had a permalink without the standard leading and trailing slashes, causing path conflicts.

Before:

---
dg-publish: false
tags:
  - programming-language
permalink: java
hide: true
aliases:
  - java
---

After:

---
dg-publish: false
tags:
  - programming-language
permalink: /java/
hide: true
aliases:
  - java
---

Explanation: This change standardizes the permalink format to include leading and trailing slashes, consistent with the rest of the site's URL structure. This prevents a conflict where both a file and directory were trying to be created at the same path (dist/java).

6. Fixing Root File Resolution

Issue: Wiki-links to files in the root directory (e.g., Daily Notes Archive) were incorrectly resolving to /notes/daily-notes instead of using the file's permalink (/daily-notes/).

Problem in getAnchorAttributes Function

First, the issue was thought to be in the getAnchorAttributes function, and we added a check for direct file matches in the root directory:

// Check for exact file match in the root directory first
const rootFilePath = `${startPath}${fileName}${fileName.endsWith('.md') ? '' : '.md'}`;
if (fs.existsSync(rootFilePath)) {
  // Direct match in root directory
  fullPath = rootFilePath;
} else if (fileName.includes('/')) {
  // If the path already contains a directory structure...
}

However, this change alone didn't resolve the issue.

The critical issue was in the link filter function, which processes wiki-links. The function was incorrectly finding files during the recursive search process:

Before:

eleventyConfig.addFilter("link", function (str) {
  return (
    str &&
    str.replace(/\[\[(.*?)(?:\|(.*?))?\]\]/g, function (match, fileLink, linkTitle) {
      // ...other checks...
      
      // Try to find the file by alias or by searching through the file structure
      const startPath = "./src/site/notes/";
      let foundFilePath = null;
      
      try {
        const files = fs.readdirSync(startPath, { recursive: true });
        
        // First try to find a direct match by filename
        const exactMatch = files.find(file => {
          return file.toLowerCase() === `${fileLink.toLowerCase()}.md` || 
                 file.toLowerCase() === fileLink.toLowerCase();
        });
        
        if (exactMatch) {
          foundFilePath = exactMatch.replace(/\.md$/, '');
        } else {
          // Try to find by alias...
        }
      } catch (e) {
        // ... error handling
      }
      
      // Use the found path or fall back to the original fileLink
      return getAnchorLink(foundFilePath || fileLink, linkTitle || fileLink);
    })
  );
});

After:

eleventyConfig.addFilter("link", function (str) {
  return (
    str &&
    str.replace(/\[\[(.*?)(?:\|(.*?))?\]\]/g, function (match, fileLink, linkTitle) {
      // ...other checks...
      
      // Try to find the file by alias or by searching through the file structure
      const startPath = "./src/site/notes/";
      let foundFilePath = null;
      
      try {
        // First, check for exact file match in root directory
        const rootFilePath = fileLink.endsWith('.md') 
          ? `${startPath}${fileLink}` 
          : `${startPath}${fileLink}.md`;
          
        if (fs.existsSync(rootFilePath)) {
          // Direct match in root directory, keep file name as is
          foundFilePath = fileLink;
        } else {
          // If not found in root, search recursively
          const files = fs.readdirSync(startPath, { recursive: true });
          
          // Try to find a direct match by filename
          const exactMatch = files.find(file => {
            if (!file.includes('/')) return false; // Skip root files (already checked)
            return file.toLowerCase() === `${fileLink.toLowerCase()}.md` || 
                  file.toLowerCase() === fileLink.toLowerCase();
          });
          
          if (exactMatch) {
            foundFilePath = exactMatch.replace(/\.md$/, '');
          } else {
            // Try to find by alias...
          }
        }
      } catch (e) {
        // ... error handling
      }
      
      // Use the found path or fall back to the original fileLink
      return getAnchorLink(foundFilePath || fileLink, linkTitle || fileLink);
    })
  );
});

Explanation: The key issue was that when searching for files, the recursive directory search would find all files, including those in subdirectories. For a wiki-link like Daily Notes Archive, it could find both "Daily Notes.md" in the root and potentially files within a "Daily Notes" directory.

The fix prioritizes checking for an exact file match in the root directory first. Only if no match is found in the root does it proceed to search subdirectories, and in that case, it explicitly skips root files (which were already checked). This ensures that for a file like "Daily Notes.md" in the root directory, its permalink will be correctly used for wiki-links.

Results

These changes resulted in:

  1. Successful resolution of wiki-links using permalinks
  2. Elimination of EISDIR errors during build
  3. Prevention of stub file creation
  4. Correct handling of file paths without duplication
  5. Consistent permalink formatting
  6. Proper resolution of wiki-links to files in the root directory, using their frontmatter permalinks

The build system now properly handles wiki-links in all formats, including direct permalink references and links to files in the root directory, and doesn't attempt to create unnecessary stub files.