Thanks for helping make KEDA better 😍.
There are many areas we can use contributions - documenting scalers, adding FAQ, troubleshooting, samples, and more.
Our documentation is versioned so it's important to make the changes for the correct KEDA version. If you need to introduce a new version, we recommend reading our documentation about it here.
If you have a question about KEDA or how best to contribute, the #KEDA channel on the Kubernetes slack channel (get an invite if you don't have one already) is a good place to start. We also have regular community stand-ups to track ongoing work and discuss areas of contribution. For any issues with the product you can create an issue in this repo.
We provide easy ways to introduce new content:
- Contributing to KEDA
Learn more how to create and build a local environment.
Are you using KEDA in production? Do you want to become a listed user? Say no more!
You can easily get listed by following these steps:
- Upload your logo to
static/img/logos/
(350x180) - Configure your company as a new user in
config.toml
(sorted alphabetically)
[[params.users]]
url = "https://coralogix.com/"
logo = "coralogix.gif"
Here's a good example of Coralogix becoming a listed user!
Do you offer commercial support for KEDA and want to become a listed commercial offering? Say no more!
You can easily get listed by following these steps:
- Upload your logo to
static/img/logos/
(350x180) - Configure your company as a new user in
config.toml
(sorted alphabetically)
[[params.vendors]]
name = "Red Hat"
logo = "vendors/red-hat.png"
description = """
Red Hat integrates KEDA with OpenShift through the **Custom Metrics Autoscaler** (CMA) available through the OpenShift Marketplace.
"""
urls = [
{ text = "Learn more about the CMA", url = "https://cloud.redhat.com/blog/custom-metrics-autoscaler-on-openshift" }
]
To add a new post to the KEDA blog:
$ hugo new blog/my-new-post.md
This creates a boilerplate Markdown file in content/blog/my-new-post.md
whose
contents you can modify. The following fields are required:
title
date
(inYYYY-MM-DD
format)author
To add documentation for a new KEDA scaler:
$ hugo new --kind scaler docs/<VERSION>/scalers/my-new-scaler.md
This creates a boilerplate Markdown file in
content/docs/<VERSION>/scalers/my-new-scaler.md
whose contents you can modify.
Make sure to update the following metadata fields:
title
availability
maintainer
description
To add documentation for a new provider:
$ hugo new --kind provider docs/<VERSION>/providers/my-new-provider.md
This creates a boilerplate Markdown file in
content/docs/<VERSION>/providers/my-new-provider.md
whose contents you can modify.
Make sure to update the following metadata fields:
title
To update the KEDA FAQ page, update the TOML file at
data/faq20.toml
. Here's an example question/answer pair:
[[qna]]
q = "How can I add a new question/answer pair?"
a = "You're looking at it! 😀"
To add a new section to the troubleshooting page:
$ hugo new troubleshooting/<VERSION>/my-new-issue.md
To adjust the order in which the troubleshooting tiles appear, use the weight
parameter in each page's metadata.
In order to maintain the style consistency across different scalers, all the parameters which are listed have to be written using this convention:
- name - Description. (Values: x, y, z, Default: y, Optional, Extra Info)
If a parameter is required or doesn't have defined/default values, the missing info should be removed from the pattern.
Here are a few examples:
targetMetricValue
- Target value for your metric.metricFilter
- Aggregation method of the metric. (Values:max
,min
,average
,sum
,variance
, Default:average
, Optional)metricPeriod
- Granularity of the metric. (Default:300
, Optional)subscriptionName
- Name of the Azure Service Bus queue to scale on. (Optional, Required whentopicName
is specified)
The KEDA documentation is versioned. Each version has its own subdirectory under content/docs. To add a new version, follow these steps:
- Copy the directory for the most recent version. Here's an example:
$ cp -rf content/docs/<CurrentVersion> content/docs/<NewVersion>
- Copy the file for the most recent faq data in the
data
directory. Here's an example:
$ cp -rf data/faq<CurrentVersion> data/faq<NewVersion>
- Navigate to the new faq file:
$ cd content/docs/<NewVersion>/reference/faq.md
- Update the versionData option
{{< faq20 versionData="NEW_FAQ_FILE_NAME" >}}
Replace NEW_FAQ_FILE_NAME
with the file name of the faq data for the new version.
By default, new documentation versions are not listed as available version so it's safe to make changes to them. After every release, the version will be published as new version.
Remember to create the folder for next version with already existing docs in current version.
Make sure that the version on content/docs/{next-version}/deploy.md
is updated
and uses the next version, instead of the current one. Ensure that Kubernetes cluster version is updated as well.
Ensure that compatibility matrix on content/docs/{next-version}/operate/cluster.md
is updated with the compatibilities for the incoming version.
Once a version is ready to be published, we must add the version to the
params.versions.docs
list in config.toml.
More recent versions should be placed first in the list (ordering does matter because the first element in that list is considered the latest version).
Note: Remember to prepare the next version.
The Developer Certificate of Origin (DCO) is a lightweight way for contributors to certify that they wrote or otherwise have the right to submit the code they are contributing to the project. Here is the full text of the DCO, reformatted for readability:
By making a contribution to this project, I certify that:
(a) The contribution was created in whole or in part by me and I have the right to submit it under the open source license indicated in the file; or
(b) The contribution is based upon previous work that, to the best of my knowledge, is covered under an appropriate open source license and I have the right under that license to submit that work with modifications, whether created in whole or in part by me, under the same open source license (unless I am permitted to submit under a different license), as indicated in the file; or
(c) The contribution was provided directly to me by some other person who certified (a), (b) or (c) and I have not modified it.
(d) I understand and agree that this project and the contribution are public and that a record of the contribution (including all personal information I submit with it, including my sign-off) is maintained indefinitely and may be redistributed consistent with this project or the open source license(s) involved.
Contributors sign-off that they adhere to these requirements by adding a Signed-off-by
line to commit messages.
This is my commit message
Signed-off-by: Random J Developer <random@developer.example.org>
Git even has a -s
command line option to append this automatically to your commit message:
$ git commit -s -m 'This is my commit message'
Each Pull Request is checked whether or not commits in a Pull Request do contain a valid Signed-off-by line.
No worries - You can easily replay your changes, sign them and force push them!
git checkout <branch-name>
git reset $(git merge-base main <branch-name>)
git add -A
git commit -sm "one commit on <branch-name>"
git push --force
To build or serve the site locally, follow these steps:
- Fork and clone this repository (for local development only).
- Install the latest LTS release of Node, using nvm for example:
Note: on Windows, the argument to install is
$ nvm install --lts
lts
. - Get npm packages and other prerequisites:
$ npm install
- To build the site, run:
You'll find the generated site files under
$ npm run build
public
. - Serve the site locally at localhost:8888 using:
$ npm run serve
To add a new filter option, simply follow these steps:
- Navigate to the doc file you want to annotate.
- In the frontmatter, add your new filter option.
+++
FILTER_NAME = "filter_value"
+++
Replace FILTER_NAME with any desired name of your choice. Same applies to the value.
- Navigate to the
list.lunr.json
file to edit:cd layouts/_default/list.lunr.json
. - Open the file and go down to line 3. You will notice the format of the data represented in a key/value pair. Just before the closing parenthesis, append your new option like this:
"FILTER_NAME" $scalers.Params.FILTER_NAME
.
Replace FILTER_NAME with the same name represented in the frontmatter (see step 2 above for reference).
- Head over to
config.toml
file. In theparams.lunr
section, you will see two arrays namedvars
andparams
. Add your new filter option's name to each of the arrays:
vars = ["title", "maintainer", "description", "availability", "category", "type", "FILTER_NAME"]
params = ["availability", "maintainer", "category", "type", "FILTER_NAME"]
- Navigate to
javascript.html
and scroll down to where thelunr()
function is being called. You will notice a callback function is being passed to thelunr()
function. Within the callback function, you will also noticethis.field
being called with some values passed in. Append this block of code right after the lastthis.field
function call:
this.field("FILTER_NAME", {
boost: 5,
});
Replace FILTER_NAME with the same name represented in the frontmatter (see step 2 above for reference).
Right after the lunr()
function block, You will find where the parse
object is being modified. Append your new filter option to the object:
parse[doc.title] = {
href: doc.href,
title: doc.title,
maintainer: doc.maintainer,
description: doc.description,
availability: doc.availability,
category: doc.category,
type: doc.type,
FILTER_NAME: doc.FILTER_NAME,
};
- Navigate to
layouts/partials/scaler-layout.html
. Locate the div with a class name offilter-options
. Within the div, add this new block:
<div class="has-extra-top-margin">
<h6>FILTER_NAME</h6>
{{ $FILTER_NAME := slice }} {{ range $scalers := where site.RegularPages
".CurrentSection.Title" "Scalers" }} {{ with $scalers.Params.FILTER_NAME }} {{
$FILTER_NAME = $categories | append ($scalers.Params.FILTER_NAME) }} {{
$FILTER_NAME = uniq $FILTER_NAME }} {{ end }} {{ end }} {{ range $FILTER_NAME
}} {{ $item := . }}
<div>
<input
id="{{ . }}"
type="checkbox"
name="resource_filter"
value="FILTER_NAME:{{ . }}"
/>
<label for="{{ . }}">{{ . }}</label>
</div>
{{ end }}
</div>
Replace FILTER_NAME with the same name represented in the frontmatter (see step 2 above for reference).
- Save your changes and rebuild your frontend.