A bot to automatically merge GitHub PRs
Enabling the "require branches be up to date" feature on GitHub repositories is great because, when coupled with CI, master will always be green.
However, as the number of collaborators on a GitHub repo increases, a repetitive behavior emerges where contributors are updating their branches manually hoping to merge their branch before others.
Kodiak fixes this wasteful behavior by automatically updating and merging branches. Contributors simply mark their PR with a (configurable) label to indicate the PR is ready to merge and Kodiak will do the rest; handling branch updates and merging using the minimal number of branch updates to land the code on master.
This means that contributors don't have to worry about keeping their PRs up to date with the latest on master or even pressing the merge button. Kodiak does this for them! ๐
Kodiak efficiently updates pull requests by only updating a PR when it's ready to merge. This prevents spurious CI jobs from being created as they would if all PRs were updated when their targets were updated.
-
Kodiak receives a webhook event from GitHub and adds it to a per-installation queue for processing
-
Kodiak processes these webhook events and extracts the associated pull requests for further processing
-
Pull request mergeability is evaluated using PR data
- kodiak configurations are checked
- pull request merge states are evaluated
- branch protection rules are checked
- the branch is updated if necessary
-
If the PR is mergeable it's queued in a per-repo merge queue
-
A task works serially over the merge queue to update a PR and merge it
-
The pull request is merged ๐
- Kodiak intentionally requires branch protection to be enabled to function, Kodiak won't merge PRs if branch protection is disabled.
- Due to a limitation with the GitHub API, Kodiak doesn't support requiring signed commits. (kodiak#89)
- Due to permission limitations for GitHub Apps, Kodiak doesn't support updating forks of branches. (kodiak#104)
- GitHub CODEOWNERS are not supported. Kodiak will prematurely update PRs that still require a review from a Code Owner. However, Kodiak will be able to merge the PR once all checks pass. (kodiak#87)
- Travis-CI's deprecated commit status updates are not supported. Please upgrade to the newer GitHub Check-based solution. (kodiak#163)
- Using
merge.block_on_reviews_requested
is not recommended. If a PR is blocked by this rule a reviewer's comment will allow the PR to be merged, not just a positive approval. This is a limitation of the GitHub API. Please try GitHub's required approvals branch protection setting instead. (kodiak#153)
-
Install the GitHub app
-
Create a
.kodiak.toml
file in the root of your repository with the following contents:Here are a few examples to pick from and modify.
# .kodiak.toml # Minimal config. version is the only required field. version = 1
# .kodiak.toml version = 1 [merge] method = "squash" delete_branch_on_merge = true dont_wait_on_status_checks = ["WIP"] # handle github.com/apps/wip [merge.message] title = "pull_request_title" body = "pull_request_body" include_pr_number = true body_type = "markdown" strip_html_comments = true # remove html comments to auto remove PR templates
# .kodiak.toml # version is the only required setting in a kodiak config. # it must be set to 1 version = 1 [merge] # label to use to enable Kodiak to merge a PR automerge_label = "automerge" # default: "automerge" # require that the automerge label be set for Kodiak to merge a PR. if you # disable this Kodiak will immediately attempt to merge every PR you create require_automerge_label = true # if this title regex matches, Kodiak will not merge the PR. this is useful # to prevent merging work in progress PRs blacklist_title_regex = "" # default: "^WIP:.*", options: "" (disables regex), a regex string (e.g. ".*DONT\s*MERGE.*") # if these labels are set Kodiak will not merge the PR blacklist_labels = [] # default: [], options: list of label names (e.g. ["wip"]) # choose a merge method. If the configured merge method is disabled for a # repository, Kodiak will report an error in a status message. method = "merge" # default: "merge", options: "merge", "squash", "rebase" # once a PR is merged into master, delete the branch delete_branch_on_merge = false # default: false # Deprecated: Due to limitations with the GitHub API this feature is # fundamentally broken and cannot be fixed. Please use the GitHub branch # protection "required reviewers" setting instead. See this issue/comment # for more information about why this feature is not fixable: https://github.com/chdsbd/kodiak/issues/153#issuecomment-523057332. # # if you request review from a user, don't merge until that user provides a # review, even if the PR is passing all checks block_on_reviews_requested = false # default: false # if there is a merge conflict, make a comment on the PR and remove the # automerge label. this is disabled when require_automerge_label is enabled notify_on_conflict = true # default: true # if there are running status checks on a PR when it's up for merge, don't # wait for those to finish before updating the branch optimistic_updates = false # default: true # use this for status checks that run indefinitely, like deploy jobs or the # WIP GitHub App dont_wait_on_status_checks = [] # default: [], options: list of check names (e.g. ["ci/circleci: lint_api"]) # immediately update a PR whenever the target updates. If enabled, Kodiak will # not be able to efficiently update PRs. Any time the target of a PR updates, # the PR will update. # # If you have multiple PRs against a target like "master", any time a commit # is added to "master" _all_ of those PRs against "master" will update. # # For N PRs against a target you will potentially see N(N-1)/2 updates. If # this configuration option was disabled you would only see N-1 updates. # # If you have continuous integration (CI) run on every commit, enabling this # configuration option will likely increase yourย CI costs if you pay per # minute. If you pay per build host, this will likely increase job queueing. update_branch_immediately = false # default: false # if a PR is passing all checks and is able to be merged, merge it without # placing it in the queue. This will introduce some unfairness where those # waiting in the queue the longest will not be served first. prioritize_ready_to_merge = false # default: false [merge.message] # by default, github uses the first commit title for the PR of a merge. # "pull_request_title" uses the PR title. title = "github_default" # default: "github_default", options: "github_default", "pull_request_title" # by default, GitHub combines the titles of a PR's commits to create the body # text of a merge. "pull_request_body" uses the content of the PR to generate # the body content while "empty" simple gives an empty string. body = "github_default" # default: "github_default", options: "github_default", "pull_request_body", "empty" # GitHub adds the PR number to the title of merges created through the UI. # This setting replicates that feature. include_pr_number = true # default: true # markdown is the normal format for GitHub merges body_type = "markdown" # default: "markdown", options: "plain_text", "markdown", "html" # useful for stripping HTML comments created by PR templates when the `markdown` `body_type` is used. strip_html_comments = false # default: false
See
kodiak/test/fixtures/config
for more examples andkodiak/config.py
for the Python models. -
Configure branch protection
Branch protection must configured on your target branch (typically master) for Kodiak to enable itself. See the excellent GitHub docs for a quick guide to enabling branch protection: https://help.github.com/en/articles/configuring-protected-branches
-
Create an automerge label
The default label is "automerge" and is configured via the
merge.automerge_label
config. If you have disabledmerge.require_automerge_label
, you can skip this step. -
Start auto merging PRs with Kodiak! ๐
Label your PRs with your automerge label and let Kodiak do the rest!
If you have any questions please open a ticket.
Name | Works With Branch Protection | Auto Merging | Auto Update Branches | Update Branches Efficiently | Open Source | Practice Dogfooding | Language |
---|---|---|---|---|---|---|---|
Kodiak | โ | โ | โ | โ | โ | โ | Python |
Bors | โ | โ | โ | โ | โ | โ | Python |
Homu | โ | โ | โ | โ | โ | โ | Python |
Shipit | โ | โ | โ | โ | โ | โ | Ruby |
Gullintanni | โ | โ | โ | โ | โ | โ | Elixir |
Popuko | โ | โ | โ | โ | โ | โ | Go |
Bors-ng | โ | โ | โ | โ | โ | โ | Elixir |
Marge-bot | โ | โ | โ | โ | โ | โ | Python |
Bulldozer | โ | โ | โ | โ | โ | โ | Go |
Mergify | โ | โ | โ | โ | โ | โ | Python |
Autorebase | โ | โ | โ | โ | โ | โ | TypeScript |
Auto Merge | โ | โ | โ | โ | โ | โ | JavaScript |
Merge when green | โ | โ | โ | โ | โ | โ | JavaScript |
Always Be Closing | ๐คทโ | โ | โ | ๐คทโ | โ | ๐คทโ | ๐คทโ |
Auto Merge | ๐คทโ | โ | ๐คทโ | ๐คทโ | โ | ๐คทโ | ๐คทโ |
Ranger | โ โ | โ | โ โ | โ โ | โ | ๐คทโ | ๐คทโ |
- doesn't require changing CI
- follows commit statuses & GitHub checks
- works with PRs โ some services create separate test branches for merging that circumvent the simpler GitHub PR workflow
- automatically merges PR once up to date with master and all required statuses and checks pass
- ensures branches are automatically updated to the latest version of master
- an improvement upon Auto Update Branches where branches are only updated when necessary, as opposed to updating all branches any time their target branch (usually master) updates
Kodiak needs read/write access to PRs as well as your repository to update and merge PRs. This means that Kodiak can see all the code in your repository. Below is a table of all the required permissions and the reasons they are necessary.
name | level | reason |
---|---|---|
repository administration | read-only | branch protection info |
checks | read/write | PR mergeability and status report |
repository contents | read/write | update PRs, read configuration |
issues | read/write | support closing issues using keywords |
pull requests | read/write | PR mergeability, merge PR |
commit statuses | read-only | PR mergeability |
organization members | read-only | view review requests for users/teams of a PR to calculate mergeability |
If you don't want to use the GitHub App you can run Kodiak on your own infrastructure. These instructions describe setting up Kodiak on Heroku using a Docker container, but you should be able to adapt this for other container platforms.
-
Create a new GitHub app via https://github.com/settings/apps/new with the permissions described in the Permissions sections of this document and with the event subscriptions specified below
More information on creating a GitHub app can be found at: https://developer.github.com/apps/building-github-apps/creating-a-github-app/
The necessary event subscriptions are:
event name check run pull request pull request review pull request review comment - For the homepage URL any link should work.
- A GitHub App secret is required for Kodiak to run.
- Download your private key for later and copy your GitHub app ID and secret key for later.
- Use your Heroku app hostname for the webhook URL with
/api/github/hook
appended. Something likehttps://my-kodiak-app.herokuapp.com/api/github/hook
.
-
Setup container on Heroku
Kodiak depends on Redis v5 for persistence.
# a unique name for the heroku app export APP_NAME='kodiak-prod' # create app with container stack heroku apps:create $APP_NAME heroku stack:set container -a $APP_NAME # login to registry heroku container:login # download latest release from docker hub and tag for push to heroku docker pull cdignam/kodiak docker tag cdignam/kodiak registry.heroku.com/$APP_NAME/web # push tagged image to Heroku docker push registry.heroku.com/$APP_NAME/web # create gihub app at https://developer.github.com/apps/building-github-apps/creating-a-github-app/ # The APP_ID and PRIVATE_KEY are needed to run the app. You must also set a SECRET_KEY to pass to the app. # configure app environment (this can also be done through the Heroku web ui) heroku config:set -a $APP_NAME GITHUB_APP_ID='<GH_APP_ID>' SECRET_KEY='<GH_APP_SECRET>' GITHUB_PRIVATE_KEY="-----BEGIN RSA PRIVATE KEY-----\nsome/private/key\nbits\n-----END RSA PRIVATE KEY-----\n" # Redis v5 is required and provided by RedisCloud heroku addons:create -a $APP_NAME rediscloud:30 --wait # release app heroku container:release web -a $APP_NAME
# install dependencies
poetry install
# start dev server
poetry run uvicorn kodiak.main:app --reload
# type check and lint
s/lint
# format code
s/fmt
# test code
s/test
Due to the nature of a GitHub bot, testing relies largely on mocks. For testing to see if a given feature will work it is recommended to create a GitHub App and a testing GitHub repo.
Create a GitHub App via https://github.com/settings/apps/new
-
Configure the app as described in the heroku setup instructions above.
-
Setup the webhook URL
You probably want to use something like
ngrok
for this. If you do usengrok
, you may also want to signup for an account via thengrok
website so that yourngrok
url for the webhook doesn't expire.With
ngrok
installed, we can run it with the Kodiak's dev port.ngrok http 8000
Now we can copy the Forwarding url into the GitHub app form. Don't forget to append the path:
/api/github/hook
and sure to copy thehttps
.Then hit create.
-
Now install the GitHub App
Use the Install option in the sidebar for the GitHub App.
You will want to create a testing GitHub repo with a Kodiak config file with the
app_id
option set to your GitHub app's id.You will also need to setup branch protection in
settings > branches
. Make sure the Branch name pattern matchesmaster
. Then check Require status checks to pass before merging and the sub-option Require branches to be up to date before merging.This allows for the production version of Kodiak to be setup on all repos, while allowing the testing version to run on the configured repo. If the production version of Kodiak finds a non-matching app_id, it will ignore the repository, leaving your local version to handle it.
-
Setup secrets
After creating we need to add a Webhook secret. The field is labeled (optional) but it is necessary for Kodiak to work.
You can fill it in with a UUID -- be sure to hold onto it, we'll need it later.
Now you need to generate a private key via the generate private key button under the Private keys section.
Move the secret key to directory where you are running Kodiak.
Note: you need to replace the $SHARE_SECRET
, $GH_PRIVATE_KEY_PATH
and $GITHUB_APP_ID
with your own values.
The GitHub App ID can be found in the About sections of your GitHub App.
SECRET_KEY=$SHARED_SECRET GITHUB_PRIVATE_KEY_PATH=$GH_PRIVATE_KEY_PATH GITHUB_APP_ID=$GITHUB_APP_ID poetry run uvicorn kodiak.main:app
You can create a test PR via the following shell function.
Note: you need to have hub
installed.
create_mock_pr() {
git pull &&
uuidgen >> "$(uuidgen).txt" &&
git checkout -b $(uuidgen) &&
git add . &&
git commit -am $(uuidgen) &&
git push --set-upstream origin $(git symbolic-ref --short HEAD) &&
hub pull-request -l automerge -m "$(uuidgen)" &&
git checkout master
}
GIT_SHA='62fcc1870b609f43b95de41b8be41a2858eb56bd'
APP_NAME='kodiak-prod'
docker pull cdignam/kodiak:$GIT_SHA
docker tag cdignam/kodiak:$GIT_SHA registry.heroku.com/$APP_NAME/web
docker push registry.heroku.com/$APP_NAME/web
heroku container:release -a $APP_NAME web