Create and configure a GitHub App so the assistant can push commits, open PRs, and comment under its own bot identity.
assistant skills install vellum-github-app-setupThis skill creates a GitHub App under a GitHub organization, giving the assistant its own bot identity for git operations. After setup, commits, PRs, and comments will attribute to <app-name>[bot] instead of the user's personal account.
Total manual effort: 4 interactions — Continue to GitHub → Create App → Install on repo → (optional) upload avatar. Everything else is automated.
Note: This skill currently supports GitHub organizations only, not personal accounts.
contents:write, pull_requests:write, checks:read, metadata:read)bin/gh-app-token.mjs (in the workspace root) for refreshing auth tokensassistant credentials CLI must be availablebun in the container (for the token helper script)Run the manifest flow script on the user's host machine (it needs to open a browser and catch a localhost callback):
python3 scripts/create-github-app.py \
--org=ORG_NAME \
--name=APP_NAME \
--url="https://example.com" \
--output=/tmp/github-app-credentials.json
This starts a local server on localhost:29170, opens the browser, and guides the user through two clicks:
The callback server catches the redirect and exchanges the code for credentials automatically. The credentials JSON is saved to --output.
Important manifest details (learned the hard way):
hook_attributes MUST include a nested url field, even if webhooks are disabled. Without it, GitHub rejects the manifest with a misleading "url" wasn't supplied error — it's not about the top-level url.<input type="text"> (not hidden) with the value set via JSON.stringify() in JavaScript.state query parameter must be included on the form action URL.redirect_url must point to the local callback server.Read the credentials JSON from the host and store each field in the assistant's vault. The fields available immediately after Step 1:
assistant credentials set --service github-app --field app_id "APP_ID"
assistant credentials set --service github-app --field app_slug "APP_SLUG"
assistant credentials set --service github-app --field client_id "CLIENT_ID"
assistant credentials set --service github-app --field client_secret "CLIENT_SECRET"
assistant credentials set --service github-app --field webhook_secret "WEBHOOK_SECRET"
For the PEM (private key), the -----BEGIN prefix gets parsed as CLI flags. Use -- to signal end of options:
assistant credentials set --service github-app --field pem -- "$(cat pem-file.txt)"
Transferring the PEM from host to container: The Vellum security layer redacts secrets in transit between host and container. To work around this, base64-encode the PEM on the host, read the base64 string via host_file_read (which won't be redacted), decode it inside the container, and store it with assistant credentials set.
Note: The
installation_idcredential is stored in Step 3 after installing the app.
Open the installation page for the user:
https://github.com/apps/APP_SLUG/installations/select_target
The user selects the org, chooses "Only select repositories", picks the target repos, and clicks Install.
After installation, retrieve the installation ID using the app's JWT auth. Generate a JWT from the stored credentials:
// Use the JWT generation logic from gh-app-token.mjs,
// but call /app/installations instead of /access_tokens
const resp = await fetch("https://api.github.com/app/installations", {
headers: {
Authorization: `Bearer ${jwt}`,
Accept: "application/vnd.github+json",
},
});
const installations = await resp.json();
// installations[0].id is the installation ID
Then store it:
assistant credentials set --service github-app --field installation_id "INSTALLATION_ID"
Copy the token helper to the workspace:
cp scripts/gh-app-token.mjs "$WORKSPACE_ROOT/bin/gh-app-token.mjs"
chmod +x "$WORKSPACE_ROOT/bin/gh-app-token.mjs"
Configure the local repo clone:
cd "$WORKSPACE_ROOT/REPO_NAME"
git config user.name "APP_SLUG[bot]"
git config user.email "APP_ID+APP_SLUG[bot]@users.noreply.github.com"
Before each push, refresh the remote URL with a fresh token (tokens expire after 1 hour):
TOKEN=$(bun "$WORKSPACE_ROOT/bin/gh-app-token.mjs")
git remote set-url origin "https://x-access-token:${TOKEN}@github.com/OWNER/REPO.git"
git push origin BRANCH
The manifest flow does not support setting a logo — there's no field for it and no REST API endpoint. The logo can only be uploaded through the GitHub web UI.
If the assistant has an avatar (typically at data/avatar/avatar-image.png in the workspace root), send it to the user as a chat attachment:
<vellum-attachment source="sandbox" path="data/avatar/avatar-image.png" />
Then direct them to the app settings page to upload it:
https://github.com/organizations/ORG/settings/apps/APP_SLUG
Scroll to "Display information" → "Upload a logo" → drag in the image file → Save.
Test the full flow:
APP_SLUG[bot]Installation tokens expire after 1 hour. The helper script gh-app-token.mjs generates a fresh one each time by:
app_id and pem from the vaultPOST /app/installations/{id}/access_tokensAlways refresh before pushing:
TOKEN=$(bun "$WORKSPACE_ROOT/bin/gh-app-token.mjs")
git remote set-url origin "https://x-access-token:${TOKEN}@github.com/OWNER/REPO.git"
The default permission set covers the full range of actions the assistant may need to perform on a repository:
| Permission | Level | Purpose |
|---|---|---|
contents | write | Push commits, create/delete branches |
pull_requests | write | Open PRs, post PR comments and reviews |
checks | read | Read CI check-run status (e.g. gh pr checks) |
metadata | read | Required by GitHub for all App installations (auto-added) |
All credentials are stored under service: github-app:
| Field | Description | When Set |
|---|---|---|
app_id | Numeric GitHub App ID | Step 2 |
app_slug | URL-friendly app name (e.g. credence-the-bot) | Step 2 |
client_id | OAuth client ID | Step 2 |
client_secret | OAuth client secret | Step 2 |
pem | RSA private key for JWT signing | Step 2 |
webhook_secret | Webhook verification secret | Step 2 |
installation_id | Numeric installation ID for the org | Step 3 |
The manifest JSON must include hook_attributes.url — a nested URL inside the hook_attributes object. GitHub's error message is misleading; it's not referring to the top-level url field.
Check that all three required credentials are set: app_id, pem, installation_id. Use assistant credentials list --search github-app to verify.
The installation token may have expired (1-hour lifetime). Regenerate with bun bin/gh-app-token.mjs (from the workspace root) and update the remote URL.
The -----BEGIN prefix gets parsed as a CLI flag. Use -- before the value:
assistant credentials set --service github-app --field pem -- "$PEM_VALUE"
Another process is using the callback port. Either kill it or pass --port=DIFFERENT_PORT to the creation script.
The manifest flow script currently only supports organization-owned apps (it uses the /organizations/{org}/settings/apps/new endpoint). Personal account apps would use /settings/apps/new instead — this is not yet implemented.