This directory houses all of the assets used to build Vector's website and documentation, available at vector.dev.
In order to run the site locally, you need to have these installed:
- The Hugo static site generator. Make sure to install the extended version (with Sass and ESBuild support), specifically the version specified in
netlify.toml
. - The CLI tool for the CUE configuration and validation language.
- Node.js and the Yarn package manager (for static assets and some scripting).
- htmltest for link checking.
vector.dev is a complex site with a lot of moving parts. This section breaks the site down into some key components.
vector.dev is built by and hosted on the Netlify platform. Deploy previews are built for all pull requests to the Vector project (though this may change in the near future).
You can update site configuration and see the results of site builds on the Netlify project page. The configuration for Netlify is in the root of this repo, in the [netlify.toml
][netlify_toml] file.
The current Vector release branch (vX.X
, for example v0.15
) branch is used to build the "production" site at https://vector.dev. All changes should be targeted to master
. If you want to release a website change outside of the normal release cadence, you can cherry-pick the commit to the release branch.
The master
branch, on the other hand, often contains unreleased, "nightly" changes to Vector that we don't yet want reflected on the website. The nightly version of the site is available at https://master.vector.dev. This version of the site may be useful for Vector users taking advantage of not-yet-released features.
vector.dev is built using the Hugo static site generator. The site configuration is in config.toml
. The standard Hugo directory structure is obeyed.
The Vector documentation relies heavily on structured data supplied using the CUE configuration and data validation language. Uses of CUE data include the docs for Vector's many components and the docs for Vector Remap Language.
All of the CUE sources for the site are in the cue
directory. Whenever you build the Vector site, the CUE sources are compiled into a single JSON file that's stored at data/docs.json
. That information is then used in conjunction with Hugo's templating system to build HTML.
There's a variety of helper commands available for working with CUE. Run make cue-help
for CLI docs.
Having trouble with CUE? See CUE pro tips below for some pointers.
For the most part, vector.dev uses the Alpine framework for interactive functionality. If you see directives like x-show
, x-data
, @click
, and :class
in HTML templates, those are Alpine directives. Alpine was chosen over jQuery and other frameworks for the sake of maintainability. Alpine directives live inside your HTML rather than in separate JavaScript files, which enables you to see how a component behaves without referring to an external .js
file.
The Spruce library is used for all JavaScript state management. It stores things like light/dark mode preferences in localStorage
and makes those values available in Alpine-wired components. See the app.js
for managed state values.
The Tocbot library is used to auto-generate documentation table of contents on each page. The TOC is generated at page load time.
You'll also find two React.js components on the site: the spinning globe on the main page and the interactive search bar. The TypeScript for those components is in home.tsx
and search.tsx
, respectively. React.js compilation is configured using the babel.config.js
file and TypeScript compilation is configured using the tsconfig.json
file.
All JavaScript for the site is built using Hugo Pipes rather than tools like Webpack, Gulp, or Parcel.
Most of the site's CSS is provided by Tailwind, which is a framework based on CSS utility classes. The Tailwind configuration is in tailwind.config.js
; it mostly consists of default values but there are some custom colors, sizes, and other attributes provided there. Tailwind was chosen for the sake of maintainability; having most CSS inside the HTML templates makes it easier to understand and update a given component's styling. CSS post-processing for Tailwind is performed by PostCSS, which is configured via the postcss.config.js
file.
In addition to Tailwind classes, some CSS is built from Sass (all Sass files are in assets/sass
):
home.sass
styles some elements that are only on the home pagesyntax.sass
provides the colors for syntax highlightingtoc.sass
styles documentation pages' table of contents. Tailwind doesn't work for this because the HTML for the TOCs is generated at page load time by Tocbot.unpurged.sass
contains all the CSS that should not be run through PostCSS. The problem in some cases is that PostCSS purges classes that aren't found in the HTML that's built by Hugo because they're built by other processes, like JavaScript that runs at load time. Anything inunpurged.sass
escapes the purging process.
Search for vector.dev is provided by Algolia. Our search solution is largely custom:
- The
algolia-index.ts
script indexes all of the relevant pages on the site and stores the entire index in a single JSON file (output topublic/search.json
). - The
atomic-algolia
tool syncs the generated JSON index with the Algolia backend, performing all the necessary create, update, and delete operations.
The Algolia configuration for the site is controlled via the algolia.json
file. The Algolia CLI syncs this config with the Algolia API.
Everything needed to configure Algolia search for vector.dev is in this repo; you should never make manual configuration changes through the Algolia dashboard.
If you need to prevent a page from being indexed, you can add noindex: true
to the page's metadata. Here's an
example:
---
title: Don't index me, bro
noindex: true
---
vector.dev uses two different icon sets for different purposes:
- Ionicons is used for corporate logos (Twitter, GitHub, etc.)
- Heroicons is used for everything else. In general the outline variants are preferred.
Redirects for vector.dev are defined in three difference places (depending on the use):
- Domain-level redirects, e.g. the chat.vector.dev redirect to our Discord server, are defined in
netlify.toml
in the repo root. - Splat-style redirects (which can't be defined as Hugo aliases) are defined in
./static/_redirects
. - Redirects for specific pages are defined in the
aliases
field in the relevant page's front matter.
vector.dev uses the htmltest link checker to sniff out broken links. Whenever the site is built in CI, be it the preview build or the production build, all internal links on the site are checked. If any link is broken, the build fails. If you push changes to the docs/site and your build yields a big red X but running the site locally works fine, scan the CI output for broken links. There's a good chance that that's your culprit.
You can run the full CI builds locally, with link checking included:
# Production
make local-production-build
# Preview
make local-preview-build
The standard link checking configuration is in .htmltest.yml
. As you can see from this config, external links are not checked (CheckExternal: false
). That's because external link checking makes builds highly brittle, as they become dependent upon the availability of external websites, i.e. if CloudFlare has an outage or Wikipedia goes down, the vector.dev build fails. The trade-off here, of course, is that broken external links can go undetected. The half-solution is to periodically run ad hoc external link checks:
make local-production-build
make run-external-link-checker
That second make command runs htmltest using the .htmltest.external.yml
configuration, which sets CheckExternal
to true
. We should strive to run this periodically in local environments to make sure we don't have too much drift over time.
Below is a list of common tasks that maintainers will need to carry out from time to time.
make serve
This builds all the necessary prereqs for the site and starts up a local web server. Navigate to http://localhost:1313 to view the site.
When you make changes to the Markdown sources, Sass/CSS, or JavaScript, the site re-builds and Hugo automatically reloads the page that you're on. If you make changes to the structured data sources, however, you need to stop the server and run make serve
again.
-
Add the new version to the
versions
list incue/reference/versions.cue
. Make sure to preserve reverse ordering. -
Generate a new CUE file for the release by running
make release-prepare
in the root directory of the Vector repo. This generates a CUE file atcue/releases/{VERSION}.cue
. -
Add a new Markdown file to
content/en/releases
, where the filename is{version}.md
(e.g.0.12.0.md
) and the file has metadata that looks like this:--- title: Vector v0.13.0 release notes weight: 19 ---
The
title
should reflect the version, while theweight
should be the weight of the next most recent version plus 1. The file for version 0.8.1, for example, has a weight of 8, which means the weight for version 0.8.2 (the next higher version) is 9. This metadata is necessary because Hugo can't sort semantic versions, so we need to make the ordering explicit. If Hugo ever does allow for semver sorting, we should remove theweight
s.
Lighthouse scores for the website are produced automatically by Netlify's Lighthouse plugin. Those reports are available at ${ROOT}/reports/lighthouse
, where ROOT
is the root URL for a version of the site. Thus, reports for the production version of the site would be available at https://vector.dev/reports/lighthouse. Reports are also generated for deploy previews and branch deploys.
- Tailwind's typography plugin is used to render text throughout the site. It's a decent library in general but is also rather buggy, with some rendering glitches in things like lists and tables that we've tried to compensate for in the
extend.typography
block in the Tailwind config, but it will take some time to iron all of these issues out.
CUE can be tricky, tripping up even the most seasoned veterans of the language. Below are some tips that might help you get over the hump with whatever CUE logic you're trying to add to the Vector docs.
We generally advise writing CUE in an incremental way. If you add a lot of new CUE logic and then validate what you've added, the likelihood of encountering inscrutable errors and having little insight into where specifically you went wrong is quite high. Instead, add and then validate little bits at a time. Tools like watchexec
can help with this. Here's an example command (run here in the website
directory):
watchexec "make cue-build"
This runs the CUE build every time you save a change to your CUE sources. The feedback loop is typically 2-5 seconds.
Good:
description: """
Here is a long string...
"""
Bad:
description: """
Here is a long string...
"""
Also bad:
description: """
Here is a long string...
"""