Setting up the Blog
Tasks:
-create folders in Obsidian for posts and for where the website will be located
-install go(google golang)
-install hugo
-download and install ‘Terminal’ theme for hugo and update hugo config
REF: https://github.com/panr/hugo-theme-terminal
-create ‘images’ subfolder of static folder in hugo website to sync posts images to
-install latest Python
-copy python script and modify for my settings
Posts syncing command:
robocopy “C:\Users\username\iCloudDrive\iCloudmdobsidian\master\posts” “C:\Users\username\iCloudDrive\iCloudmdobsidian\master\MyWebsite\jblog\content\posts” /mir
Hugo Local Server Test Command
hugo server -t terminal
Hugo was not recognizing the line breaks from the Obsidian posts.
Hugo uses a Markdown renderer (e.g., Goldmark by default). To make Hugo recognize single line breaks as valid line breaks, you need to enable the hardWraps
option.
Steps:#
- Open the hugo.toml file
- Add or update the following configuration:
- For TOML
[markup]
[markup.goldmark]
[markup.goldmark.renderer]
hardWraps = true
Hugo was not recognizing the YAML front matter that I put into the obsidian first post, so I had to add this to the hugo.toml config file:
[frontmatter]
default = "yaml"
created subfolder ‘images’ in the hugo site under subfolder ‘static’. This is where the script will copy the related images the script finds for each blog post in obsidian
Obsidian install was set to store all post attachments in the master folder which is a mess. So I had to create a new folder, tell Obsidian to use this instead and then move all attachments to that folder:
Steps to Set a Custom Folder for Attachments:#
-
Create the Folder Manually:
- Open your file explorer (or use Obsidian’s file explorer) and manually create the folder you want (e.g.,
all_attachments
).
- Open your file explorer (or use Obsidian’s file explorer) and manually create the folder you want (e.g.,
-
Set the Folder Path:
- Go back to Obsidian settings (
Settings > Files & Links
). - Click on Choose folder… again.
- Instead of naming it manually, navigate to the folder you just created (
all_attachments
), and select it.
- Go back to Obsidian settings (
I had python installed to the user profile only, so I decided to uninstall python and then I used Uniget to do the install to be available systemwide.
This added python to the system path, but then only python –version would respond, but python3 –version did nothing. I had to add a symlink for python3.exe to python.exe by running this from an elevated CMD prompt (not PowerShell)
mklink "C:\Windows\System32\python3.exe" "C:\Program Files\Python313\python.exe"
symbolic link created for C:\Windows\System32\python3.exe <<===>> C:\Program Files\Python313\python.exe
now python –version works
Displays: Python 3.13.1
Now get the script from NetworkChuck’s blog here: https://blog.networkchuck.com/posts/my-insane-blog-pipeline/
from the root of the hugo site, type: code images.py , and then past the script and modify the 3 paths at the top to the ones I’m using on my site:
import os
import re
import shutil
# Paths (using raw strings to handle Windows backslashes correctly)
posts_dir = r"C:\Users\username\iCloudDrive\iCloud~md~obsidian\master\MyWebsite\nameblog\content\posts"
attachments_dir = r"C:\Users\username\iCloudDrive\iCloud~md~obsidian\master\all_attachments"
static_images_dir = r"C:\Users\username\iCloudDrive\iCloud~md~obsidian\master\MyWebsite\nameblog\static\images"
# Step 1: Process each markdown file in the posts directory
for filename in os.listdir(posts_dir):
if filename.endswith(".md"):
filepath = os.path.join(posts_dir, filename)
with open(filepath, "r", encoding="utf-8") as file:
content = file.read()
# Step 2: Find all image links in the format 
images = re.findall(r'\[\[([^]]*\.png)\]\]', content)
# Step 3: Replace image links and ensure URLs are correctly formatted
for image in images:
# Prepare the Markdown-compatible link with %20 replacing spaces
markdown_image = f"})"
content = content.replace(f"[[{image}]]", markdown_image)
# Step 4: Copy the image to the Hugo static/images directory if it exists
image_source = os.path.join(attachments_dir, image)
if os.path.exists(image_source):
shutil.copy(image_source, static_images_dir)
# Step 5: Write the updated content back to the markdown file
with open(filepath, "w", encoding="utf-8") as file:
file.write(content)
print("Markdown files processed and images copied successfully.")
the just run the script from the root of the hugo site:
❯ python3 .\images.py
Markdown files processed and images copied successfully.
Script seems to be working, but then running the hugo server on localhost, the image doesn’t display, just the name of the attachment.
Just running this command fixed the issue:
hugo server –ignoreCache -t terminal
Git Setup
## Generate an SSH key (Mac/Linux/Windows)
ssh-keygen -t rsa -b 4096 -C "email address"
Output:
Generating public/private rsa key pair.
Enter file in which to save the key (C:\Users\username/.ssh/id_rsa):
Created directory ‘C:\Users\username/.ssh’.
Enter passphrase (empty for no passphrase):
Enter same passphrase again:
Your identification has been saved in C:\Users\username/.ssh/id_rsa
Your public key has been saved in C:\Users\username/.ssh/id_rsa.pub
The go to github.com > settings > ssh and gpg keys section > add new ssh key and paste the output of cat id_rsa.pub into the field and then add it.
Then test authentication
ssh -T [email protected]
Hi username! You've successfully authenticated, but GitHub does not provide shell access.
Then, change into the root directory of our hugo website, and then create a remote git repo using this command:
❯ git remote add origin [email protected]:githubname/nameblog.git
then just run command: hugo
this takes all the markdown files and converts them to a website (similar to the hugo server -t terminal command was doing)
Now add all the files to git repo and then commit them to local repo, then push them to github
git add .
git commit -m "my first commit for the blog"
git push -u origin master
Now create a git subtree branch called ‘webhost-publish’ that will contain just the ‘public’ folder of the master branch as this is essentially only the actual website files that needs to be synced with whatever webhost I will use.
Then push it to the remote repo(ie origin) on github.com. it will create that same branch (webhost-publish)
After you’ve pushed webhost-publish
to the remote repository, you no longer need the local webhost-publish
branch, so it is deleted from your local repository.
The webhost-publish
branch was created as an intermediate step to extract and push the contents of the public
directory to the remote. After you’ve pushed the necessary content to the remote repository, the local branch webhost-publish
is no longer needed.
# Step 8: Push the public folder to the webhost-publish branch using subtree split and force push
echo "Deploying to GitHub webhost-publish..."
git subtree split --prefix public -b webhost-publish
git push origin webhost-publish --force
git branch -D webhost-publish
❯ git subtree split –prefix public -b webhost-publish
Created branch ‘webhost-publish’
99e334008887cf7ed12f300155f9d396a53db133
❯ git push origin webhost-publish –force
Enumerating objects: 72, done.
Counting objects: 100% (72/72), done.
Delta compression using up to 12 threads
Compressing objects: 100% (58/58), done.
Writing objects: 100% (72/72), 288.04 KiB | 3.56 MiB/s, done.
Total 72 (delta 20), reused 0 (delta 0), pack-reused 0 (from 0)
remote: Resolving deltas: 100% (20/20), done.
remote:
remote: Create a pull request for ‘webhost-publish’ on GitHub by visiting:
remote: https://github.com/githubname/nameblog/pull/new/webhost-publish
remote:
To github.com:githubname/nameblog.git
- [new branch] webhost-publish -> webhost-publish
❯ git branch -D webhost-publish
Deleted branch webhost-publish (was 99e3340).
Webhosting
sign up for a year of Hostinger.com webhosting
add a blank site and attach theconspiracygod.com domain to it
generate an SSH key and then go add it to github > settings > ssh
go to cloudflare dashboard and update the ‘A’ records for the domain and the ‘www’ entry
On Hostinger, choose AutoDeploy and get the webhook address, then click the github link and copy the webhook link into github and save.
Setup script to automate all steps
modified networkchuck’s script with my settings and saved as publish.ps1 in the hugo site root directory
# PowerShell Script for Windows
# Set variables for Obsidian to Hugo copy
$sourcePath = "C:\Users\username\iCloudDrive\iCloud~md~obsidian\master\posts"
$destinationPath = "C:\Users\username\iCloudDrive\iCloud~md~obsidian\master\MyWebsite\nameblog\content\posts"
# Set Github repo
$myrepo = "[email protected]:githubname/nameblog.git"
# Set error handling
$ErrorActionPreference = "Stop"
Set-StrictMode -Version Latest
# Change to the script's directory
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Definition
Set-Location $ScriptDir
# Check for required commands
$requiredCommands = @('git', 'hugo')
# Check for Python command (python or python3)
if (Get-Command 'python' -ErrorAction SilentlyContinue) {
$pythonCommand = 'python'
} elseif (Get-Command 'python3' -ErrorAction SilentlyContinue) {
$pythonCommand = 'python3'
} else {
Write-Error "Python is not installed or not in PATH."
exit 1
}
foreach ($cmd in $requiredCommands) {
if (-not (Get-Command $cmd -ErrorAction SilentlyContinue)) {
Write-Error "$cmd is not installed or not in PATH."
exit 1
}
}
# Step 1: Check if Git is initialized, and initialize if necessary
if (-not (Test-Path ".git")) {
Write-Host "Initializing Git repository..."
git init
git remote add origin $myrepo
} else {
Write-Host "Git repository already initialized."
$remotes = git remote
if (-not ($remotes -contains 'origin')) {
Write-Host "Adding remote origin..."
git remote add origin $myrepo
}
}
# Step 2: Sync posts from Obsidian to Hugo content folder using Robocopy
Write-Host "Syncing posts from Obsidian..."
if (-not (Test-Path $sourcePath)) {
Write-Error "Source path does not exist: $sourcePath"
exit 1
}
if (-not (Test-Path $destinationPath)) {
Write-Error "Destination path does not exist: $destinationPath"
exit 1
}
# Use Robocopy to mirror the directories
$robocopyOptions = @('/MIR', '/Z', '/W:5', '/R:3')
$robocopyResult = robocopy $sourcePath $destinationPath @robocopyOptions
if ($LASTEXITCODE -ge 8) {
Write-Error "Robocopy failed with exit code $LASTEXITCODE"
exit 1
}
# Step 3: Process Markdown files with Python script to handle image links
Write-Host "Processing image links in Markdown files..."
if (-not (Test-Path "images.py")) {
Write-Error "Python script images.py not found."
exit 1
}
# Execute the Python script
try {
& $pythonCommand images.py
} catch {
Write-Error "Failed to process image links."
exit 1
}
# Step 4: Build the Hugo site
Write-Host "Building the Hugo site..."
try {
hugo
} catch {
Write-Error "Hugo build failed."
exit 1
}
# Step 5: Add changes to Git
Write-Host "Staging changes for Git..."
$hasChanges = (git status --porcelain) -ne ""
if (-not $hasChanges) {
Write-Host "No changes to stage."
} else {
git add .
}
# Step 6: Commit changes with a dynamic message
$commitMessage = "New Post on $(Get-Date -Format 'yyyy-MM-dd HH:mm:ss')"
$hasStagedChanges = (git diff --cached --name-only) -ne ""
if (-not $hasStagedChanges) {
Write-Host "No changes to commit."
} else {
Write-Host "Committing changes..."
git commit -m "$commitMessage"
}
# Step 7: Push all changes to the main branch
Write-Host "Deploying to GitHub Master..."
try {
git push origin master
} catch {
Write-Error "Failed to push to Master branch."
exit 1
}
# Step 8: Push the public folder to the webhost-publish branch using subtree split and force push
Write-Host "Deploying to GitHub webhost-publish branch..."
# Check if the temporary branch exists and delete it
$branchExists = git branch --list "webhost-publish"
if ($branchExists) {
git branch -D webhost-publish
}
# Perform subtree split
try {
git subtree split --prefix public -b webhost-publish
} catch {
Write-Error "Subtree split failed."
exit 1
}
# Push to webhost-publish branch with force
try {
git push origin webhost-publish --force
} catch {
Write-Error "Failed to push to webhost-publish branch."
git branch -D webhost-publish
exit 1
}
# Delete the temporary branch
git branch -D webhost-publish
Write-Host "All done! Site synced, processed, committed, built, and deployed."
Everything is now working.
I can create a new obsidian blog post and then all I have to do is run the powershell script ‘publish.ps1’ from the root of the hugo site folder.