Hello! 👋 and thank you for considering investing your time in contributing to Launchpad Namespaces! As an open source project, it depends on a strong community to florish, and we welcome any type of contribution (not just code) that aligns with our Code of Conduct.
Some of the ways to contribute:
- Community: by hanging with our community at Discord (The Graph), even if just to let us know you're using Namespaces we would appreciate to hear from you. We don't bite, promise!
- Opening Issues: by being a user and taking the time to report issues (or feature requests) you've ran into. Please see the Opening Issues section below on how to do just that.
- Code: by channeling your skills and knowledge to craft valuable pull requests (PRs). We wholeheartedly welcome your contributions. Please see the Contributing Code section below on how to do just that.
To ensure a consistent and efficient response to your issues, we have created two issue templates that will provide guidance and streamline the process. When creating a new issue in the repository, you will be presented with the option to choose from these templates. This approach aims to enhance clarity and facilitate the information gathering process for a faster resolution.
To contribute code, there's a few requirements you need to go through first:
Our Git hooks system and some of our dependencies for tasks such as code generating or templating are being managed by yarn with zero-installs support. Using corepack is advised:
corepack enable
corepack yarn prepare
Some of our documentation is templated with this tool, and for those tasks to run sucessfully tera-cli must be available
Namespaces schemas are written in CUE and you will need this tool, follow the upstream guide on how to install it: CUE Installation
Important: Please ensure that you install version greater or equal to 0.6.0-alpha.1
of CUE as previous versions contain a bug that can cause issues with our code.
Please note that due to licensing restrictions, Apple ships macOS with GNU Bash v3.2, which is an outdated version dating back to 2007. To ensure compatibility and access to the latest features, we recommend installing a more recent version of bash. You can easily accomplish this by using Homebrew:
brew install bash
Although macOS includes a BSD-based grep utility, it's worth noting that the shipped version is outdated and lacks support for certain newer options, such as -P for Perl regular expression pattern search. To overcome this limitation, you have the option to install a more recent version of grep using Homebrew:
brew install grep
After installing the newer version, please be aware that it is accessed using the command ggrep
instead of the default grep
. This naming distinction is in place to avoid conflicts with the preinstalled macOS grep
utility.
Once you have successfully fulfilled the previous requirements in your operating system, the next logical step is to clone this repository and initialize the yarn packages using the following command:
yarn install
You're all set up and ready to go! Continue reading for conventions and a concise overview of the repository layout and implementation details.
We follow conventional commits.
In brief, each commit message consists of a header, with optional body and footer:
<type>[optional scope]: <description>
[optional body]
[optional footer(s)]
<type>
must be one of the following:
- feat: A new feature
- fix: A bug fix
- docs: Documentation only changes
- style: Changes that do not affect the meaning of the code (white-space, formatting, missing semi-colons, etc)
- refactor: A code change that neither fixes a bug nor adds a feature
- perf: A code change that improves performance
- test: Adding missing tests
- chore: Changes to the build process or auxiliary tools and libraries such as documentation generation
- revert: If the commit reverts a previous commit, contains the header of the reverted commit.
Make sure to include an exclamation mark after the commit type and scope if there is a breaking change.
<scope>
optional and could be anything that specifies the place of the commit change, e.g. solver, [filename], tests, lib, ... we are not very restrictive on the scope. The scope should just be lowercase and if possible contain of a single word.
<description>
contains succinct description of the change with imperative, present tense. don't capitalize first letter, and no dot (.) at the end.
<body>
include the motivation for the change, use the imperative, present tense
<footer>
contain any information about Breaking Changes and reference GitHub issues that this commit closes
Commits in a pull request should be structured in such a way that each commit consists of a small logical step towards the overall goal of the pull request. Your pull request should make it as easy as possible for the reviewer to follow each change you made. For example, it is a good idea to separate simple mechanical changes like renaming a method that touches many files from logic changes. Your pull request should not be structured into commits according to how you implemented your feature, often indicated by commit messages like 'Fix problem' or 'Cleanup'. Flex a bit, and make the world think that you implemented your feature perfectly, in small logical steps, in one sitting without ever having to touch up something you did earlier in the pull request. (In reality, that means you'll use git rebase -i
a lot).
Please do not merge the remote branch into yours as you develop your pull request; instead, rebase your branch on top of the latest remote if your pull request branch is long-lived.
Releases are tag-based, and the tags must follow semantic versioning as well as the pattern <namespace>-v<version>
. Given Namespaces as a software is distributed via git refs, tags are the actual releases and creating GitHub Releases serves only the purpose of providing Release Notes.
We do continuous integration on the main branch and canary releases (with a prerelease suffix of -canary.X
) are automatically made on a commit-by-commit basis depending on which namespaces the commit impacts. The versioning of the canary releases is always the next patch version, and a prerelease suffix that gets incremented starting on -canary.1
.
As such, the release process consists of tagging a namespace release and pushing that tag, i.e.
git checkout main
git pull --all
git tag -a "storage-v1.1.2" -m "Release storage-v1.1.2"
git push --tags
The CI pipeline will generate a GitHub Release with release notes, and will then point the relevant semantic tags such as storage-v1.1
, storage-v1
, storage-latest
to the new tag.
The usual flow is then:
- merge a PR to main
- a new canary tag will be created automatically
- (...do final tests and fix whatever may come up)
- decide the next version number
- tag [namespace]-v[version] and push the tag
This repository has a lot of dynamically generated content and artifacts, and the source of truth for all of those resides in /src
In here you'll discover a collection of schemas which are the source of truth for each namespace helmfile. Amongst them also resides generate_tool.cue
, which implements the logic for actually generating all the outputs from the schemas.
Here, you will find a repository containing convenient scripts for regenerating helmfiles, documentation, OpenAPI JSON schema files, and more. These scripts are designed to streamline the process and make it easier for you to update and maintain these essential components.
Here, you will find base templates for several document files, and some macros used in the templating.
Each namespace has its own folder where their specific artifacts reside
The default values for a given namespace. In case the namespace has more than one flavor (different sets of defaults), then those reside in /<namespace>/values/<flavor>
, and the _common
folder contains the common values shared by all flavors.
A Namespace is comprised of two things:
- A set (or more, when the namespace has different flavors) of default values for the releases (helm charts)
- A
helmfile.yaml
file that incorporates client-side templating logic, evaluated dynamically by helmfile during runtime. This templating logic provides a rich set of features and interfaces, such as dictionary merging of user-provided values with default values, as well as the ability to toggle specific releases on or off based on feature requirements. It empowers you to efficiently customize and configure your deployments with ease.
The default values are static data, and can be found in the {namespace}/values
folders.
The helmfile.yaml file is dynamically built from schema files written in CUE (check the repository layout for a more comprehensive description of the structure).
The schema files completely define the namespace in its many characteristics, such as:
- what releases they bundle, and the helm chart repos in which to find those releases
- the features it supports
- the labels its releases get deployed with
This is often implemented in the helmfiles in code such as
{{ $__helmDefaults := `{"recreatePods":true}` | fromJson }}
{{ with ( .Values | get "helmDefaults" dict ) }}
{{ $_ := (deepCopy . | mergeOverwrite $__helmDefaults) }}
{{ end }}
which consists of declaring a variable holding a default value, and deep merging with the user passed values, giving precedence to those. Injecting the default value, recreatePods in this example, is done by the CUE tool when building the helmfiles.
Flavors are implemented by having different sets of values in different subfolders, and having the helmfile lookup the folder dynamically, as shown in this example
{{- if ( hasKey .Values "flavor" ) }}
- ./values/_common/{{` "`{{ .Release.Name }}`" `}}.yaml
- ./values/{{ .Values.flavor }}/{{` "`{{ .Release.Name }}`" `}}.yaml
{{- else }}
- ./values/{{` "`{{ .Release.Name }}`" `}}.yaml
{{- end -}}
Features are implemented by wrapping the releases in the helmfile with a conditional, as seen in the following example:
{{ if has "metrics" ( .Values | get "features" list ) }}
{{- $release := "kube-prometheus-stack" }}
- name: 'kube-prometheus-stack'
inherit:
- template: 'kube-prometheus-stack'
values:
{{- tpl $_tplReleaseValues (dict "Values" .Values "release" $release) | indent 4 -}}
{{- end -}}
The source of truth for those is kept in src/charts.cue
, which contains a structure of repositories, and the charts they contain.
You just need to add more (or change them) following the pattern that is being used and taking care to include the metadata fields (descriptions, url, ...)
The structure looks like this:
#repositories: {
"<repository name>": {
url: "<relevant repository URL>"
description: """
Relevant repository description
lorem ipsolum
"""
charts: {
<chart name>: {
url: "<relevant chart URL>"
description: """
Relevant chart description
lorem ipsolum
"""
}
After changing this file, the docs, helmfiles and OpenAPI JSON schema should all be updated. Either by the pre-commit hook or by running the appropriate scripts manually (keep reading for more on those).
For doing this, you need two things:
- Create a new schema file in
src/schemas/<namespace-name>.cue
- Add a README.md.tera file to the
<namespace-name>/
folder
To create the schema file, the easiest approach would be to duplicate an existing one and adjust as required.
It is important that the file has this comment line:
// schema:type=namespace schema:namespace=<namespace name>
and has this block:
// instantiate namespace ojects for internal usage
_namespaces: "ethereum": _#namespaceTemplate & {_key: #namespaces.#ethereum}
The rest is namespace specific in terms of what releases it includes, and if it supports features or flavors, and so on.
Any included releases must link to what's defined in the charts.cue
file via the _repositories
struct, like so:
releases: {
erigon: {
chart: {_repositories.graphops.charts.erigon}
_template: {version: "0.6.0"}
}
so, if you need to add new repositories and charts, see the previous section about how to do just that.
To add the README.md.tera file, you can also just copy any of the existing ones in other namespaces, and take care to adjust this line appropriately:
{% set name = "<namespace name>" %}
Note The following tasks are all run by the husky pre-commit git hook, so you don't need to run them yourself unless you're developing and want to do that without commiting, or have any other reason for it.
Provided you already have the cue
command installed and available, you should go into the src/schemas
folder (needs to be run from that directory) and run:
cue cmd -t namespace="<namespace>" build:helmfile
so, i.e. for the ethereum namespace you'd run:
cue cmd -t namespace="ethereum" build:helmfile
and should get the helmfile.yaml as output
Alternatively, to build all helmfiles you can also run the src/scripts/update-form-schemas.sh
script.
We have a script that automates that task under src/scripts/update-docs.sh
.
Provided you have all the requisites installed (you ran yarn install
and you have tera-cli available), you can just run it and it will
update the OpenAPI JSON schema, template all the documentation, and do git add on those
We have a script that generates the OpenAPI schema under src/scripts/generate-openapi.sh
. You should be able to run it and you will get it as output.
Part of a pre-commit hook, but you can manually run the src/scripts/update-renovate-repos.sh
script as well.
That script generates the repositories section for the renovate's config, by running a CUE cmd from the src/schemas
directory:
cue cmd build:renovate
and replaces the corresponding section in .github/renovate.json5
.