The new AJ0EF.com!
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

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:
-
Recursive Conversion:
- Processes all Markdown files in the specified
content_directory
and its subdirectories.
- Processes all Markdown files in the specified
-
Image Syntax Transformation:
- Converts Obsidian-style image syntax to a Hugo-compatible
<div>
format with inline styles.
- Converts Obsidian-style image syntax to a Hugo-compatible
-
Handles Missing Captions:
- Images without captions are handled by wrapping them in the same
<div>
structure, withalt
andtitle
attributes empty.
- Images without captions are handled by wrapping them in the same
-
Image Processing:
- Copies images from the
image_source_directory
to theimage_output_directory
, renaming files to replace spaces with%20
.
- Copies images from the
-
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.