Handbook Maintenance
On a regular basis (currently monthly), an issue with a list of maintenance tasks is created and assigned to the relevant DRIs.
The issues, pipeline schedule, and all relevant code reside in the maintenance tasks project.
Note: The maintenance project uses the TRIAGE_BOT
group access token.
See access tokens for more on general management of tokens.
Content tasks
Content tasks are assigned to the content DRI. They should consider a designated backup from their team if needed.
Codeowners check
The “Employment GitLab Bot” should create MRs to remove users who are offboarding through the work done in employment-automation#270
.
However, we want to double check that it’s catching everyone.
In each handbook project:
- Look for the latest
build
job on themain
(default) branch. - See if there are any messages similar to
WARN Code owner some-username not found... falling back to public GitLab profile.
- Check Slack or Workday to see if these users are no longer at GitLab.
- Create a MR to remove the users who are no longer at GitLab from the
CODEOWNERS
file. - If the user is still at GitLab, create a MR to update the team member entry such that their GitLab username reflects their username.
- Follow the instructions in the latest “Employment GitLab Bot” MR on where to file an issue for the engineer to investigate why the user was not removed.
Image compressing
This is a semi-automated task.
Manual process
If you have Ruby installed, you can use the local version of the script: ruby ./maintenance-tasks/scripts/compress_images_local.rb -u your-username
Alternatively:
-
Ensure you have each image compression tool installed locally,
pngquant
andjpegoptim
. -
Run the commands below in the root folder for the relevant project.
find . -name '*.png' -size +100k -print0 | xargs -0 -P8 -L1 pngquant --ext .png --force --skip-if-larger ## Run the following a second time with the `.jpeg` extension find . -name '*.jpg' -size +50k -print0 | xargs -0 -P8 -L1 jpegoptim -t -T5
-
Create a MR with the compressed images.
Note: If you’re looking into decreasing GIF file sizes, consider Gifsicle
, which you can use to drop frames or other methods. You can combine it with the Lossy GIF encoder for added compression.
Image use checking
This is a semi-automated task.
Manual process
- Download the unlinked images script if you do not already have it, ensuring it’s the latest version.
- Run the script in for each handbook repository:
/path/to/unlinked_images.sh /path/to/handbook/repository
. - Make a MR to remove all unlinked files. If there are a lot, consider breaking it up into smaller MRs.
- Ask the person who added or recently edited the images to review the MR.
Link checking (external)
To check links, ensure you have lychee installed locally.
- Download the lychee configuration file if you do not already have it, ensuring it’s the latest version.
- Ensure each handbook repository is up to date. You will need to complete the task for each repository. The rest of the instructions refer to a single repository.
- In the root folder of the repository, run
lychee -i -E -u "curl/8.4.0" -b "htts://handbook.gitlab.com" -c ~/Downloads/lychee.toml -f markdown --include-fragments content/**/*.md
, replacing the configuration file location to wherever you have downloaded it to.- The base
-b
is included to prevent errors, but handbook links are excluded since we usehugolint
to check relative links. - The insecure
-i
, exclude private-E
, and user agent-u
flags are optional. They are included in an attempt to cut down on errors.
- The base
- Edit the repository files to fix all links, as much as possible.
- If a (sub)domain should be ignored, update the configuration file to exclude it. Domains that require a login should be excluded.
- If the link redirects, replace the link with the new URL.
- If a page is under maintenance or unavailable at this time, but it is likely to be available again in the future, leave it as is.
- If the site still exists, but the page doesn’t exist, try searching for a new version.
- If the site no longer exists, remove the link, and optionally, make a note
(link no longer available)
. - Some sites (such as LinkedIn) may be rate limited (throws a 429), check these manually or leave as is.
- If some pages or folders should be ignored, such as the
node_modules
folder used for local development, add them to the exclusion list.
- Optionally, run lychee again to verify there are no changes you missed.
- Once you have fixed the links, create an MR.
- If needed, create an issue with a report on broken links that you could not fix, and tag the maintainers (or post in a relevant Slack channel).
Link checking (internal)
Each site has hugolint
running as part of the pipeline to check for broken relative links.
For each site:
- Look for the latest
hugolint
job. - Browse or download the artifacts.
- The
missinglinks.txt
lists the top 20 broken links, while thelinkcheck
file lists all of them. - Fix at least the top 20 from the
missinglinks
. The number before the path indicates the number of occurrences.- Most of the time, there is a redirect in place, so visiting the “broken” path will take you to the new one. Otherwise, search the same handbook, then the other handbook for a similar page.
- Create a MR with the fixed links.
Once all relative links are fixed in one of the handbooks, the hugolint
job should no longer be allowed to fail.
For more information, see the Improve internal link job epic.
Linter related tasks
Check for configuration updates
Check for changes to the product documentation markdownlint and vale configuration changes.
Update the handbook projects as needed. Not all rules are the same as the product documentation.
The markdown guide explains a lot of the usage for markdownlint
.
Vale usage refers to product documentation as the reasons are the same.
Check for and fix errors
To run the linters locally, ensure that you have markdownlint
and vale
installed.
You can reference the product documentation local testing information on how to install them.
More information how they are used in the handbook can also be found in the development docs.
Fix trailing spaces
This is a semi-automated task.
Manual process
If the job isn’t working, here’s the manual process.
- Ensure you have the task markdownlint config file.
- Run
markdownlint-cli2 --config tasks/task.markdownlint-cli2.jsonc **/*.md --fix
, replacing the file path with where you have downloaded the configuration file. - Commit the changes, and create an MR. You can combine the MR with other linter tasks.
Check and fix errors
The linter jobs do not run on main
due to how slow they can be.
Part of maintenance then is to double check that errors are not somehow being merged.
The instructions are for a single repository. Do the following for each repository:
- In the root folder, run
markdownlint-cli2 "content/**/*.md"
. - In the root folder, run
vale --no-wrap --minAlertLevel error content/**/*.md
. - For each report, note the number of errors in the maintenance issue.
- Fix all errors.
- Note: For
markdownlint
, you can run it again with the--fix
flag to automatically fix some errors.
- Note: For
- Create a MR with fixes.
Once completed, look back at the previous maintenance issues.
If in the last 3 months, we consistently have errors, discuss with the handbook DRIs about adding linting to main
.
If there are no errors in the last 3 months, we can likely safely remove this task from maintenance.
Redirects removal
- In each repository, look for the
index.redirects
file. - Remove any redirects past the removal date as noted in a comment.
- Create a MR.
Backend tasks
Backend tasks are assigned to the technical DRI. They should consider a designated backup from the “Code Maintainer” group if needed.
Dependencies
The dependencies should be reviewed and bumped as appropriate.
Pages to check
When upgrading depencies, we should check that everything is build and renders correctly.
To help with this process, below are a list of elements and examples.
Applies to both public and internal handbook:
- Navigation bar (at the top of every page): check links, and search.
- Sidebars: left navigation, and right table of contents.
- Home page: table of content generation.
- Images and videos render.
- Public: Values page, includes SVG for hierarchy.
- Internal: Self-help troubleshooting
- Tableau embeds:
- Public: Support KPIs
- Internal: KPI indicators, click through to a page for charts
- Mermaid charts:
- Public: Mermaid charts
- Internal: Identity verification
Public handbook:
- Pages generated using layouts and
www-gitlab-com
data files. - Pages generated using shortcodes and data file.
Docsy
Whenever the upstream Docsy theme releases a new version, open a new issue and select the Docsy-upgrade template. Follow the steps in the issue to upgrade the theme.
Golang / Hugo / NodeJS
It’s important to keep the language versions up-to-date and in sync across the projects. When upgrading versions be sure to make those changes to docsy-gitlab
, handbook
, and internal-handbook
.
Versions are defined in both the .tool-versions
file and the .gitlab-ci.yml
file. Make sure those versions stay in sync or you might see inconsistent behavior between local and CI builds.
Review changelogs for each tool to be aware of any potential breaking changes. Changelogs can be found:
Process:
- When upgrading dependencies, test the changes first.
- If there are no concerns, next, release a new version of the Docsy theme with the changes.
- With the new version of the theme available, create new MRs in
handbook
andinternal-handbook
to apply the version changes and upgrade to the newly released versions of thedocsy-gitlab
. Use review apps for both MRs to test the dependency changes in a production-like environment.
markdownlint and vale
The linters shouldn’t need to be updated often, but may include fixes or new useful features.
- markdownlint changelog, refer to markdownlint project tags for dates
- Vale releases
Review the release log for each and determine if a version bump is appropriate. If so:
- Update your local linter to the appropriate version.
- Test the linters locally to ensure there are no errors.
- Bump the version in
.tool-versions
and.gitlab-ci.yml
. - Create MR.
Hugo mod
The hugo mod
command should be used for managing dependencies for handbook projects. Below are some usage examples:
Adding a dependency
hugo mod get server.com/project-name@version
Example:
hugo mod get github.com/google/docsy@v0.9.1
Tidy
Tidy will clean up unused entries in the go.mod
and go.sum
files. This should be run during monthly maintenance.
hugo mod tidy
Updating a dependency
To update a specific dependency to its latest version run the following command:
hugo mod get -u github.com/google/docsy
To update all dependencies run:
hugo mod get -u ./...
And remember to commit both go.mod
and go.sum
files.
Semi-automated tasks
Numerous maintenance jobs have been made into manual CI jobs that are created with the maintenance issue. After 3 months, if there are no issues with the CI jobs when triggered manually, consider making them non-manual jobs to be “automated”.
For any “semi-automated job”:
- Run the job by going to the latest scheduled pipeline in the maintenance project.
- Click on the play button for the relevant job.
- Optionally, open the job log to see what it’s doing.
- Once the job is finish, you should be assigned 2-3 MRs.
- If you do not get any MRs assigned, check the job log for errors, or if no changes were necessary.
If needed, check the manual process details to see what might be going wrong, or to run the task manually.
Automated tasks
Some tasks are automated.
Clean up stopped environments
GitLab does not automatically delete old stopped environments.
As part of the monthly maintenance schedule, a CI job uses the API to delete stopped review apps.
Refer to the API calls for the settings.
Count words and pages
The handbook has an update counts script, which counts the number of words and pages.
The CI schedule should be owned by a current handbook DRI. When the job runs, the script with create a MR with the relevant changes and assign it to the user who owns the schedule.
Irregular tasks
Some other tasks that may be occasionally done, but are not part of monthly maintenance.
Delete old branches
One way to delete old branches is to use git
to sort by date and delete a specified number of branches.
git fetch --all --prune
git branch -r --sort=authordate:iso8601 | head -n 10 | sed 's/ origin\///' | grep -v '^master$' | xargs git push origin -d