AJ0EF - (Mis)Adventures in Amateur Radio

Hello World! This is the most epic subtitle ever.

The new AJ0EF.com!

2024-11-30 4 min read General Informational Joe Fox

I know that this isn’t Amateur Radio, but I thought that I would give a little insight on what it took to put this site together. I’m just trying to get Obsidian, Hugo, GitHub, and Hostinger to work together.

I’m following a tutorial/howto by NetworkChuck.

Now I’m a pretty technically savvy, but this was kind of kicking my butt. Part of me is wondering if it’s because I chose to use WSL for Hugo and my main Windows 11 installation for running Obsidian.

Obsidian on Windows 11

Obsidian on Windows 11

Torturing myself with WSL

Torturing myself with WSL

Using images stored in Obsidian is not a trivial task. It took me some time to get everything worked out the way that I wanted it to work with this theme.

Below is what I have so far - a script to make Obsidian images work with Hugo. I’ve had to modify the original script in NetworkChuck’s video with a little help from ChatGPT. (Thanks NetworkChuck and ChatGPT)!

I’ve included how the script works after the code for those who are curious.

Now, I hopefully I can get the convert_obsidian_to_hugo_centered.py script to work (which later will be renamed to images.py).

import os
import re
import shutil


def convert_file(input_file, output_file, image_src_dir, image_out_dir):
    """
    Converts a single Obsidian Markdown file with Image Captions syntax to Hugo-compatible Markdown.
    Handles cases where captions may or may not be present.

    Args:
        input_file (str): Path to the input Markdown file.
        output_file (str): Path to save the converted Markdown file.
        image_src_dir (str): Path to the source directory containing images.
        image_out_dir (str): Path to the output directory for processed images.
    """
    with open(input_file, 'r', encoding='utf-8') as file:
        content = file.read()

    # Regex to match Obsidian image syntax with or without caption
    pattern = r"!\[\[(.+?)(?:\|(.+?))?\]\]"

    def replacement(match):
        image_file = match.group(1)
        caption = match.group(2)

        # Replace spaces with %20 in the image filename
        new_image_file = image_file.replace(" ", "%20")

        # Define source and destination image paths
        src_image_path = os.path.join(image_src_dir, image_file)
        dest_image_path = os.path.join(image_out_dir, new_image_file)

        # Copy the image to the new location with updated name
        if os.path.exists(src_image_path):
            os.makedirs(os.path.dirname(dest_image_path), exist_ok=True)
            shutil.copy2(src_image_path, dest_image_path)
        else:
            print(f"Warning: Image not found - {src_image_path}")

        # Return Hugo-compatible Markdown syntax
        return (
            f'<div style="text-align: center; font-size: 80%">\n'
            f'  <img src="/images/{new_image_file}" alt="{caption}" title="{caption}">\n'
            f'  <p>{caption}</p>\n'
            f'</div>'
        )

    # Perform the replacement
    converted_content = re.sub(pattern, replacement, content)

    # Save the converted content
    with open(output_file, 'w', encoding='utf-8') as file:
        file.write(converted_content)


def convert_obsidian_to_hugo_recursive(content_dir, image_src_dir, image_out_dir):
    """
    Recursively converts all Obsidian Markdown files in a directory to Hugo-compatible Markdown.
    Handles images with or without captions.

    Args:
        content_dir (str): Path to the content directory.
        image_src_dir (str): Path to the source directory containing images.
        image_out_dir (str): Path to the output directory for processed images.
    """
    for root, _, files in os.walk(content_dir):
        for file_name in files:
            if file_name.endswith('.md'):
                input_file = os.path.join(root, file_name)
                output_file = os.path.join(root, file_name)  # Overwrite the same file
                print(f"Converting: {input_file}")
                convert_file(input_file, output_file, image_src_dir, image_out_dir)
    print("Conversion complete for all files in the directory.")


# Example usage
if __name__ == "__main__":
    # User-defined paths
    content_directory = "content"  # Replace with your content directory path
    image_source_directory = "images"  # Replace with your image source directory path
    image_output_directory = "static/images"  # Replace with your desired image output directory path

    convert_obsidian_to_hugo_recursive(content_directory, image_source_directory, image_output_directory)

Summary of the Script

This Python script converts Obsidian-style Markdown files containing images (with or without captions) into Hugo-compatible Markdown format. It processes Markdown files recursively and handles image files by copying and renaming them with web-compatible filenames (replacing spaces with %20).


Key Features:

  1. Recursive Conversion:

    • Processes all Markdown files in the specified content_directory and its subdirectories.
  2. Image Syntax Transformation:

    • Converts Obsidian-style image syntax to a Hugo-compatible <div> format with inline styles.
  3. Handles Missing Captions:

    • Images without captions are handled by wrapping them in the same <div> structure, with alt and title attributes empty.
  4. Image Processing:

    • Copies images from the image_source_directory to the image_output_directory, renaming files to replace spaces with %20.
  5. Error Handling:

    • If an image file is missing in the source directory, the script outputs a warning.

I also think I need a Markdown tutorial!

It’s also going to take me a while to get the theme down. I’ve chosen to use the Bilberry theme for Hugo.

Here’s the whole video if you’re interested. This contains the rest of the secret sauce! Contains sponsored links.

The YouTube player can not be loaded with disabled JavaScript.
The following video is embedded here:
https://youtube.com/watch?v=dnE7c0ELEH8