Configuration
Project configuration for Craft is stored in .craft.yml in the project root.
GitHub Project
Section titled “GitHub Project”Craft tries to determine GitHub repo information from the local git repo. You can also hard-code it:
github: owner: getsentry repo: sentry-javascriptPre-release Command
Section titled “Pre-release Command”This command runs on your release branch as part of craft prepare. Default: bash scripts/bump-version.sh.
preReleaseCommand: bash scripts/bump-version.shThe command is executed with the following environment variables:
CRAFT_OLD_VERSION: The previous version (or0.0.0if no previous version exists)CRAFT_NEW_VERSION: The new version being released
The script should:
- Use these environment variables to perform version replacement
- Replace version occurrences
- Not commit changes
- Not change git state
Note: For backward compatibility, the old and new versions are also passed as the last two command-line arguments to the script, but using environment variables is safer and recommended.
Example script:
#!/bin/bashset -eux
# Use CRAFT_NEW_VERSION provided by craftexport npm_config_git_tag_version=falsenpm version "${CRAFT_NEW_VERSION}"Automatic Version Bumping
Section titled “Automatic Version Bumping”When minVersion: "2.21.0" or higher is set and no custom preReleaseCommand is defined, Craft automatically bumps version numbers based on your configured publish targets. This eliminates the need for a scripts/bump-version.sh script in most cases.
How It Works
Section titled “How It Works”- Craft examines your configured
targetsin.craft.yml - For each target that supports version bumping, Craft updates the appropriate project files
- Targets are processed in the order they appear in your configuration
- Each target type is only processed once (e.g., multiple npm targets won’t bump
package.jsontwice)
Supported Targets
Section titled “Supported Targets”| Target | Detection | Version Bump Method |
|---|---|---|
npm | package.json exists | npm version --no-git-tag-version (with workspace support) |
pypi | pyproject.toml exists | hatch, poetry, setuptools-scm, or direct edit |
crates | Cargo.toml exists | cargo set-version (requires cargo-edit) |
gem | *.gemspec exists | Direct edit of gemspec and lib/**/version.rb |
pub-dev | pubspec.yaml exists | Direct edit of pubspec.yaml |
hex | mix.exs exists | Direct edit of mix.exs |
nuget | *.csproj exists | dotnet-setversion or direct XML edit |
npm Workspace Support
Section titled “npm Workspace Support”For npm/yarn/pnpm monorepos, Craft automatically detects and bumps versions in all workspace packages:
- npm 7+: Uses
npm version --workspacesto bump all packages at once - yarn/pnpm or npm < 7: Falls back to bumping each non-private package individually
Workspace detection checks for:
workspacesfield in rootpackage.json(npm/yarn)pnpm-workspace.yaml(pnpm)
Private packages ("private": true) are skipped during workspace version bumping.
Python (pypi) Detection Priority
Section titled “Python (pypi) Detection Priority”For Python projects, Craft detects the build tool and uses the appropriate method:
- Hatch - If
[tool.hatch]section exists →hatch version <version> - Poetry - If
[tool.poetry]section exists →poetry version <version> - setuptools-scm - If
[tool.setuptools_scm]section exists → No-op (version derived from git tags) - Direct edit - If
[project]section withversionfield exists → Editpyproject.tomldirectly
Enabling Automatic Version Bumping
Section titled “Enabling Automatic Version Bumping”To enable automatic version bumping, ensure your .craft.yml has:
minVersion: '2.21.0'targets: - name: npm # or pypi, crates, etc. # ... other targetsAnd either:
- Remove any custom
preReleaseCommand, or - Don’t define
preReleaseCommandat all
Disabling Automatic Version Bumping
Section titled “Disabling Automatic Version Bumping”To disable automatic version bumping while still using minVersion 2.21.0+:
minVersion: '2.21.0'preReleaseCommand: '' # Explicitly set to empty stringOr define a custom script:
minVersion: '2.21.0'preReleaseCommand: bash scripts/my-custom-bump.shError Handling
Section titled “Error Handling”If automatic version bumping fails:
- Missing tool: Craft reports which tool is missing (e.g., “Cannot find ‘npm’ for version bumping”)
- Command failure: Craft shows the error from the failed command
- No supported targets: Craft warns that no targets support automatic bumping
In all error cases, Craft suggests defining a custom preReleaseCommand as a fallback.
Recovery from Failed Prepare
Section titled “Recovery from Failed Prepare”If version bumping succeeds but craft prepare fails mid-way (e.g., during changelog generation or git operations), you may need to clean up manually:
-
Check the release branch: If a release branch was created, you can delete it:
Terminal window git branch -D release/<version> -
Revert version changes: If files were modified but not committed, reset them:
Terminal window git checkout -- package.json pyproject.toml Cargo.toml # or whichever files were changed -
Re-run prepare: Once the issue is fixed, run
craft prepareagain. Version bumping is idempotent—running it multiple times with the same version is safe.
Post-release Command
Section titled “Post-release Command”This command runs after a successful craft publish. Default: bash scripts/post-release.sh.
postReleaseCommand: bash scripts/post-release.shRelease Branch Name
Section titled “Release Branch Name”Override the release branch prefix. Default: release.
releaseBranchPrefix: publishFull branch name: {releaseBranchPrefix}/{version}
Changelog Policies
Section titled “Changelog Policies”Craft supports simple and auto changelog management modes.
Simple Mode
Section titled “Simple Mode”Reminds you to add a changelog entry:
changelog: CHANGESOr with options:
changelog: filePath: CHANGES.md policy: simpleAuto Mode
Section titled “Auto Mode”Automatically generates changelog from commits:
changelog: policy: autoAuto mode uses .github/release.yml to categorize PRs. This file follows GitHub’s release.yml format with Craft-specific extensions.
Craft Extensions to release.yml
Section titled “Craft Extensions to release.yml”Craft extends GitHub’s format with two additional fields:
| Field | Description |
|---|---|
commit_patterns | Array of regex patterns to match commit/PR titles (in addition to labels) |
semver | Version bump type for auto-versioning: major, minor, or patch |
Default Configuration
Section titled “Default Configuration”If .github/release.yml doesn’t exist, Craft uses these defaults based on Conventional Commits:
changelog: exclude: labels: - skip-changelog categories: - title: Breaking Changes 🛠 commit_patterns: - "^(?<type>\\w+(?:\\((?<scope>[^)]+)\\))?!:\\s*)" semver: major - title: New Features ✨ commit_patterns: - "^(?<type>feat(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" semver: minor - title: Bug Fixes 🐛 commit_patterns: - "^(?<type>fix(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" - '^Revert "' semver: patch - title: Documentation 📚 commit_patterns: - "^(?<type>docs?(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" semver: patch - title: Internal Changes 🔧 commit_patterns: - "^(?<type>(?:build|refactor|meta|chore|ci|ref|perf)(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" semver: patchExample Configuration
Section titled “Example Configuration”changelog: categories: - title: Features labels: - enhancement commit_patterns: - "^(?<type>feat(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" semver: minor - title: Bug Fixes labels: - bug commit_patterns: - "^(?<type>fix(?:\\((?<scope>[^)]+)\\))?!?:\\s*)" semver: patchCustom Changelog Entries from PR Descriptions
Section titled “Custom Changelog Entries from PR Descriptions”By default, the changelog entry for a PR is generated from its title. However, PR authors can override this by adding a “Changelog Entry” section to the PR description. This allows for more detailed, user-facing changelog entries without cluttering the PR title.
To use this feature, add a markdown heading (level 2 or 3) titled “Changelog Entry” to your PR description, followed by the desired changelog text:
### Description
Add `foo` function, and add unit tests to thoroughly check all edge cases.
### Changelog Entry
Add a new function called `foo` which prints "Hello, world!"
### Issues
Closes #123The text under “Changelog Entry” will be used verbatim in the changelog instead of the PR title. If no such section is present, the PR title is used as usual.
Advanced Features
Section titled “Advanced Features”-
Multiple Entries: If you use multiple top-level bullet points in the “Changelog Entry” section, each bullet will become a separate changelog entry:
### Changelog Entry- Add OAuth2 authentication- Add two-factor authentication- Add session management -
Nested Content: Indented bullets (4+ spaces or tabs) are preserved as nested content under their parent entry:
### Changelog Entry- Add authentication system- OAuth2 support- Two-factor authentication- Session managementThis will generate:
- Add authentication system by @user in [#123](url)- OAuth2 support- Two-factor authentication- Session managementNote: Nested items do NOT get author/PR attribution - only the top-level entry does.
-
Plain Text: If no bullets are used, the entire content is treated as a single changelog entry. Multi-line text is automatically joined with spaces to ensure valid markdown output.
-
Content Isolation: Only content within the “Changelog Entry” section is included in the changelog. Other sections (Description, Issues, etc.) are ignored.
Scope Grouping
Section titled “Scope Grouping”Changes are automatically grouped by scope (e.g., feat(api): groups under “Api”):
changelog: policy: auto scopeGrouping: true # defaultScope headers are only shown for scopes with more than one entry. Entries without a scope are listed at the bottom of each category section without a sub-header.
Example output with scope grouping:
### New Features
#### Api
- Add user endpoint by @alice in [#1](https://github.com/...)- Add auth endpoint by @bob in [#2](https://github.com/...)
#### Ui
- Add dashboard by @charlie in [#3](https://github.com/...)
- General improvement by @dave in [#4](https://github.com/...)Title Stripping (Default Behavior)
Section titled “Title Stripping (Default Behavior)”By default, conventional commit prefixes are stripped from changelog entries.
The type (e.g., feat:) is removed, and the scope is preserved when entries
aren’t grouped under a scope header.
This behavior is controlled by named capture groups in commit_patterns:
(?<type>...)- The type prefix to strip (includes type, scope, and colon)(?<scope>...)- Scope to preserve when not under a scope header
| Original Title | Scope Header | Displayed Title |
|---|---|---|
feat(api): add endpoint | Yes (Api) | Add endpoint |
feat(api): add endpoint | No | (api) Add endpoint |
feat: add endpoint | N/A | Add endpoint |
To disable stripping, provide custom patterns using non-capturing groups:
commit_patterns: - "^feat(?:\\([^)]+\\))?!?:" # No named groups = no strippingSkipping Changelog Entries
Section titled “Skipping Changelog Entries”You can exclude PRs or commits from the changelog in several ways:
Magic Word
Section titled “Magic Word”Add #skip-changelog anywhere in your commit message or PR body:
chore: Update dependencies
#skip-changelogSkip Label
Section titled “Skip Label”PRs with the skip-changelog label are automatically excluded.
Configuration
Section titled “Configuration”Configure exclusions in .github/release.yml:
changelog: exclude: labels: - skip-changelog - dependencies authors: - dependabot[bot] - renovate[bot]Configuration Options
Section titled “Configuration Options”| Option | Description |
|---|---|
changelog | Path to changelog file (string) OR configuration object |
changelog.filePath | Path to changelog file. Default: CHANGELOG.md |
changelog.policy | Mode: none, simple, or auto. Default: none |
changelog.scopeGrouping | Enable scope-based grouping. Default: true |
Versioning
Section titled “Versioning”Configure default versioning behavior:
versioning: policy: auto # auto, manual, or calverVersioning Policies
Section titled “Versioning Policies”| Policy | Description |
|---|---|
auto | Analyze commits to determine version bump (default when using craft prepare auto) |
manual | Require explicit version argument |
calver | Use calendar-based versioning |
Calendar Versioning (CalVer)
Section titled “Calendar Versioning (CalVer)”For projects using calendar-based versions:
versioning: policy: calver calver: format: '%y.%-m' # e.g., 24.12 for December 2024 offset: 14 # Days to look back for date calculationFormat supports:
%y- 2-digit year%m- Zero-padded month%-m- Month without padding
Minimal Version
Section titled “Minimal Version”Require a minimum Craft version:
minVersion: '0.5.0'Smart Defaults
Section titled “Smart Defaults”Setting minVersion to 2.21.0 or higher enables smart defaults that simplify configuration:
minVersion: '2.21.0'| Feature | Default with minVersion >= 2.21.0 | Default without |
|---|---|---|
changelog.policy | auto | none |
versioning.policy | auto (with >= 2.14.0) | manual |
This means a minimal configuration like this:
minVersion: '2.21.0'targets: - name: npm - name: githubWill automatically:
- Generate changelogs from conventional commits
- Determine version bumps from commit analysis
- Create
CHANGELOG.mdif it doesn’t exist
Required Files
Section titled “Required Files”Ensure specific artifacts exist before publishing:
requireNames: - /^sentry-craft.*\.tgz$/ - /^gh-pages.zip$/Status Provider
Section titled “Status Provider”Configure build status checks:
statusProvider: name: github config: contexts: - Travis CI - BranchArtifact Provider
Section titled “Artifact Provider”Configure where to fetch artifacts from:
artifactProvider: name: github # or 'gcs' or 'none'GitHub Artifact Provider Configuration
Section titled “GitHub Artifact Provider Configuration”By default, the GitHub artifact provider looks for artifacts named exactly as the commit SHA. You can customize this with the artifacts configuration option.
Pattern Syntax
Section titled “Pattern Syntax”- Regex patterns: Wrapped in
/(e.g.,/^build-.*$/) - Exact strings: Plain text (e.g.,
build,release-artifacts)
Configuration Formats
Section titled “Configuration Formats”1. Single artifact pattern - searches all workflows:
artifactProvider: name: github config: artifacts: /^sentry-.*\.tgz$/2. Multiple artifact patterns - searches all workflows:
artifactProvider: name: github config: artifacts: - /^sentry-.*\.tgz$/ - release-bundle3. Workflow-scoped patterns - filter by workflow name:
artifactProvider: name: github config: artifacts: build: release-artifacts # exact workflow → exact artifact /^build-.*$/: artifacts # workflow pattern → exact artifact ci: # exact workflow → multiple artifacts - /^output-.*$/ - bundle /^release-.*$/: # workflow pattern → multiple artifacts - /^dist-.*$/ - checksumsCommon Examples
Section titled “Common Examples”Fetch artifacts named craft-binary and craft-docs from the “Build & Test” workflow:
artifactProvider: name: github config: artifacts: Build & Test: - craft-binary - craft-docsFetch all .tgz files from any workflow:
artifactProvider: name: github config: artifacts: /\.tgz$/Backward Compatibility
Section titled “Backward Compatibility”When artifacts is not configured, the provider uses the legacy behavior where it searches for an artifact with a name matching the commit SHA exactly. This ensures existing configurations continue to work without changes.
Targets
Section titled “Targets”List release targets in your configuration:
targets: - name: npm - name: github - name: registry id: browser type: sdk onlyIfPresent: /^sentry-browser-.*\.tgz$/See Target Configurations for details on each target.
Per-target Options
Section titled “Per-target Options”These options apply to all targets:
| Option | Description |
|---|---|
includeNames | Regex: only matched files are processed |
excludeNames | Regex: matched files are skipped |
id | Unique ID for the target (use with -t target[id]) |
Example:
targets: - name: github includeNames: /^.*\.exe$/ excludeNames: /^test.exe$/