Module development
Develop your own Hugo modules compatible with Hinode.
Hinode fully supports Hugo modules to provide a flexible and extensible modular framework. By default, Hinode includes core building blocks for Bootstrap, FlexSearch, and Font Awesome. The following paragraphs describe the coding conventions of Hinode modules, illustrate how to automate your build and release process, and give an overview of common issues and resolutions.
Hinode uses several conventions for the modules it maintains. You are encouraged to use the same conventions, especially when contributing your own module for sharing.
The GitHub repositories of modules maintained by Hinode start with the mod-
prefix. Although this is no strict requirement, you are encouraged to follow the same convention when contributing your own modules.
Modules should define a single entrypoint for their stylesheets in assets/scss/{MODULE}
, replacing {MODULE}
with the name of the module without the mod-
prefix. Even if the module uses plain CSS files, the entrypoint should have the .scss
extension to ensure the file is transpiled correctly. The additional source files should be mounted into assets/scss/modules/{MODULE}/
. You can set the value showSCSS
to true
in the debugging
section of the site’s parameters to show which files are processed in which order.
JavaScripts should be mounted in assets/js/modules/{MODULE}/
. Hinode bundles these files into a single script if the module is a core module. Be aware that the script files within the module are processed alphabetically, should you have any interdependencies in your scripts. You can set the value showJS
to true
in the debugging
section of the site’s parameters to show which files are processed in which order.
Hinode modules use GitHub actions to keep dependencies up-to-date and to publish new releases automatically. Review the following sections how to configure dependency upgrades, how to automate Pull Requests merges, and how to publish new releases after each succesful merge.
You can configure dependabot or set up a custom GitHub action to automatically upgrade your dependencies. The applicable approach depends wether you use npm packages or Hugo modules as your module source. Review the next two section how to configure automated dependency upgrades.
Dependabot automatically keeps the dependencies and npm packages used in your repository updated to the latest version. The Hinode module template includes a basic configuration that is enabled by default. It checks for any version updates on a daily basis. The configuration is defined in .github/dependabot.yml
and includes a commit-message that is used for release automation.
version: 2
updates:
- package-ecosystem: npm
directory: "/"
schedule:
interval: daily
commit-message:
prefix: fix
include: scope
At this moment, Dependabot has no support for Hugo modules yet (see feature request #6860
). The Hinode module template provides a custom workflow in .github/workflows/update.yml
to check for available Hugo module upgrades on a daily interval. It creates a Pull Request (PR) from a feature branch when it has found any upgrades. The workflow uses the create-pull-request action from Peter Evans to ensure the PR includes commit messages that trigger the semantic-release bot (using the fix
prefix).
name: Update Hugo Dependencies
on:
workflow_dispatch:
# TODO: uncomment
# schedule:
# - cron: '0 3 * * *' # run daily at 03:00 AM
The workflow requires elevated privileges to your module repository. Create a fine-grained Personal Access Token (PAT) first. Set up the token in the Developer settings
of your Account settings on GitHub. The token requires access to your module repository with the following permissions:
When done, head over action secret
in the security
section of the repository configuration. Create a new Repository token
with the name HUGO_MOD_PR
in your repository configuration and paste the PAT as content.
GitHub’s auto-merge feature automatically merges proposed Pull Requests (PRs) when all conditions have been met. This feature is especially helpful to merge dependency upgrades prepared by dependabot (see the dependency upgrades section for more details). You are strongly encouraged to set up branch protection first, to prevent PRs from breaking your builds.
The Hinode module template provides a basic workflow to test the build. Uncomment the following lines in .github/workflows/test.yml
to enable automated testing upon each PR or push submitted to the main
branch. The workflow calls the test
script defined in the repository’s npm package.json
. By default, it tests on the latest versions of macOS, Windows, and Ubuntu for the latest stable releases of Node.js (currently v18
and v20
).
name: Test
on:
workflow_dispatch:
# TODO: uncomment
# push:
# tags:
# - v*
# branches: [ main ]
# pull_request:
# branches: [ main ]
[...]
Head over to Branches
section within Code and automation
of your repository configuration on GitHub. Create a new rule for your main branch. Hinode uses the following settings for the modules it maintains:
Specify the test you defined previously as required status check. When using Hinode’s default test, you would see six different checks (three platforms with two Node versions each).
Set Allow auto-merge
to enabled in the general
section of your repository configuration. Next, click the button Enable auto-merge
on any PR to actually enable the feature. Alternatively, you can enable the .github/workflows/auto-merge.yml
workflow by adjusting the comments:
name: Dependabot auto-merge
on: pull_request_target
permissions:
pull-requests: write
contents: write
jobs:
review-dependabot-pr:
runs-on: ubuntu-latest
# TODO: to enable auto-merge remove first if-statement and uncomment second if-statement
if: false
# if: ${{ github.event.pull_request.user.login == 'dependabot[bot]' }}
The Hinode module template provides a basic workflow to automate releases. Uncomment the following lines in .github/workflows/release.yml
to enable automated releases upon each merge to the main
branch.
name: Release
on:
workflow_dispatch:
# TODO: uncomment
# push:
# branches:
# - main
The workflow uses the semantic-release bot to automate the creation and publication of releases upon each merge to the main branch. The bot updates the repository content, such as new distribution files added by the postinstall
npm script. It also scans all commit messages and determines the type of release. Finally, it publishes a new release with auto-generated release notes.
Hinode uses the Angular Commit Message conventions. In brief, add the following prefixes to your commit messages to determine the type of release:
feat!:
A breaking change (creates a new major release)feat:
A new feature (creates a new minor release)fix:
A bug fix (creates a new patch release, also triggered by dependabot upgrades)chore:
Changes to the build process or auxiliary tools and libraries such as documentation generation (does not trigger a new release)The workflow requires two secrets within your repository. Add them as action secret
in the security
section of the repository configuration.
SEMANTIC_RELEASE_GIT
The bot requires elevated privileges to your module repository. Create a fine-grained Personal Access Token (PAT) first. Set up the token in the Developer settings
of your Account settings on GitHub. The token requires access to your module repository with the following permissions:
When done, create a new Repository token
with the name SEMANTIC_RELEASE_GIT
in your repository configuration and paste the PAT as content.
NPM_TOKEN
The bot uses npm to automatically update the version defined in package.json
and package-lock.json
. You will need to publish your module to npm first. You can use npm publish
from the command line to authenticate yourself with the npm server and to create a new package if needed. On npm, go to the Access tokens
menu below the avatar of your personal account.
Click on the button Generate New Token
and select Granular Access Token
. Assign read and write permissions to the module package and click on Generate token
. Next, create create a new Repository token
with the name NPM_TOKEN
in your repository configuration on GitHub and paste the npm token as content.
Hugo modules have several constraints to work properly. The below overview provides some common challenges and how to overcome them.
postinstall
script.v4.x.y
to Release 5.x.y
(notice the drop of the v
prefix). As a result, Hugo only downloads the old v4.x.y
release. A workaround is to create a fork for version 6.x only and to use this as a source instead. This requires periodic synchronization of the fork though. Another approach is to use the npm release of Font Awesome instead and to mount the required files. This is the approach taken by the Font Awesome module.Hugo provides a configuration option to replace a remote module with a local folder to simplify development and testing. For example, the FlexSearch module uses a module replacement in the file exampleSite/hugo.toml
. The replacement tells Hugo to use the module code of the parent folder, instead of downloading the remote release assets. However, if the module mod-flexsearch
uses other Hugo modules itself (so-called transitive dependencies), Hugo will throw an error Error: failed to load modules
. Vendor your modules with hugo mod vendor
to fix this issue.
[module]
replacements = 'github.com/gethinode/mod-flexsearch -> ../..'
hugo mod clean
to clear the Hugo module cache and then rerun hugo mod get -u
.